美文网首页Android知识Android开发经验谈Android开发
【Android源码】BroadcastReceiver的工作过

【Android源码】BroadcastReceiver的工作过

作者: 指间沙似流年 | 来源:发表于2017-07-05 09:30 被阅读137次

    BroadcastReceiver的使用

    通常情况下,我们使用广播的方式,首先定义广播接收者,继承BroadcastReceiver并重写onReceive方法:

    public class MyReceiver extends BroadcastReceiver{
        @Override
        public void onReceive(Context context, Intent intent){
            Log.d("TAG", "on receive action = " + intent.getAction());
        }
    }
    

    定义好广播之后,就可以注册广播接收者了。
    有两种方式:静态注册和动态注册。

    静态注册:

    <receiver android:name=".MyReceiver">
        <intent-filter>
            <action android:name="com.fastaoe.receiver.LAUNCH"/>
        </intent-filter>
    </receiver>
    

    动态注册:

    IntentFilter filter = new IntentFilter();
    filter.addAction("com.fastaoe.receiver.LAUNCH");
    registerReceiver(new MyReceiver(), filter);
    

    当注册完成之后就可以通过send来发送广播了:

    Intent intent = new Intent();
    intent.setAction("com.fastaoe.register.LAUNCH");
    sendBroadcast(intent);
    

    BroadcastReceiver的注册过程

    静态注册的过程其实就是PackageManagerService解析的过程,其实四大组件都是有PMS来解析并注册的可以参考【Android源码】PackageManagerService 浅析

    我们现在只分析动态注册的过程:

    同样的动态注册和Activity和Service一样都是在ContextWrapper中,而其实mBase的具体实现类是ContextImpl对象:

    // ContextWrapper.java
    @Override
    public Intent registerReceiver(
       BroadcastReceiver receiver, IntentFilter filter) {
       return mBase.registerReceiver(receiver, filter);
    }
    
    // ContextImpl.java
    @Override
    public Intent registerReceiver(BroadcastReceiver receiver, IntentFilter filter) {
       return registerReceiver(receiver, filter, null, null);
    }
    
    @Override
    public Intent registerReceiver(BroadcastReceiver receiver, IntentFilter filter,
           String broadcastPermission, Handler scheduler) {
       return registerReceiverInternal(receiver, getUserId(),
               filter, broadcastPermission, scheduler, getOuterContext());
    }
    
    private Intent registerReceiverInternal(BroadcastReceiver receiver, int userId,
           IntentFilter filter, String broadcastPermission,
           Handler scheduler, Context context) {
       IIntentReceiver rd = null;
       if (receiver != null) {
           if (mPackageInfo != null && context != null) {
               if (scheduler == null) {
                   scheduler = mMainThread.getHandler();
               }
               rd = mPackageInfo.getReceiverDispatcher(
                   receiver, context, scheduler,
                   mMainThread.getInstrumentation(), true);
           } else {
               if (scheduler == null) {
                   scheduler = mMainThread.getHandler();
               }
               rd = new LoadedApk.ReceiverDispatcher(
                       receiver, context, scheduler, null, true).getIIntentReceiver();
           }
       }
       try {
           final Intent intent = ActivityManagerNative.getDefault().registerReceiver(
                   mMainThread.getApplicationThread(), mBasePackageName,
                   rd, filter, broadcastPermission, userId);
           if (intent != null) {
               intent.setExtrasClassLoader(getClassLoader());
               intent.prepareToEnterProcess();
           }
           return intent;
       } catch (RemoteException e) {
           throw e.rethrowFromSystemServer();
       }
    }
    

    上述代码主要做了这样几件事:

    1. mPackageInfo中获取IIntentReceiver对象。

      因为广播是可以跨进程通信的,所以不能直接使用BroadcastReceiver,而是使用InnerReceiver extends IIntentReceiver.Stub的类型,也就是Binder接口,这个其实和Service的绑定流程类似。

    2. 通过AMS注册广播。

    // ActivityManagerService.java
    public Intent registerReceiver(IApplicationThread caller, String callerPackage,
                IIntentReceiver receiver, IntentFilter filter, String permission, int userId) {
        mRegisteredReceivers.put(receiver.asBinder(), rl);
        
        BroadcastFilter bf = new BroadcastFilter(filter, rl, callerPackage,
                        permission, callingUid, userId);
        rl.add(bf);     
    }  
    

    通过registerReceiver方法将IIntentReceiver和IntentFilter保存起来,这个时候广播就被注册好了。

    BroadcastReceiver的发送和接收过程

    同样的sendBroadcast也是由ContextImpl来实现的:

    @Override
    public void sendBroadcast(Intent intent) {
       warnIfCallingFromSystemProcess();
       String resolvedType = intent.resolveTypeIfNeeded(getContentResolver());
       try {
           intent.prepareToLeaveProcess(this);
           ActivityManagerNative.getDefault().broadcastIntent(
                   mMainThread.getApplicationThread(), intent, resolvedType, null,
                   Activity.RESULT_OK, null, null, null, AppOpsManager.OP_NONE, null, false, false,
                   getUserId());
       } catch (RemoteException e) {
           throw e.rethrowFromSystemServer();
       }
    }
    

    上述代码什么都没做,只是使用AMS调用broadcastIntent

    public final int broadcastIntent(IApplicationThread caller,
           Intent intent, String resolvedType, IIntentReceiver resultTo,
           int resultCode, String resultData, Bundle resultExtras,
           String[] requiredPermissions, int appOp, Bundle bOptions,
           boolean serialized, boolean sticky, int userId) {
       enforceNotIsolatedCaller("broadcastIntent");
       synchronized(this) {
           intent = verifyBroadcastLocked(intent);
    
           final ProcessRecord callerApp = getRecordForAppLocked(caller);
           final int callingPid = Binder.getCallingPid();
           final int callingUid = Binder.getCallingUid();
           final long origId = Binder.clearCallingIdentity();
           int res = broadcastIntentLocked(callerApp,
                   callerApp != null ? callerApp.info.packageName : null,
                   intent, resolvedType, resultTo, resultCode, resultData, resultExtras,
                   requiredPermissions, appOp, bOptions, serialized, sticky,
                   callingPid, callingUid, userId);
           Binder.restoreCallingIdentity(origId);
           return res;
       }
    }
    

    从上面的代码可以看到是调用broadcastIntentLocked方法:

    intent.addFlags(Intent.FLAG_EXCLUDE_STOPPED_PACKAGES);
    

    在开始的时候添加了一个特殊的标记,这个标记表明默认情况下,广播不会发送给已经停止的应用。

    if ((receivers != null && receivers.size() > 0)
          || resultTo != null) {
      BroadcastQueue queue = broadcastQueueForIntent(intent);
      BroadcastRecord r = new BroadcastRecord(queue, intent, callerApp,
              callerPackage, callingPid, callingUid, resolvedType,
              requiredPermissions, appOp, brOptions, receivers, resultTo, resultCode,
              resultData, resultExtras, ordered, sticky, false, userId);
    
      if (DEBUG_BROADCAST) Slog.v(TAG_BROADCAST, "Enqueueing ordered broadcast " + r
              + ": prev had " + queue.mOrderedBroadcasts.size());
      if (DEBUG_BROADCAST) Slog.i(TAG_BROADCAST,
              "Enqueueing broadcast " + r.intent.getAction());
    
      boolean replaced = replacePending && queue.replaceOrderedBroadcastLocked(r);
      if (!replaced) {
          queue.enqueueOrderedBroadcastLocked(r);
          queue.scheduleBroadcastsLocked();
      }
    }
    

    之后broadcastIntentLocked内部,会根据intent-filter来过滤匹配所有的广播接收者,最终满足所有条件的广播接收者会被添加到BroadcastQueue中,之后BroadcastQueue会通过scheduleBroadcastsLocked将广播发送给这些符合条件的广播接收者。

    public void scheduleBroadcastsLocked() {
       if (DEBUG_BROADCAST) Slog.v(TAG_BROADCAST, "Schedule broadcasts ["
               + mQueueName + "]: current="
               + mBroadcastsScheduled);
    
       if (mBroadcastsScheduled) {
           return;
       }
       mHandler.sendMessage(mHandler.obtainMessage(BROADCAST_INTENT_MSG, this));
       mBroadcastsScheduled = true;
    }
    

    scheduleBroadcastsLocked,系统并没有直接发送广播,而是通过handler来发送消息给BroadcastHandler,而BroadcastHandler在接收到BROADCAST_INTENT_MSG之后调用了processNextBroadcast

    while (mParallelBroadcasts.size() > 0) {
         r = mParallelBroadcasts.remove(0);
         r.dispatchTime = SystemClock.uptimeMillis();
         r.dispatchClockTime = System.currentTimeMillis();
         final int N = r.receivers.size();
         if (DEBUG_BROADCAST_LIGHT) Slog.v(TAG_BROADCAST, "Processing parallel broadcast ["
                 + mQueueName + "] " + r);
         for (int i=0; i<N; i++) {
             Object target = r.receivers.get(i);
             if (DEBUG_BROADCAST)  Slog.v(TAG_BROADCAST,
                     "Delivering non-ordered on [" + mQueueName + "] to registered "
                     + target + ": " + r);
             deliverToRegisteredReceiverLocked(r, (BroadcastFilter)target, false, i);
         }
         addBroadcastToHistoryLocked(r);
         if (DEBUG_BROADCAST_LIGHT) Slog.v(TAG_BROADCAST, "Done with parallel broadcast ["
                 + mQueueName + "] " + r);
    }
    

    通过循环遍历mParallelBroadcasts并将广播发送给接收者,就是通过deliverToRegisteredReceiverLocked来完成的:

    performReceiveLocked(filter.receiverList.app, filter.receiverList.receiver,
                            new Intent(r.intent), r.resultCode, r.resultData,
                            r.resultExtras, r.ordered, r.initialSticky, r.userId);
                            
    void performReceiveLocked(ProcessRecord app, IIntentReceiver receiver,
                Intent intent, int resultCode, String data, Bundle extras,
                boolean ordered, boolean sticky, int sendingUser) throws RemoteException {
        app.thread.scheduleRegisteredReceiver(receiver, intent, resultCode,
                                data, extras, ordered, sticky, sendingUser, app.repProcState);           
    }
    

    最终调用了ApplicationThread的scheduleRegisteredReceiver来完成广播的接收:

    public void scheduleRegisteredReceiver(IIntentReceiver receiver, Intent intent,
          int resultCode, String dataStr, Bundle extras, boolean ordered,
          boolean sticky, int sendingUser, int processState) throws RemoteException {
      updateProcessState(processState, false);
      receiver.performReceive(intent, resultCode, dataStr, extras, ordered,
              sticky, sendingUser);
    }
    

    receiver其实就是之前我们所讲的InnerReceiver

    @Override
    public void performReceive(Intent intent, int resultCode, String data,
        Bundle extras, boolean ordered, boolean sticky, int sendingUser) {
        rd.performReceive(intent, resultCode, data, extras,
                                ordered, sticky, sendingUser);   
    }
    
    public void performReceive(Intent intent, int resultCode, String data,
          Bundle extras, boolean ordered, boolean sticky, int sendingUser) {
      final Args args = new Args(intent, resultCode, data, extras, ordered,
              sticky, sendingUser);
      if (intent == null) {
          Log.wtf(TAG, "Null intent received");
      } else {
          if (ActivityThread.DEBUG_BROADCAST) {
              int seq = intent.getIntExtra("seq", -1);
              Slog.i(ActivityThread.TAG, "Enqueueing broadcast " + intent.getAction()
                      + " seq=" + seq + " to " + mReceiver);
          }
      }
      if (intent == null || !mActivityThread.post(args)) {
          if (mRegistered && ordered) {
              IActivityManager mgr = ActivityManagerNative.getDefault();
              if (ActivityThread.DEBUG_BROADCAST) Slog.i(ActivityThread.TAG,
                      "Finishing sync broadcast to " + mReceiver);
              args.sendFinished(mgr);
          }
      }
    }
    

    这里有一段关键代码mActivityThread.post(args),其中args是Args的实例,而Args实现了Runnable接口,mActivityThread是ActivityThread中的mH的Handler对象,在Args的run方法中:

    ClassLoader cl =  mReceiver.getClass().getClassLoader();
    intent.setExtrasClassLoader(cl);
    intent.prepareToEnterProcess();
    setExtrasClassLoader(cl);
    receiver.setPendingResult(this);
    receiver.onReceive(mContext, intent);
    

    BroadcastReceiver的onReceive就被执行了,这个使用app也就接收到了广播。

    相关文章

      网友评论

        本文标题:【Android源码】BroadcastReceiver的工作过

        本文链接:https://www.haomeiwen.com/subject/kgjacxtx.html