这个问题,一般都会问到,属于高频问题了。做了Android这么多年我也没有弄得很清楚,那这次我们就来仔仔细细的来理一理。
IntentFilter
,顾名思义,就是Intent
的过滤器。回想一下,好像Intent
在我们的代码中出现的频率还是挺高的,是不是也没有特别的去理解它。
先来聊一下Intent
Intent
的意思是意图
, 而就和它的意思差不多,每当我们使用 Intent
的时候,总是去想干一些事情:
startActivity(Intent intent)
startService(Intent intent)
bindService(Intent intent)
sendBrodcast(Intent intent)
.......
对的,我们再很多地方都会使用Intent
。对于这些请求,我们都会传入一个Intent
,用来Filter
并启动相应的Activity
、Service
、BroadcastReceiver
。而在这里,我们就有两种调用方式:显示调用和隐式调用。
显式调用
什么是显示调用呢?那我们用一个启动Activity的例子来说明一下吧:
Intent _Itent = new Intent();
_Itent.setClass(ActivitA.this, ActivityB.class);
startActivity(_Itent);
哎,原来这是显示调用呀。之说以叫做显式调用,我们为Intent
清楚的指出了被启动组件的信息(这里就是ActivityB
),当调用了startActivity(itent)
后,我们就只会很明确的知道,这次的任务是启动ActivityB
,而没有其它的过程。
隐式调用
知道了显式调用,应该很容易猜得到了,隐式调用就是没有明确的指出组件信息。而是通过Filter
去过滤出需要的组件。还是来个例子说明一下吧:
Intent _Intent = new Intent();
_Intent.setAction(Intent.ACTION_BATTERY_LOW);
_Intent.addCategory(Intent.CATEGORY_APP_EMAIL);
_Intent.setDataAndType(Uri.EMPTY, "video/mpeg");
startActivity(_Intent);
这里就是一个隐式的调用,可以看到我为Intent
设置了三个属性Action
、Category
、Data
。然后startActivity(intent)
就会根据我们设置的这三个属性去筛选合适的组件来打开,也就是因为这样,所以有时候,当我们APP来分享一个东西的时候,会有很多组件(比如QQ、微信、微博...)来供我们选择,因为他们都满足Filter
条件。或许你还有许多疑问,来我们就接着来看IntentFilter
吧!
再来聊一下IntentFilter及他的匹配规则
IntentFilter
的意思就是意图过滤器
,当我们隐式的启动系统组件的时候,就会根据IntentFilter
来筛选出合适的进行启动。现在我们知道了可以在Intent
启动的时候对应设置Action
、Category
、DataAndType
,这里设置的是为了过滤的时候对应IntentFilter
匹配action
、category
、data
。除开过滤广播的的IntentFilter
可以在代码中创建外,其它的IntentFilter
都得在AndroidManifest.xml
中给设置。
Reciver _Reciver = new Reciver();
IntentFilter _IntentFilter = new IntentFilter();
_IntentFilter.addAction(Intent.ACTION_BATTERY_LOW);
_IntentFilter.addCategory(Intent.CATEGORY_APP_EMAIL);
_IntentFilter.addDataType("video/mpeg");
registerReceiver(_Reciver, intentFilter);
这里就是在代码中设置IntentFilter
,可以看到我们设置了三个属性。让后我们再看看在AndroidManifest.xml
的设置方法:
<activity android:name=".ActivityB"
android:label="@string/ActivityB"
android:launchMode="singleInstance">
<intent-filter>
<action android:name="android.intent.action.ANSWER" />
<category android:name="android.intent.category.APP_EMAIL" />
<data android:host="www.aoaoyi.com"
android:port="80"
android:scheme="http" />
</intent-filter>
</activity>
我们在这里给Activity
设置了一个IntentFilter
,但是值得注意的是,一个组件可以有多个IntentFilter
,在过滤的时候只要有一个符合要求的,就会被视为过滤通过。
那我们就看看是怎样过滤
的吧,首先我们应该明白一个大的思路:当我们隐式的启动一个组件的时候,就会一个一个的去过滤对应组件的全部,(比如你是隐式的启动一个Activity
,就会一个一个的在全部Activity
中筛选),然后根据Intent
的所设置的action
、category
、data
去比较IntentFilter
所设置的这三个属性,相同的话就过滤留下来了。
action的匹配
action的匹配要求Intent中的action存在且必须和过滤规则中的其中一个action相同,区分大小写。
首先,action
是一个字符串,匹配的话就是说两个action
的字符串完全相同(Intent和IntentFilter中的action
)。然后我们就看看具体的匹配方法:
。如果IntentFilter
中有action,Intent
中必须有action
。Intent
中的action
必须在相应IntentFilter
中存在
。Intent
中只需要有一个action
和IntentFilter
中相同即可
IntentFilter
中可以设置多个action
,Intent
中也可以设置多个action
,这里就是说我们Intent
中的action
必须存在IntentFilter
中,但是Intent
中不必包括IntentFilter
中全部的action
,但是至少包括一个。
category的匹配
category 要求Intent中可以没有category,但是你一旦有category,不管几个,每个都要和IntentFilter中的category相同。
这里我们说Intent
中可以没有category
,其实不然,只是在我们启动组件(eg:startActivity( ))的时候,默认给我们的Intent
给加了一个category
("android.intent.category.DEFAULT" ).
嗯,我们知道了这里,那么匹配就和action
差不多了,就是我们的Intent
中的category
必须在IntentFilter
中存在。这里得注意,Intent
中都会包括默认的category
,并且如果你想隐式启动某个组件,那么就得在IntentFilter中添加android.intent.category.DEFAULT
这个category
才行。
data的匹配
如果IntentFilter中有定义data,那么Intent中也须也要定义可以匹配的data,平且data数据能够完全比配过滤规则中的某一个data。
聊了上面的两个的匹配规则,发现其实还是很有规律的,对的,data
的匹配也差不多(其实我认为是一样的,只是data
的结构要复杂些)。
<data android:mimeType="text/plain"
android:host="www.aoaoyi.com"
android:path="/myfolder/my.txt"
android:pathPattern="/myfolder/*"
android:port="80"
android:scheme="http" />
但其实一个data主要包括的就是一个URI和mimeType。mimeType
就是媒体类型,就像"text/plain"
这样的,可以表示data是图片呀、文本呀、视频等等。其它的就是URI
的了,简单点,就是除开mimeType
,剩下的全部都是属于URI
的,它们组成了URI
。而URI
中属性就特别容易懂了,就像host
指的是主机名、Scheme
指的是URI
的模式、Port
指的端口号......
在Inten
中,我们通过setDataAndType(Uri data, String type)
方法对date
进行设置。这个方法接受两个参数,第一个就是URI
,第二个就是String
类型的mimeType
了,通过这一个方法,我们就可以给Intent
设置data
了。、
Scheme:URI的模式,比如Http、file、content等,如果URI中没有指定scheme,那么整个URI的其他参数无效,这也意味着URI是无效的。
Host:URI的主机名,比如www.aoaoyi.com,如果host没有指定,那么整个URI中的其他参数无效,这也意味着URI是无效的。
Port:URI中的端口号,比如80,仅当URI中指定了scheme和host参数的时候port参数才会有意义。
Path、pathPattern、pathPrefix:这三个参数表述路径信息,其中path表示完整的路径信息;pathpatten也表示完整路径信息,但是它的里面可以包含通配符“*”,“*”表示0或多个任意字符,需要注意的是,由于正则表达式的规范,如果想表示真实的字符串,那么“*”要写成“\\*”,“\”要写成“\\\\”;pathPrefix表示路径的前缀信息。
关于data还有一个特殊的情况需要说一下,这也是和action不同的地方,如下两种特殊的写法,他们的作用是一样的。
<intent-filter>
<data android:scheme="file" android:host="www.aoaoyi.com" />
......
</intent-filter>
<intent-filter>
<data android:scheme="file" />
< android:host="www.aoaoyi.com" />
......
</intent-filter>
隐式启动的判断
当我们隐式启动的时候,获取无法过滤找到所需要的组件,这样的话就会报错了。那么有没有什么方法来判断隐式启动的Intent
是否能找到相应的组件呢?
1、通过PackageManager
来判断
Intent _Intent = new Intent();
_Intent.setAction(Intent.ACTION_BATTERY_LOW);
_Intent.addCategory(Intent.CATEGORY_APP_EMAIL);
_Intent.setDataAndType(Uri.EMPTY, "video/mpeg");
PackageManager _PackageManager = getPackageManager();
List<ResolveInfo> _InfoList = _PackageManager.queryIntentActivities(_Intent, PackageManager.MATCH_DEFAULT_ONLY);
Log.i(TAG, _InfoList.size() + "");
if (_InfoList.size() != 0) {
startActivity(_Intent);
} else {
Log.e(TAG, "没有匹配到Activity");
}
2、通过Intent
的方法:
Intent _Intent = new Intent();
_Intent.setAction(Intent.ACTION_BATTERY_LOW);
_Intent.addCategory(Intent.CATEGORY_APP_EMAIL);
_Intent.setDataAndType(Uri.EMPTY, "video/mpeg");
PackageManager _PackageManager = getPackageManager();
ComponentName _Name = intent.resolveActivity(getPackageManager());
if (_Name != null) {
startActivity(_Intent);
} else {
Log.e(TAG, "没有匹配到Activity");
}
Android面试题6–BroadcastReceiver(广播)
文章评论
That's a slick answer to a chnalengilg question
This is way more helpful than annhityg else I've looked at.
This is good. Thanks!
This is good. Thanks!