安卓编程技巧总结(2) 基础组件开发


1.intent传值

Activity 间的数据通信,对于数据量比较大的,避免使用 Intent + Parcelable
的方式,可以使用 EventBus、RxJava 等替代,以免造成 TransactionTooLargeException。
  1. ActivityNotFoundException
Activity 间通过隐式 Intent 的跳转,在发出 Intent 之前尽可能通过 resolveActivity
检查,避免找不到合适的调用组件,造成 ActivityNotFoundException 的异常。

正确做法:
Intent intent = new Intent(“Action”);
intent.setDataAndType("Data", "Type");
if (getPackageManager().resolveActivity(intent, PackageManager.MATCH_DEFAULT_ONLY) != null) {
try {
    startActivity(intent);
} catch (ActivityNotFoundException e) {
}
  1. 避免主线程执行耗时操作
在主线程执行耗时操作,会导致 UI 不流畅或ANR。下面情况应多加注意:

避免在 BroadcastReceiver#onReceive()中执行耗时操作,如果有耗时工作,
应该创建 IntentService 完成,而不应该在 BroadcastReceiver 内创建子线程去做。
@Override
public void onReceive(Context context, Intent intent) {
  Intent intent = new Intent();
  intent .setClass(this, TestService.class);
  context .startService(intent );
}

public class TestService extends IntentService {
......
@Override
protected void onHandleIntent(Intent intent) {
synchronized (this) {
try {
......
} catch (Exception e) {
}

避免在 Service.onStartCommand()/onBind()方法中执行耗时操作,如果确实有需求,应改用 IntentService 或采用其他异步机制完成。

  1. 广播的效率与安全处理
避免使用隐式 Intent 广播敏感信息,信息可能被其他注册了对应
BroadcastReceiver 的 App 接收。

通过 Context.sendBroadcast()发送的隐式广播会被所有感兴趣的 receiver 接收,恶意应用注册监听该广播的 receiver 可能会获取到 Intent 中传递的敏感信息,并进行其他危险操作。

通过 Context.sendOrderedBroadcast()方法发送的有序广播,优先级较高的恶意 receiver 可能直接丢弃该广播,造成服务不可用,或者向广播结果塞入恶意数据。

如果广播仅限于应用内,应该使用 LocalBroadcastManager.sendBroadcast()实
现,避免敏感信息外泄和 Intent 拦截的风险,也提高了广播接收的效率。

import android.support.v4.content.LocalBroadcastManager;
/**
 * 类:MessageTip 消息提醒类
 * 了解更多,请参考《消息系统实现方案》
 * 作者: qxc
 * 日期:2017/9/1.
 */
public class MessageTip {
    /**
     * 消息提醒类实例
     */
    private static MessageTip messageTip;

    /**
     * 本地广播数据类型实例
     */
    private LocalBroadcastManager localBroadcastManager;   

    ......

    /**
     * 检查消息
     * @param context 上下文
     */
    public void checkMessage(final Context context){
        //获得本地广播实例
        if(localBroadcastManager==null) {
            localBroadcastManager = LocalBroadcastManager.getInstance(context);
        }

        //定时刷新数据
        ......
    }

    /**
     * 获得消息提醒数量并发送本地广播
     */
    public void getMessageTipNum(Context context){
        //判断是否显示消息提示数量
        if(AppTools.getCurrUser().getUid()==0 || !(NetWorkUtils.getInstance().isNetworkAvailable(context))){
            //发送本地广播
            if(localBroadcastManager != null) {
                Intent intent  = new Intent();
                intent.setAction(ACTIONMESSAGETIP);
                intent.putExtra(KEYISSHOWTIPNUM, isShowTipNum);
                localBroadcastManager.sendBroadcast(intent);
            }
            return;
        }
        //获得用户消息数量
        getUserMessageNum(context);
    }
  ......

Activity或者Fragment中动态注册BroadCastReceiver时,registerReceiver()和 unregisterReceiver()要成对出现!!!
  1. Activity释放资源的时机
不要在 Activity#onDestroy()内执行释放资源的工作,例如一些工作线程的
销毁和停止,因为 onDestroy()执行的时机可能较晚。可根据实际需要,在
Activity#onPause()/onStop()中结合 isFinishing()的判断来执行。

6.Fragment

如非必须,避免使用嵌套的 Fragment。Fragment 嵌套使用会有一些坑,容易出现 bug。
onActivityResult()方法的处理错乱;
内嵌的 Fragment 可能收不到该方法的回调,需要由宿主 Fragment 进行转发处理;
被继承的 setRetainInstance(),导致在 Fragment 重建时多次触发不必要的逻辑。

添加Fragment 时,确保 FragmentTransaction#commit() 在Activity#onPostResume()或者 FragmentActivity#onResumeFragments()内调用。
Activity可能因为各种原因被销毁,Android支持页面被销毁前通过
Activity#onSaveInstanceState() 保 存 自 己 的 状 态 。 但 如 果FragmentTransaction.commit()发生在 Activity 状态保存之后,就会导致 Activity 重建、恢复状态时无法还原页面状态,从而可能出错。
为了避免给用户造成不好的体验,系统会抛出 IllegalStateExceptionStateLoss 异常。
推荐的做法是在 Activity 的onPostResume() 或 onResumeFragments() ( 对 FragmentActivity ) 里
执行FragmentTransaction.commit(),如有必要也可在 onCreate()里执行。

7.显示调用service等组件

用显式 Intent 启动或者绑定 Service,且不要为服务声明 Intent Filter,保证应用的安全性。
如果确实需要使用隐式调用,则可为 Service 提供 Intent Filter并从 Intent 中排除相应的组件名称,
但必须搭配使用 Intent#setPackage()方法设置Intent 的指定包名,这样可以充分消除目标服务的不确定性。

Service 需要以多线程来并发处理多个启动请求,建议使用 IntentService;

8.Activity效率

当前Activity的onPause方法执行结束后才会执行下一个Activity的onCreate方法,
所以在 onPause 方法中不适合做耗时较长的工作,这会影响到页面之间的跳转效率。
  1. Adapter 滑动错乱
使用 Adapter 的时候,如果使用了 ViewHolder 做缓存,在 getView()的方法中无论这项 convertView 的
每个子控件是否需要设置属性(比如某个 TextView设置的文本可能为 null,某个按钮的背景色为透明,某控件的颜色为透明等),
都需要为其显式设置属性(Textview 的文本为空也需要设置 setText(""),背景透明也需要设置),
否则在滑动的过程中,因为 adapter item 复用的原因,会出现内容的显示错乱。

10.日志输出

不要使用 System.out.println 打印,尽量使用 log,因为log更可控一些:
log可分等级输出
log可过滤
log可检索

注意:log的tag不能为空,即“”,因为日志的 tag 是空字符串没有任何意义,不利于过滤日志。
......

第一篇: 安卓编程技巧总结(1) 资源与UI布局处理

https://www.jianshu.com/p/ff97b15d5c9d

第二篇: 安卓编程技巧总结(2) 基础组件开发

https://www.jianshu.com/p/b05752377887

第三篇:安卓编程技巧总结(3) 进程与线程处理

https://www.jianshu.com/p/7d05c8a368bd

第四篇:安卓编程技巧总结(4) 数据文件处理

https://www.jianshu.com/p/0515df3b697d

第五篇:安卓编程技巧总结(5) 图片处理

https://www.jianshu.com/p/76690b2ba310

第六篇:安卓编程技巧总结(6) APP安全分析

https://www.jianshu.com/p/4347ff392122

第七篇:安卓编程技巧总结(7) 性能检测代码分析

https://www.jianshu.com/p/687f3c641408

你可能感兴趣的:(安卓编程技巧总结(2) 基础组件开发)