Android广播机制实现源码浅析(一)

http://blog.csdn.net/hehui1860/article/details/30726609


Android系统的广播机制应用非常的广泛,是一种方便快捷的进程间通信的方式。同时它也有一些很有特殊的使用方式,比如它的两种注册方式,三种类型的广播等,这些充斥在整个系统框架中,那么为了用好广播,很有必要对其源码进行分析,从而避免一些低级失误。

本文将对整个广播机制涉及到的知识做个粗略的讲解,为大家自学抛砖引玉。

首先我们从注册当时入手,牵引出与本机制有关的一些类:Packagemanager
在系统服务启动时添加PackageManagerService,在这个过程中packagemanager就会对各个app安装目录的apk文件进行扫描解析。

[html]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. public PackageManagerService(Context context, Installer installer,  
  2.             boolean factoryTest, boolean onlyCore) {  
  3.             ……  
  4. mSystemAppDir = new File(Environment.getRootDirectory(), "app");  
  5.             mSystemInstallObserver = new AppDirObserver(  
  6.                 mSystemAppDir.getPath(), OBSERVER_EVENTS, true);  
  7.             mSystemInstallObserver.startWatching();  
  8.             scanDirLI(mSystemAppDir, PackageParser.PARSE_IS_SYSTEM  
  9.                     | PackageParser.PARSE_IS_SYSTEM_DIR, scanMode, 0);  
  10. //后续还有其他路径下的app文件扫描  
  11. ……  
  12. }  
  13. private void scanDirLI(File dir, int flags, int scanMode, long currentTime) {  
  14.      ……  
  15. for (i=0; i<files.length; i  ) {  
  16. PackageParser.Package pkg = scanPackageLI(file,  
  17.                     flags|PackageParser.PARSE_MUST_BE_APK, scanMode, currentTime, null);  
  18.   
  19. ……  
  20. }  
  21. }  
  22. private PackageParser.Package scanPackageLI(File scanFile,  
  23.             int parseFlags, int scanMode, long currentTime, UserHandle user) {  
  24.         PackageParser pp = new PackageParser(scanPath);  
  25.         pp.setSeparateProcesses(mSeparateProcesses);  
  26.         pp.setOnlyCoreApps(mOnlyCore);  
  27.         final PackageParser.Package pkg = pp.parsePackage(scanFile,  
  28.                 scanPath, mMetrics, parseFlags);  
  29.         …….  
  30.         PackageParser.Package scannedPkg = scanPackageLI(pkg, parseFlags, scanMode  
  31.                 | SCAN_UPDATE_SIGNATURE, currentTime, user);  
  32.         ……  
  33. }  
  34. public Package parsePackage(File sourceFile, String destCodePath,  
  35.             DisplayMetrics metrics, int flags) {  
  36.   
  37. try {  
  38.             // XXXX todo: need to figure out correct configuration.  
  39.             pkg = parsePackage(res, parser, flags, errorText);  
  40.         } catch (Exception e) {  
  41.             errorException = e;  
  42.             mParseError = PackageManager.INSTALL_PARSE_FAILED_UNEXPECTED_EXCEPTION;  
  43.         }  
  44. }  
  45. private Package parsePackage(  
  46.         Resources res, XmlResourceParser parser, int flags, String[] outError)  
  47.         throws XmlPullParserException, IOException {  
  48. ……  
  49. String tagName = parser.getName();  
  50.             if (tagName.equals("activity")) {  
  51.                 Activity a = parseActivity(owner, res, parser, attrs, flags, outError, false,  
  52.                         hardwareAccelerated);  
  53.                 if (a == null) {  
  54.                     mParseError = PackageManager.INSTALL_PARSE_FAILED_MANIFEST_MALFORMED;  
  55.                     return false;  
  56.                 }  
  57.   
  58.                 owner.activities.add(a);  
  59.   
  60.             } else if (tagName.equals("receiver")) {  
  61.                 Activity a = parseActivity(owner, res, parser, attrs, flags, outError, true, false);  
  62.                 if (a == null) {  
  63.                     mParseError = PackageManager.INSTALL_PARSE_FAILED_MANIFEST_MALFORMED;  
  64.                     return false;  
  65.                 }  
  66.   
  67.                 owner.receivers.add(a);  
  68.   
  69.             }  
  70.             …….  
  71. }  
  72.   
  73. private PackageParser.Package scanPackageLI(PackageParser.Package pkg,  
  74.             int parseFlags, int scanMode, long currentTime, UserHandle user) {  
  75.      ……  
  76. mPackages.put(pkg.applicationInfo.packageName, pkg);  
  77. ……  
  78. N = pkg.receivers.size();  
  79.             r = null;  
  80.             for (i=0; i<N; i  ) {  
  81.                 PackageParser.Activity a = pkg.receivers.get(i);  
  82.                 a.info.processName = fixProcessName(pkg.applicationInfo.processName,  
  83.                         a.info.processName, pkg.applicationInfo.uid);  
  84.                 mReceivers.addActivity(a, "receiver");  
  85. }  
  86. }  


