注册广播接收器registerReceiver

注册广播接收器registerReceiver从registerReceiver(BroadcastReceiverreceiver,IntentFilterfilter)出发所经历的类和方法:registerReceiver(receiver,filter)–>ContextWrapper.java$registerReceiver(receiver,filter);@OverridepublicIntentregis…

大家好,又见面了,我是你们的朋友全栈君。如果您正在找激活码,请点击查看最新教程,关注关注公众号 “全栈程序员社区” 获取激活教程,可能之前旧版本教程已经失效.最新Idea2022.1教程亲测有效,一键激活。

Jetbrains全家桶1年46,售后保障稳定

从registerReceiver(BroadcastReceiver receiver,IntentFilter filter)出发

所经历的类和方法:registerReceiver(receiver,filter)–>ContextWrapper.java$registerReceiver(receiver,filter);

@Override
public Intent registerReceiver(
    BroadcastReceiver receiver, IntentFilter filter) {
    return mBase.registerReceiver(receiver, filter);//Context mBase
}

Jetbrains全家桶1年46,售后保障稳定

Context是接口,其实现类ContextImpl,因此其具体实现是ContextImpl$reisterReceiver(receiver,filter);

@Override
public Intent registerReceiver(BroadcastReceiver receiver, IntentFilter filter) {
    return registerReceiver(receiver, filter, null, null);
}

@Override
public Intent registerReceiver(BroadcastReceiver receiver, IntentFilter filter,
        int flags) {
    return registerReceiver(receiver, filter, null, null, flags);
}

@Override
public Intent registerReceiver(BroadcastReceiver receiver, IntentFilter filter,
        String broadcastPermission, Handler scheduler) {
    return registerReceiverInternal(receiver, getUserId(),
            filter, broadcastPermission, scheduler, getOuterContext(), 0);
}

用以上代码可知调用registerReceiver以后其内部调用了其重载函数,其重载函数又进一步调了registerReceiverInternal()方法

