Andorid 8.0 对广播的使用做了变更。
当广播接收器使用静态注册方式使用时,除了一些例外,这个接收器接收不到隐式广播。 注意这个“隐式”是重点。
看了网上几篇文章,对这个变更理解有误。错误的理解是:8.0后,广播接收器使用静态注册,是无法使用的。
实时并非如此。
先看一个例子:
首先,定义一个简单的广播接收器:
public class MyReceiver extends BroadcastReceiver { @Override public void onReceive(Context context, Intent intent) { Toast.makeText(context,context.toString(),Toast.LENGTH_SHORT).show(); } }
它对接收到广播的行为就是打印一句话。
第二,我们将他注册到Manifest文件中。
<receiver android:name=".broadcast.MyReceiver"> <intent-filter> <action android:name="com.demo.recriver"/> </intent-filter> </receiver>
最后,在Activity中发送一个广播,intent通过设置Action为com.demo.recriver的形式发送隐式广播。
public class MainActivity extends AppCompatActivity { @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); Intent intent = Intent intent = new Intent("com.demo.recriver");//隐式intent,发送隐式广播 sendBroadcast(intent); } }
运行这个demo,发现在8.0以下的手机上,会有Toast显示,8.0以上的手机不会弹出,说明没有接收到广播。
原因在于这个广播 是“隐式” 发送的,8.0中,静态注册的广播接收者无法接受 隐式 广播。
为了解决这个问题,有两个方法:
1 在Activity或其他组件中动态注册广播
2 发送显示广播
对于1 ,如果想广播让接收者工作,必须要在某个Activity或者其他组件中调用registerReceiver()进行注册,在onDestroy()时还要反注册,代码稍显复杂,而且静态注册的广播接收者仍处于不可用的状态。不合理。
而后一种方法 ,因为sendBroaccast是自己主动发送的,明显知道要哪个broadcastReceiver来进行处理,直接发送显示广播即可。
具体的代码如下,将MainActivity做修改:
public class MainActivity extends AppCompatActivity { @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); Intent intent = new Intent(); intent.setComponent(new ComponentName(MainActivity.this,MyReceiver.class));//显示指定组件名称 sendBroadcast(intent); } }
实际上,这种写法与更常见的以下写法相同:
public class MainActivity extends AppCompatActivity { @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); Intent intent = new Intent(MainActivity.this,MyReceiver.class);//显示指定组件名 sendBroadcast(intent); } }
运行,发现可以弹出Toast,静态注册的BroadcastReceiver接收到了广播。证明静态注册是可以接收到广播的。
顺便插一句:
Intent指定action, 这个Intent则为隐式Intent,使用它发送的广播则为隐式广播。隐式广播接收者是通过IntentFilter去查找的。
Intent指定了组件名称,这个Intent为显式Intent,用他发送的广播为显式广播。广播接收者直接就是指定的组件名称对应的广播接收者。
猜测显式Intent不使用IntentFilter去查找组件(Activtiy,Service,BroadcastReceiver),这点读者有兴趣可以验证是否正确。
再来看一下Intent的setComponent()方法:
Intent设置了组件名称(比如 new Intent(MainActivity.this,MyReceiver.class);)则通过IntentFilter匹配所需要的action,data,type,category信息会被忽略。
如果刚才的例子做如下改动:
public class MainActivity extends AppCompatActivity { @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); Intent intent = new Intent(); intent.setAction("com.abcdefg.hijklmn");//指定action--不存在的action intent.setComponent(new ComponentName(MainActivity.this,MyReceiver.class));//同时显式指定组件名 sendBroadcast(intent); } }
同时指定了隐式的action 以及显式的组件名称,action是一个不存在的action。这时仍可以接收到广播。
虽然通过action不可能匹配到一个广播接收者,但显示设置了组件,action就被忽略了。
扯远了。回到最初的那个话题。8.0以后,静态注册的广播接受者是可以接收到广播的,只要广播是通过显示方式发送的。