到这里可以看到系统是如何解析manifest里面注册的广播接收器的,并且最终将所有的接收器保存到了
// All available receivers, for your resolving pleasure.
final ActivityIntentResolver mReceivers = new ActivityIntentResolver();中,请注意这个成员变量,后面会有用的,这里也暗含了一个静态的排序,也就是首先扫描目录的排序,显示system/app后面是其他的,还有一个就是file.list()对目录下apk文件的默认排序。
还有一种动态注册的方式,平常我们使用registerReceiver其实最终是ContextImpl里面的registerReceiverInternal方法来实现的,可以看看:
[html]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. private Intent registerReceiverInternal(BroadcastReceiver receiver, int userId,  
  2.             IntentFilter filter, String broadcastPermission,  
  3.             Handler scheduler, Context context) {  
  4.         IIntentReceiver rd = null;  
  5.         if (receiver != null) {  
  6.             if (mPackageInfo != null && context != null) {  
  7.                 if (scheduler == null) {  
  8.                     scheduler = mMainThread.getHandler();  
  9.                 }  
  10.                 rd = mPackageInfo.getReceiverDispatcher(  
  11.                     receiver, context, scheduler,  
  12.                     mMainThread.getInstrumentation(), true);  
  13.             } else {  
  14.                 if (scheduler == null) {  
  15.                     scheduler = mMainThread.getHandler();  
  16.                 }  
  17.                 rd = new LoadedApk.ReceiverDispatcher(  
  18.                         receiver, context, scheduler, null, true).getIIntentReceiver();  
  19.             }  
  20.         }  
  21.         try {  
  22.             return ActivityManagerNative.getDefault().registerReceiver(  
  23.                     mMainThread.getApplicationThread(), mBasePackageName,  
  24.                     rd, filter, broadcastPermission, userId);  
  25.         } catch (RemoteException e) {  
  26.             return null;  
  27.         }  
  28. }  


看到最后其实还是activitymanager在工作,android的设计就是如此,由service管理各个组件省去了很多的工作。
[html]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. public Intent registerReceiver(IApplicationThread caller, String callerPackage,  
  2.             IIntentReceiver receiver, IntentFilter filter, String permission, int userId) {  
  3.        ……  
  4. mReceiverResolver.addFilter(bf);  
  5.        ……  
  6.             // Enqueue broadcasts for all existing stickies that match  
  7.             // this filter.  
  8.             if (allSticky != null) {  
  9.                 ArrayList receivers = new ArrayList();  
  10.                 receivers.add(bf);  
  11.   
  12.                 int N = allSticky.size();  
  13.                 for (int i=0; i<N; i  ) {  
  14.                     Intent intent = (Intent)allSticky.get(i);  
  15.                     BroadcastQueue queue = broadcastQueueForIntent(intent);  
  16.                     BroadcastRecord r = new BroadcastRecord(queue, intent, null,  
  17.                             null, -1, -1, null, receivers, null, 0, null, null,  
  18.                             false, true, true, -1);  
  19.                     queue.enqueueParallelBroadcastLocked(r);  
  20.                     queue.scheduleBroadcastsLocked();  
  21.                 }  
  22.             }  
  23. ……  
  24. }  