private Intent registerReceiverInternal(BroadcastReceiver receiver, int userId,
        IntentFilter filter, String broadcastPermission,
        Handler scheduler, Context context, int flags) {
    IIntentReceiver rd = null;
    if (receiver != null) {
        if (mPackageInfo != null && context != null) {
            if (scheduler == null) {
                scheduler = mMainThread.getHandler();       //1
            }
            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 = ActivityManager.getService().registerReceiver(
                mMainThread.getApplicationThread(), mBasePackageName, rd, filter,
                broadcastPermission, userId, flags);
        if (intent != null) {
            intent.setExtrasClassLoader(getClassLoader());
            intent.prepareToEnterProcess();
        }
        return intent;
    } catch (RemoteException e) {
        throw e.rethrowFromSystemServer();
    }
}

IIntentReceiver :IIntentReceiver其实是一个AIDL接口,内部就一个方法performReceive(…);
mPackageInfo:ContextImpl导入的一个LoadedApk的实例,是用来处理广播接收的。

以上参数context通过getOuterContext获取注册广播时所在的context,方法走到//1处,scheduler = mMainThread.getHandler();该scheduler是一个Handler类型,mMainThread是ActivityThread类型,这里handler主要用来负责分发AMS发送过来的广播用的。

public final class ActivityThread {
......
final H mH = new H();
private final class H extends Handler {
	......
	public void handleMessage(Message msg) {
		......
		switch (msg.what) {
		......
		}
		......
	}
	.....
}

......
final Handler getHandler() {
	return mH;
}

获取完scheduler(Handler),执行 rd = mPackageInfo.getReceiverDispatcher(…),

public IIntentReceiver getReceiverDispatcher(BroadcastReceiver r,
        Context context, Handler handler,
        Instrumentation instrumentation, boolean registered) {
    synchronized (mReceivers) {
        LoadedApk.ReceiverDispatcher rd = null;
        ArrayMap<BroadcastReceiver, LoadedApk.ReceiverDispatcher> map = null;
        if (registered) {
            map = mReceivers.get(context);
            if (map != null) {
                rd = map.get(r);
            }
        }
        if (rd == null) {
            rd = new ReceiverDispatcher(r, context, handler,
                    instrumentation, registered);
            if (registered) {
                if (map == null) {
                    map = new ArrayMap<BroadcastReceiver, LoadedApk.ReceiverDispatcher>();
                    mReceivers.put(context, map);
                }
                map.put(r, rd);
            }
        } else {
            rd.validate(context, handler);
        }
        rd.mForgotten = false;
        return rd.getIIntentReceiver();
    }
}

首先这个函数的返回类型是IIntentReceiver,这里可以看做是帮我们跨进程通信的Binder对象。

mReceivers:private final ArrayMap<Context, ArrayMap<BroadcastReceiver, ReceiverDispatcher>> mReceivers是通过这些数组和Map把注册广播所在的Context,receiver以及ReceiverDispatcher封装起来,逻辑是如果广播registered成立,以context为key获取broadcastReceiver和ReceiverDispatcher。

假设MainActivity为Key值保存在LoadedApk的成员变量mReceivers中,这样,只要给定一个Activity和BroadcastReceiver,就可以查看LoadedApk里面是否已经存在相应的广播接收发布器ReceiverDispatcher了。

看看ReceiverDispather这个类

 static final class ReceiverDispatcher {

    final static class InnerReceiver extends IIntentReceiver.Stub {
        final WeakReference<LoadedApk.ReceiverDispatcher> mDispatcher;
        final LoadedApk.ReceiverDispatcher mStrongRef;

        InnerReceiver(LoadedApk.ReceiverDispatcher rd, boolean strong) {
            mDispatcher = new WeakReference<LoadedApk.ReceiverDispatcher>(rd);
            mStrongRef = strong ? rd : null;
        }

InnerReceiver是静态内部类,继承IIntentReceiver.Stub,InnerReceiver就是Binder接口IIntentReceiver的实现类

return rd.getIIntentReceiver();-->return mIIntentReceiver
mIIntentReceiver = new InnerReceiver(this, !registered);

而通过以上rd.getIIntetReceiver可以看出rd就是InnerReceiver对象,也就是binder对象

在新建广播接收发布器ReceiverDispatcher时,会在构造函数里面创建一个InnerReceiver实例,这是一个Binder对象,实现了IIntentReceiver接口,可以通过ReceiverDispatcher.getIIntentReceiver函数来获得,获得后就会把它传给ActivityManagerService,以便接收广播。在ReceiverDispatcher类的构造函数中,还会把传进来的Handle类型的参数activityThread保存下来,以便后面在分发广播的时候使用。

接下来是 Intent intent = ActivityManager.getService().registerReceiver()

.....
Intent intent = ActivityManager.getService().registerReceiver()
...

最后进入到ActivityManagerService中的registerReceiver函数中去,ActivityMangerService$registerReceiver();看看它是怎么进去的

过往的版本貌似都是ActivityManagerNative.getDefault().registerReceiver()而不是ActivityManager.getService().registerReceiver(),看看两者的实现步骤都具体做了什么,有什么区别

<1> 我先定位到ActivityManger.getService()这个方法

  /**
 * @hide
 */
@UnsupportedAppUsage
public static IActivityManager getService() {
    return IActivityManagerSingleton.get();
}

接下来看看IActivityMangerSingletone()

 @UnsupportedAppUsage
private static final Singleton<IActivityManager> IActivityManagerSingleton =
        new Singleton<IActivityManager>() {
            @Override
            protected IActivityManager create() {
                final IBinder b = ServiceManager.getService(Context.ACTIVITY_SERVICE);
                final IActivityManager am = IActivityManager.Stub.asInterface(b);
                return am;
            }
        };

在frameworks/base/core/java/android/util/Singleton.java

public abstract class Singleton<T> {
  @UnsupportedAppUsage
 private T mInstance;
  protected abstract T create();
  @UnsupportedAppUsage
  public final T get() {
      synchronized (this) {
          if (mInstance == null) {
              mInstance = create();
          }
         return mInstance;
     }
  }  
  }

单纯从代码分析以上两段代码,IActivityManagerSingleton是Singleton变量,而singleton是一个抽象类内部声明了一个抽象方法create(),而其get方法通过由一个继承了Singleton并复写函数create()来给mInstance赋值,回到上层IActivityManagerSingleton,new了Singleton并重写了函数create(),通过Context.ACTIVITY_SERVICE获取一个binder,,并通过该对象返回了IActivityManager对象,最后ActivityManager.getService().registerReceiver()等价于IActivityManager.registerReceiver()。

具体以上的代码,就是Singleton这么一个工具类通过单例模式创建了一个对象,而这个过程是这样的,先获取了关联了系统服务的ActivityManagerService的binder对象[ final IBinder b = ServiceManager.getService(Context.ACTIVITY_SERVICE);],然后通过IActivityManager.Stub.asInterface(b)返回一个IActivityManager的代理对象,基于Binder机制,通过调用代理对象的方法,使得系统服务ActivityManagerService对应的方法被调用,因为ActivityManagerService继承了IActivityManager.Stub

public class ActivityManagerService extends IActivityManager.Stub
      implements Watchdog.Monitor, BatteryStatsImpl.BatteryCallback {
      ......
      }

<2>ActivityManagerNative.getDefault().registerReceiver()看看他是怎么个操作

static public IActivityManager getDefault(){
    return ActivityManager.getService();
}

我丢!一样啊,只是这个方法封装了一下而已,到头来还是ActivityManager.getService(),打扰了

ActivityManagerService.registerReceiver的处理

函数首先是获得调用registerReceiver函数的应用程序进程记录块:

public Intent registerReceiver(IApplicationThread caller, String callerPackage,IIntentReceiver receiver,
IntentFilter filter,String permission, int userId,int flags) {
... //初始化了一些需要的变量
synchronized(this) {
if (caller != null) {
callerApp = getRecordForAppLocked(caller);
}
if(cllerApp == null){
    ....//异常处理
}if (callerApp.info.uid != SYSTEM_UID &&
    !callerApp.pkgList.containsKey(callerPackage) &&
    !"android".equals(callerPackage)) {
    ...//抛异常
              }
    callingUid = callerApp.info.uid;//进程中第一个app的uid
    callingPid = callerApp.pid;该app应用的process id
    } else {
    callerPackage = null;
    callingUid = Binder.getCallingUid();
    callingPid = Binder.getCallingPid();
          }
instantApp = isInstantApp(callerApp, callerPackage, callingUid);

看看instantApp是什么,怎么获取的

 private boolean isInstantApp(ProcessRecord record, @Nullable String callerPackage, int uid) {
      if (UserHandle.getAppId(uid) < FIRST_APPLICATION_UID) {
          return false;
      }
      // Easy case -- we have the app's ProcessRecord.
      if (record != null) {
          return record.info.isInstantApp();
      }
      // Otherwise check with PackageManager.
      IPackageManager pm = AppGlobals.getPackageManager();
      try {
          if (callerPackage == null) {
              final String[] packageNames = pm.getPackagesForUid(uid);
              if (packageNames == null || packageNames.length == 0) {
                  throw new IllegalArgumentException("Unable to determine caller package name");
              }
              // Instant Apps can't use shared uids, so its safe to only check the first package.
              callerPackage = packageNames[0];
          }
          mAppOpsService.checkPackage(uid, callerPackage);
          return pm.isInstantApp(callerPackage, UserHandle.getUserId(uid));
      } catch (RemoteException e) {
          Slog.e(TAG, "Error looking up if " + callerPackage + " is an instant app.", e);
          return true;
      }
  }

在Process.java中FIRST_APPLICATION_UID=10000;规定应用Uid是10000-19999,因此if(UserHandle.getAppId(uid) < FIRST_APPLICATION_UID)这就是判断如果不是应用的uid就返回false.

然后通过PackageManager检查是不是instant app如果是返回ture;

以下是处理粘性广播的处理

有时候用户希望发送sticky广播,以便日后注册的receiver可以收到“错过”的sticky广播。要达到这个目的,系统必须在内部维护一张sticky广播表,在具体的实现中,AMS会把广播intent加入mStickyBroadcasts映射表中。mStickyBroadcasts是一张哈希映射表,其key值为intent的action字符串,value值为“与这个action对应的intent数组列表”的引用。当我们发送sticky广播时,新的广播intent要么替换掉intent数组列表中的某项,要么作为一个新项被添加进数组列表,以备日后使用。

 Iterator<String> actions = filter.actionsIterator();
 ...
 int[] userIds = { UserHandle.USER_ALL,UserHandle.getUserId(callingUid) };
 while (actions.hasNext()) {
 String action = actions.next();
    for (int id : userIds) {//mStickyBroadcasts保留着系统所有的粘性广播
        ArrayMap<String, ArrayList<Intent>> stickies = mStickyBroadcasts.get(id);
            if (stickies != null) {
                 ArrayList<Intent> intents = stickies.get(action);
                    if (intents != null) {
                        if (stickyIntents == null) {
                            stickyIntents = new ArrayList<Intent>();
                          stickyIntents.addAll(intents);
                      }
}

//到此为止stickyIntents是系统中所有的粘性的intent

    ArrayList<Intent> allSticky = null;
    if (stickyIntents != null) {
        final ContentResolver resolver = mContext.getContentResolver();
        // Look for any matching sticky broadcasts...
        for (int i = 0, N = stickyIntents.size(); i < N; i++) {
            Intent intent = stickyIntents.get(i);
            // If intent has scheme "content", it will need to acccess
            // provider that needs to lock mProviderMap in ActivityThread
            // and also it may need to wait application response, so we
            // cannot lock ActivityManagerService here.
            if (filter.match(resolver, intent, true, TAG) >= 0) {
                if (allSticky == null) {
                    allSticky = new ArrayList<Intent>();
                }
                allSticky.add(intent); 
            }
        }
    }
    //allSticky就是与本次注册intentfilter匹配的粘性广播

        if (allSticky != null) {
            ArrayList receivers = new ArrayList();
            receivers.add(bf);

            final int stickyCount = allSticky.size();
            for (int i = 0; i < stickyCount; i++) {
                Intent intent = allSticky.get(i);   //处理粘性广播
                BroadcastQueue queue = broadcastQueueForIntent(intent);
                BroadcastRecord r = new BroadcastRecord(queue, intent, null,
                        null, -1, -1, null, null, AppOpsManager.OP_NONE, null, receivers,
                        null, 0, null, null, false, true, true, -1);
                queue.enqueueParallelBroadcastLocked(r);
                queue.scheduleBroadcastsLocked();
            }
        }

        return sticky;
    }
}

1.mStickyBroadcasts,key是用户的id,data是广播intent数组。由此可见,粘性广播是分用户的,用户1发出的粘性广播,用户2可能收不到。

2.stickyIntents 保存的是系统所有粘性广播intent

3.allSticky 保存的是与本次registerReceiver(BroadcastReceiver receiver, IntentFilter intent)中intent匹配的粘性广播的intent

扩展:粘性消息:粘性消息在发送后就一直存在于系统的消息容器里面,等待对应的处理器去处理,如果暂时没有处理器处理这个消息则一直在消息容器里面处于等待状态,粘性广播的Receiver如果被销毁,那么下次重建时会自动接收到消息数据. 注意:普通广播和粘性消息不同被截获,而有序广播是可

如果不是用sendStickyBroadcat()发的 因此这里的sticky和allSticky都为null
会进入以下代码

  ReceiverList rl = mRegisteredReceivers.get(receiver.asBinder());
          if (rl == null) {
              rl = new ReceiverList(this, callerApp, callingPid, callingUid,
                      userId, receiver);
              if (rl.app != null) {
                  final int totalReceiversForApp = rl.app.receivers.size();
                  if (totalReceiversForApp >= MAX_RECEIVERS_ALLOWED_PER_APP) {
                      throw new IllegalStateException("Too many receivers, total of "
                              + totalReceiversForApp + ", registered for pid: "
                              + rl.pid + ", callerPackage: " + callerPackage);
                  }
                  rl.app.receivers.add(rl);
              } else {
                  try {
                      receiver.asBinder().linkToDeath(rl, 0);
                  } catch (RemoteException e) {
                      return sticky;
                  }
                  rl.linkedToDeath = true;
              }
              mRegisteredReceivers.put(receiver.asBinder(), rl);

ReceiverList rl = mRegisteredReceivers.get(receiver.asBinder());

其中的receiver参数类型是IIntentReceiver,也就是对应着之前的ReceiverDispatcher中那个binder实体。

下面是ReceiverList的代码片段:

class ReceiverList extends ArrayList<BroadcastFilter>implements IBinder.DeathRecipient {
final ActivityManagerService owner; 
public final IIntentReceiver receiver;    
public final ProcessRecord app;    
public final int pid;    
public final int uid;
BroadcastRecord curBroadcast = null;
boolean linkedToDeath = false;
String stringName;
. . . . . .

ReceiverList继承于ArrayList,而BroadcastFilter又继承于IntentFilter,所以ReceiverList可以被理解为一个IntentFilter数组列表。

就是把注册的receivers保存到这个ReceiverList表里,其中还判断了如果注册的receiver数量过多抛出异常,最后把这receivers表放在mRegisteredReceivers中,这些都是为了方便在收到广播时,快速找到对应的广播接收器的。其中rl.app可以看做是这个注册receiver的宿主app的进程。

 BroadcastFilter bf = new BroadcastFilter(filter, rl, callerPackage,
                  permission, callingUid, userId, instantApp, visibleToInstantApps);
          if (rl.containsFilter(filter)) {
         ...
         mReceiverResolver.addFilter(bf);
         }

以上代码作用就是创建一个BroadcastFilter来把广播接收器列表rl和filter关联起来,然后保存在ActivityManagerService中的成员变量mReceiverResolver中去,因为之前只保存receivers没有和其filter发生关联

BroadCastFilter代码片段:

BroadcastFilter(IntentFilter _filter, ReceiverList _receiverList,
          String _packageName, String _requiredPermission, int _owningUid, int _userId,
          boolean _instantApp, boolean _visibleToInstantApp) {
      super(_filter);
      receiverList = _receiverList;
      packageName = _packageName;
      requiredPermission = _requiredPermission;
      owningUid = _owningUid;
      owningUserId = _userId;
      instantApp = _instantApp;
      visibleToInstantApp = _visibleToInstantApp;
  }
  ...

最后将BroadcastFilter添加到IntentResolver类型的mReceiverResolver中,这样当AMS接收到广播时,就可以从mReceiverResolver中直接找到对应的广播接收者,从而达到注册广播的目的。

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。

发布者:全栈程序员-用户IM,转载请注明出处:https://javaforall.cn/210458.html原文链接:https://javaforall.cn

【正版授权,激活自己账号】: Jetbrains全家桶Ide使用,1年售后保障,每天仅需1毛

【官方授权 正版激活】: 官方授权 正版激活 支持Jetbrains家族下所有IDE 使用个人JB账号...

(0)


相关推荐

  • 【Stimulsoft Reports.WPF教程】保存并加载Designer的事件

    【下载StimulsoftReports.WPF最新版本】此示例显示如何在代码中使用“Save”和“Load”事件。您可以在应用程序初始化时将事件侦听器添加到StiOptions.Engine.GlobalEvents:publicWindow1(){StiOptions.Wpf.CurrentTheme=StiOptions.Wpf.Themes.Office20…

  • idea激活码2022【最新永久激活】

    (idea激活码2022)最近有小伙伴私信我,问我这边有没有免费的intellijIdea的激活码,然后我将全栈君台教程分享给他了。激活成功之后他一直表示感谢,哈哈~https://javaforall.cn/100143.htmlIntelliJ2021最新激活注册码,破解教程可免费永久激活,亲测有效,上面是详细链接哦~S32P…

  • C语言输入输出格式符[通俗易懂]

    C语言输入输出格式符[通俗易懂]C语言输入输出格式符printf函数(格式输出函数)1.一般格式printf(格式控制,输出表列)例如:printf(“i=%d,ch=%c\n”,i,ch);说明:(1)“格式控制”是用双撇号括起来的字符串,也称“转换控制字符串”,它包括两种信息:①格式说明:由“%”和格式字符组成,它的作用是将输出的数据转换为指定的格式输出。②普通字符,即需要原样输出的字符。(2)“输出表列”是需要输出的一些数据,可以是表达式(3)printf函数的一般形式可以表示为printf(参数1,参数2,…

  • python 字符串去空格

    python 字符串去空格一、去除字符串空格,在Python里面有它的内置方法lstrip:删除左边的空格这个字符串方法,会删除字符串s开始位置前的空格。>>>s.lstrip()’string’rstrip:删除右连的空格这个内置方法可以删除字符串末尾的所有空格,看下面演示代码:>>>s.rstrip()’string’strip:删除两端的空格有的时候我们读取文件中的内容,每行2边都有空

  • linux内核使用的编程语言_linux内核模块编程

    linux内核使用的编程语言_linux内核模块编程1、内核编程不能访问C库2、内核编程时必须使用GNUC3、内核编程时缺乏像用户空间那样的内存保护机制4、内核编程时浮点数很难使用5、内核只有一个很小的定长堆栈6、由于内核支持异步中断,抢占和SMP,因此必须时刻注意同步和并发7、要考虑可移植性的重要性

  • ssh 报Connection closed by 127.0.0.1 port 的解决办法

    ssh 报Connection closed by 127.0.0.1 port 的解决办法起了一个虚拟机,想要通过ssh从host连接。起虚拟机时加上如下命令行:-netdevuser,id=net0,hostfwd=tcp::2222-:22\-devicevirtio-net-pci,netdev=net0\这样就将host的2222端口转发给虚拟机的22端口。sshroot@localhost-p2222得到Connection……

发表回复

您的电子邮箱地址不会被公开。

关注全栈程序员社区公众号