android开发常见编程错误总结
1.设置TextView的文本颜色
1
2
3
|
TextView tv; ... tv.setTextColor(R.color.white); |
其实这样设置的颜色是 R.color.white的资源ID值所代表的颜色值,而不是资源color下的white颜色值:正确的做法如下:
1
|
tv.setTextColor(getResources().getColor(R.color.white)); |
这个出错的概率满高的,就是因为二者都是int类,导致编译器不报错。
2.读取Cursor中的值
1
2
3
4
5
6
7
|
Uri uri; Cursor cursor = contentResolver.query(uri, null , null , null , null ); if (cursor != null ){ String name = cursor.getString(1); // curosr.close(); cursor = null ; } |
上面语句中的,执行到cursor.getString(1)部分就会报异常,异常是: Caused by: android.database.CursorIndexOutOfBoundsException: Index -1 requested, with a size of 4
编译没有问题,只有在运行的时候才会发现。
正确的做法是:
1
2
3
4
5
6
7
8
9
|
Uri uri; Cursor cursor = contentResolver.query(uri, null , null , null , null ); if (cursor != null ){ if (cursor.moveToFirst()){ String name = cursor.getString(1); // } curosr.close(); cursor = null ; } |
或者:
1
2
3
4
5
6
7
8
9
|
Uri uri; Cursor cursor = contentResolver.query(uri, null , null , null , null ); if (cursor != null ){ while (cursor.moveToNext()){ String name = cursor.getString(1); // } curosr.close(); cursor = null ; } |
3.不要使用标有Deprecated的函数或者类,比如不要使用android.telephony.gsm.SmsMessage,而应该用android.telephony.SmsMessage,这样避免采用不同的3G协议时不会出现问题。
4.SQLite中的查询条件,比如一个叫name的字段,其字段类型为TEXT,如果我们要判断其name不等某个值(如zhangsan),写出如下的语句
1
|
name <> 'zhangsan' |
但是,这样写的语句,如果碰到name值为空的时候,就有问题,即name为空时 以上的布尔值为false,而不是true.
原因很可能,SQLite中的判断函数采用类似写法:
1
2
3
4
|
boolean judge(String self, String conditions){ if ( null == self) return false ; return self.equalsIgnoreCase(conditions); } |
其中 self为数据库中name的值,而conditions为上面示例中的 zhangsan。
所以,以上查询条件的正确写法是:
1
|
name <> 'zhangsan' or name is null |
除非你也想过滤掉name 为空的记录。
5.如下所示,想要在按钮显示"删 除"(没错删除中间有个空格),以下的字符串资源是错误的:
1
|
<string name= "button_delete_text" >删 除</string> |
这样的出来,最终看不到中间的空格,应该是Android SDK编译的时候,会自动过滤掉String中的空格部分,所以应该采用以下的方式:
1
|
<string name= "button_delete_text" >删\u0020除</string> |
类似地,其他的特殊符号都可以用\u00XX来转义,如 ' ---- \u0027, < ----- \u003C, > ---- \u003E 。
注意这里的数字是16进制哦。
还有一种方法是:这个应该是XML经常使用的方法(new 2013.03.28)
'
<
>
别忘了数字后面的分号哦,而且其中的数字是十进制的
6. context的问题:
如果在一个非Activity的context里面调用startActivity,那么其intent必须设置:
1
|
intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK); |
否则,会报如下类似的错误:
1
|
Calling startActivity() from outside of an Activity context requires the FLAG_ACTIVITY_NEW_TASK flag. |
而我们还要提防系统控件中的隐性调用startActivity:
1
2
3
|
TextView tv = new TextView(mContext); tv.setAutoLinkMask(Linkify.ALL); <br>tv.setText(content); |
当content内容中有电话号码/邮件/URL时,并且mContext不是非Acitvity的context,而是app的context时(XXXActivity.this.getApplicationContext()),
就会出现如下的错误:
1
2
3
4
5
|
android.util.AndroidRuntimeException: Calling startActivity() from outside of an Activity context requires the FLAG_ACTIVITY_NEW_TASK flag. Is this really what you want? E/AndroidRuntime(10382): at android.app.ContextImpl.startActivity(ContextImpl.java:622) E/AndroidRuntime(10382): at android.content.ContextWrapper.startActivity(ContextWrapper.java:258) E/AndroidRuntime(10382): at android.text.style.URLSpan.onClick(URLSpan.java:62) |
由于URLSpan.onClick中调用startActivity是由系统控制的,所以我们必须传入activity的contex,才不会出现如上的异常,导致程序退出。
7. 另外一个context的问题:如果你在一个单实例的对象中,有个注册监听器的行为的话,那么传给这个单实例
对象的context,就必须是ApplicationContext了,否则会出现:receiver leak的错误。
8. 控件有时不能充满整个屏幕:
1
2
3
4
5
6
|
LinearLayout panel = new LinearLayout( this ); LinearLayout.LayoutParams llp = new LinearLayout.LayoutParams( LinearLayout.LayoutParams.FILL_PARENT, LinearLayout.LayoutParams.FILL_PARENT); panel.setLayoutParams(llp); root.addView(panel); |
而应该是:
1
2
3
4
5
|
LinearLayout panel = new LinearLayout( this ); LinearLayout.LayoutParams llp = new LinearLayout.LayoutParams( LinearLayout.LayoutParams.FILL_PARENT, LinearLayout.LayoutParams.FILL_PARENT); root.addView(panel. llp); |
9.按照以下的方式启动service,但是service没有起来
1
2
|
Intent service = new Intent( this , FuncService.class); startService(service); |
很有可能是忘了在AndroidManifest.xml中注册FuncService
1
|
<service android:name= "com.android.example.FuncService" /> |
10.TextView中为什么会在有些行尾出现"..."字符,当然不是所有手机都是有问题,本来笔者刚开始也以为可能是
手机的ROM问题,认真review了代码,发现如下的代码:
1
2
|
mIntroView = (TextView) findViewById(R.id.description); mIntroView.setEllipsize(TruncateAt.END); |
问题是上面的第2行,之前是因为要限定文本的行数,后来去掉限制,没有去掉以上的代码。
该行代码会导致很多的ROM上:只要一个文本行的文字在一个手机屏幕行显示不下的话,就自动在
行尾截断并在行尾添加"...",而之前没有问题是因为:全部显示的时候,我调用了如下方法:
1
|
mIntro.setMaxLines(Integer.MAX_VALUE); |
11.不要太相信工具,比如Eclipse里面的断点遇到多线程什么,经常不起作用/走不到,还有就是如果语句为空的也不会走,这时候别太早下结论断点地方出错了,
所以每个工程都应该有日志的开关,通过查看日志来确认,某个路径是否走到或者某个变量的值,。。。
12.Java中的月份是从0开始的,所以格式化月份的时候,记得在原有的值上加1处理,如
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
|
Calendar calendar = Calendar.getInstance(); if (!TextUtils.isEmpty(dateTimes)){ long milliseconds = WLDateUtils.parseDayTime(dateTimes); calendar.setTimeInMillis(milliseconds); } final int old_year = calendar.get(Calendar.YEAR); final int old_month = calendar.get(Calendar.MONTH); final int old_day = calendar.get(Calendar.DAY_OF_MONTH); mDatePickerDialog = new DatePickerDialog( this , new OnDateSetListener(){ @Override public void onDateSet(DatePicker view, int year, int monthOfYear, int dayOfMonth) { if (year != old_year || monthOfYear != old_month || dayOfMonth != old_day){ String dateTimes = String.format( "%04d-%02d-%02d" , year, monthOfYear + 1, dayOfMonth); //月份是从0开始的 } } }, old_year, old_month, old_day); |
13.设置ListView的分割线,如果不是图片的话,应注意顺序:
1
2
3
4
5
6
7
|
mListView = new ListView( this ); mListView.setCacheColorHint(0); mListView.setBackgroundDrawable( null ); mListView.setDivider(getResources().getDrawable(R.drawable.list_divider)); mListView.setDividerHeight(2); 其中: <drawable name= "list_divider" > #00CCCC00</drawable> |
即 setDividerHeight 函数应该在setDivider之后,否则这个分割线无效