最终将这些动态广播接收器豆浆保存到成员变量mReceiverResolver中(既然这里是一个个add进去的,隐含了一个排序,也就是先注册的动态广播在前),并且注意allsticky这里的判断,这是sticky广播处理的逻辑,粘性广播会停留在service中等待,以便有人注册这则广播消息后能尽快的收到这条广播。
在注册完了之后,广播主要就是发送和处理了。这个过程都是在activitymanagerservice里面进行的。可以看看它的broadcastIntentLocked方法,主要的工作都是这里进行的,尤其注意的是普通广播与有序广播,以及广播的次序等信息。
[html]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. private final int broadcastIntentLocked(ProcessRecord callerApp,  
  2.             String callerPackage, Intent intent, String resolvedType,  
  3.             IIntentReceiver resultTo, int resultCode, String resultData,  
  4.             Bundle map, String requiredPermission,  
  5.             boolean ordered, boolean sticky, int callingPid, int callingUid,  
  6.             int userId) {  
  7. ……  
  8.         // Add to the sticky list if requested.  
  9.         if (sticky) {  
  10.             if (checkPermission(android.Manifest.permission.BROADCAST_STICKY,  
  11.                     callingPid, callingUid)  
  12.                     != PackageManager.PERMISSION_GRANTED) {  
  13.                 String msg = "Permission Denial: broadcastIntent() requesting a sticky broadcast from pid="  
  14.                           callingPid   ", uid="   callingUid  
  15.                           " requires "   android.Manifest.permission.BROADCAST_STICKY;  
  16.                 Slog.w(TAG, msg);  
  17.                 throw new SecurityException(msg);  
  18.             }  
  19.             if (requiredPermission != null) {  
  20.                 Slog.w(TAG, "Can't broadcast sticky intent "   intent  
  21.                           " and enforce permission "   requiredPermission);  
  22.                 return ActivityManager.BROADCAST_STICKY_CANT_HAVE_PERMISSION;  
  23.             }  
  24.             if (intent.getComponent() != null) {  
  25.                 throw new SecurityException(  
  26.                         "Sticky broadcasts can't target a specific component");  
  27.             }  
  28.             // We use userId directly here, since the "all" target is maintained  
  29.             // as a separate set of sticky broadcasts.  
  30.             if (userId != UserHandle.USER_ALL) {  
  31.                 // But first, if this is not a broadcast to all users, then  
  32.                 // make sure it doesn't conflict with an existing broadcast to  
  33.                 // all users.  
  34.                 HashMap<String, ArrayList<Intent>> stickies = mStickyBroadcasts.get(  
  35.                         UserHandle.USER_ALL);  
  36.                 if (stickies != null) {  
  37.                     ArrayList<Intent> list = stickies.get(intent.getAction());  
  38.                     if (list != null) {  
  39.                         int N = list.size();  
  40.                         int i;  
  41.                         for (i=0; i<N; i  ) {  
  42.                             if (intent.filterEquals(list.get(i))) {  
  43.                                 throw new IllegalArgumentException(  
  44.                                         "Sticky broadcast "   intent   " for user "  
  45.                                           userId   " conflicts with existing global broadcast");  
  46.                             }  
  47.                         }  
  48.                     }  
  49.                 }  
  50.             }  
  51.             HashMap<String, ArrayList<Intent>> stickies = mStickyBroadcasts.get(userId);  
  52.             if (stickies == null) {  
  53.                 stickies = new HashMap<String, ArrayList<Intent>>();  
  54.                 mStickyBroadcasts.put(userId, stickies);  
  55.             }  
  56.             ArrayList<Intent> list = stickies.get(intent.getAction());  
  57.             if (list == null) {  
  58.                 list = new ArrayList<Intent>();  
  59.                 stickies.put(intent.getAction(), list);  
  60.             }  
  61.             int N = list.size();  
  62.             int i;  
  63.             for (i=0; i<N; i  ) {  
  64.                 if (intent.filterEquals(list.get(i))) {  
  65.                     // This sticky already exists, replace it.  
  66.                     list.set(i, new Intent(intent));  
  67.                     break;  
  68.                 }  
  69.             }  
  70.             if (i >= N) {  
  71.                 list.add(new Intent(intent));  
  72.             }  
  73.         }  
  74. ……  
  75. // Figure out who all will receive this broadcast.  
  76. //静态广播接收器  
  77.         List receivers = null;  
  78. //动态广播接收器  
  79.         List<BroadcastFilter> registeredReceivers = null;  
  80.         // Need to resolve the intent to interested receivers...  
  81.         if ((intent.getFlags()&Intent.FLAG_RECEIVER_REGISTERED_ONLY)  
  82.                  == 0) {  
  83.             receivers = collectReceiverComponents(intent, resolvedType, users);  
  84.         }  
  85.         if (intent.getComponent() == null) {  
  86.             registeredReceivers = mReceiverResolver.queryIntent(intent,  
  87.                     resolvedType, false, userId);  
  88.         }  
  89. ……  
  90. //处理非有序的广播  
  91.         if (!ordered && NR > 0) {  
  92.             // If we are not serializing this broadcast, then send the  
  93.             // registered receivers separately so they don't wait for the  
  94.             // components to be launched.  
  95.             final BroadcastQueue queue = broadcastQueueForIntent(intent);  
  96.             BroadcastRecord r = new BroadcastRecord(queue, intent, callerApp,  
  97.                     callerPackage, callingPid, callingUid, requiredPermission,  
  98.                     registeredReceivers, resultTo, resultCode, resultData, map,  
  99.                     ordered, sticky, false, userId);  
  100.             if (DEBUG_BROADCAST) Slog.v(  
  101.                     TAG, "Enqueueing parallel broadcast "   r);  
  102.             final boolean replaced = replacePending && queue.replaceParallelBroadcastLocked(r);  
  103.             if (!replaced) {  
  104. //只处理了动态注册的接收器  
  105.                 queue.enqueueParallelBroadcastLocked(r);  
  106.                 queue.scheduleBroadcastsLocked();  
  107.             }  
  108.             registeredReceivers = null;  
  109.             NR = 0;  
  110.         }  
  111. ……  
  112.         // Merge into one list.  
  113. //后面的逻辑就是,如果前面是非有序广播,就先处理与之匹配的动态注册的并置为空,后面的逻辑就只有静态的接收器并处理  
  114. //如果前面是有序广播则不处理,此后的逻辑中先对静态和动态的接收器按照priority值进行一个统一的排序最后处理。  
  115.         int ir = 0;  
  116.         if (receivers != null) {  
  117.             ……  
  118.             int NT = receivers != null ? receivers.size() : 0;  
  119.             int it = 0;  
  120.             ResolveInfo curt = null;  
  121.             BroadcastFilter curr = null;  
  122.             while (it < NT && ir < NR) {  
  123.                 if (curt == null) {  
  124.                     curt = (ResolveInfo)receivers.get(it);  
  125.                 }  
  126.                 if (curr == null) {  
  127.                     curr = registeredReceivers.get(ir);  
  128.                 }  
  129. //如果动态广播接收器优先级高于或者等于静态广播接收器,那么就插到当前位置,静态广播后移  
  130. //这说明动态的要在静态的前面  
  131.                 if (curr.getPriority() >= curt.priority) {  
  132.                     // Insert this broadcast record into the final list.  
  133.                     receivers.add(it, curr);  
  134.                     ir  ;  
  135.                     curr = null;  
  136.                     it  ;  
  137.                     NT  ;  
  138.                 } else {  
  139.                     // Skip to the next ResolveInfo in the final list.  
  140.                     it  ;  
  141.                     curt = null;  
  142.                 }  
  143.             }  
  144.         }  
  145.         while (ir < NR) {  
  146.             if (receivers == null) {  
  147.                 receivers = new ArrayList();  
  148.             }  
  149.             receivers.add(registeredReceivers.get(ir));  
  150.             ir  ;  
  151.         }  
  152.   
  153.         if ((receivers != null && receivers.size() > 0)  
  154.                 || resultTo != null) {  
  155. // broadcastQueueForIntent这个方法中对当前是否前台的广播接收器队列进行了判断,前台和后台的接收器处理超时是不一样的,前台是10m,后台是60m。  
  156.             BroadcastQueue queue = broadcastQueueForIntent(intent);  
  157.             BroadcastRecord r = new BroadcastRecord(queue, intent, callerApp,  
  158.                     callerPackage, callingPid, callingUid, requiredPermission,  
  159.                     receivers, resultTo, resultCode, resultData, map, ordered,  
  160.                     sticky, false, userId);  
  161.             if (DEBUG_BROADCAST) Slog.v(  
  162.                     TAG, "Enqueueing ordered broadcast "   r  
  163.                       ": prev had "   queue.mOrderedBroadcasts.size());  
  164.             if (DEBUG_BROADCAST) {  
  165.                 int seq = r.intent.getIntExtra("seq", -1);  
  166.                 Slog.i(TAG, "Enqueueing broadcast "   r.intent.getAction()   " seq="   seq);  
  167.             }  
  168.             boolean replaced = replacePending && queue.replaceOrderedBroadcastLocked(r);  
  169.             if (!replaced) {  
  170.                 queue.enqueueOrderedBroadcastLocked(r);  
  171.                 queue.scheduleBroadcastsLocked();  
  172.             }  
  173.         }  
  174.   
  175.         return ActivityManager.BROADCAST_SUCCESS;  
  176. }  


总结一下这里面含有的排序信息:分为两种,有序广播,首先按照优先级排列,同优先级的动态广播先于静态广播,同优先级的动态广播中先注册的先处理,同优先级的静态广播中先扫描的APP广播先处理。
有序广播则,无视优先级,动态广播先于静态广播,动态广播中先注册的先处理,静态广播中先扫描的APP广播先处理。

以上是广播的注册以及send时候的分发,接下来就是ActivtyManagerService调用到用户程序的broadcastreceiver中处理的过程了,将在后续篇章中进行阐述。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值