【Android源码】BroadcastReceiver的工作过程

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也就接收到了广播。

    原文作者:指间沙似流年
    原文地址: https://www.jianshu.com/p/78c1682a8066
    本文转自网络文章,转载此文章仅为分享知识,如有侵权,请联系博主进行删除。
点赞