前言
好久没写APP了,关于Intent和<intent-filter>的匹配规则,忘了很多,索性好好读了下官方稳定,总结(翻译)如下。
1. Action 解析规则
1.1 intent-filter 填写规则
每个
<intent-filter> </intent-filter>对中,可以指定0个到N个<action>元素例如:
<intent-filter> <action android:name="android.intent.action.EDIT" /> <action android:name="android.intent.action.VIEW" /> ... </intent-filter>
1.2 解析规则
- 如果一个Intent中指定了action项
- 如果这个action至少匹配
<intent-filter> xxx </intent-filter>中的一个action,则匹配通过,否则失败; - 如果
<intent-filter> xxx </intent-filter>中没有任何action项,则直接匹配失败
- 如果这个action至少匹配
- 如果一个Intent中没有指定任何action项
- 如果
<intent-filter> xxx </intent-filter>中至少一个action,则匹配通过
- 如果
1.3 特殊说明
如果使用的是隐式Intent,安卓会自动在调用startActivity()和startActivityForResult()接口的时候,自动在intent对象中添加<category android:name="android.intent.category.DEFAULT" />声明。所以,在这种情况下,如果你想让你的activity接收这些intent对象,就必须在其<intent-filter> </intent-filter>中添加<category android:name="android.intent.category.DEFAULT" />。
例如
<activity android:name=".test.ex01_intent.test_ex01_activity1" android:label="@string/title_activity_test_ex01_activity1"> <intent-filter> <action android:name="myaction.test.ex01.activity1" />/> <category android:name="android.intent.category.DEFAULT" /> </intent-filter> </activity>调用代码如下:
mBtn01.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View view) { Intent intent = new Intent("myaction.test.ex01.activity1"); startActivity(intent); } });
2. Category 解析规则
2.1 intent-filter 填写规则
每个
<intent-filter> </intent-filter>对中,可以指定0个到N个<category>元素例如:
<intent-filter> <category android:name="android.intent.category.DEFAULT" /> <category android:name="android.intent.category.BROWSABLE" /> ... </intent-filter>
2.2 解析规则
- 如果一个Intent中指定了1到N个category项
- 则这个intent对象中的1到N个category,在
<intent-filter> xxx </intent-filter>中,需要全有,一一匹配则通过,否则失败;- 注意:
<intent-filter>中的项目可以比intent对象中的category多,只要intent中的全都匹配到,就可以通过
- 注意:
- 则这个intent对象中的1到N个category,在
- 如果一个Intent中没有指定任何category项
- 则,无论
<intent-filter> xxx </intent-filter>中有多少个category,直接匹配通过
- 则,无论
2.3 特殊说明
- 如果使用的是隐式Intent,安卓会自动在调用startActivity()和
startActivityForResult()接口的时候,自动在intent对象中添加<category android:name="android.intent.category.DEFAULT" />声明。所以,在这种情况下,如果你想让你的activity接收这些intent对象,就必须在其<intent-filter> </intent-filter>中添加<category android:name="android.intent.category.DEFAULT" />,具体可参考楼上action 1.3节。
3. Data 项解析规则
3.1 intent-filter 填写规则
每个
<intent-filter> </intent-filter>对中,可以指定0个到N个<data>元素例如:
<intent-filter> <data android:mimeType="video/mpeg" android:scheme="http" ... /> <data android:mimeType="audio/mpeg" android:scheme="http" ... /> ... </intent-filter>每个
<data>项目都可以指定一个URI和一个数据类型。每个URI都可包含
scheme,host,port,path这几属性,其格式为<scheme>://<host>:<port>/<path>
举例:
content://com.example.project:200/folder/subfolder/etc<scheme>为 content<host>为com.example.project<port>为200<path>为 folder/subfolder/etc
注意:每个
<data>项目中的URI,scheme,host,port,path这几个属性,不是全都需要指定,但有一个线性的依赖要求:- 如果
<scheme>没有被指定,则后面的<host>被忽略。 - 如果
<host>没有被指定,则后面的<port>被忽略。 - 如果
<scheme>和<host>都没有被指定,则后面的<path>被忽略。
- 如果
3.2 data 匹配规则
当Intent中的URI与
<intent-filter>中的一个URI相对比的时候:- 如果
<intent-filter>中的这个URI只有一个<scheme>,则Intent对象中拥有相同<scheme>的URI,全部通过。 - 如果
<intent-filter>中的这个URI指定了<scheme>和<authority>,但是没有指定<path>,则Intent对象中拥有相同<scheme>和<authority>的URI,不管其<path>是什么值,全部通过。 - 如果
<intent-filter>中的这个URI指定了<scheme>、<authority>和<path>,则Intent对象中拥有相同<scheme>、<authority>和<path>的URI,才能通过。 - 注意:
<path>中可以使用*通配符
- 如果
数据测试会将 Intent 中的 URI 和 MIME 类型与
<intent-filter>中指定的 URI 和 MIME 类型进行比较。规则如下:仅当
<intent-filter>未指定任何 URI 或 MIME 类型时,不含 URI 和 MIME 类型的 Intent 才会通过测试。对于包含 URI 但不含 MIME 类型(既未显式声明,也无法通过 URI 推断得出)的 Intent,仅当其 URI 与
<intent-filter>的 URI 格式匹配、且<intent-filter>同样未指定 MIME 类型时,才会通过测试。仅当
<intent-filter>列出相同的 MIME 类型且未指定 URI 格式时,包含 MIME 类型但不含 URI 的 Intent 才会通过测试。仅当 MIME 类型与
<intent-filter>中列出的类型匹配时,同时包含 URI 类型和 MIME 类型(通过显式声明,或可以通过 URI 推断得出)的 Intent 才会通过测试的 MIME 类型部分。如果 Intent 的 URI 与<intent-filter>中的 URI 匹配,或者如果 Intent 具有content:或file:URI 且<intent-filter>未指定 URI,则 Intent 会通过测试的 URI 部分。换言之,如果过滤器只是列出 MIME 类型,则假定组件支持content:和file:数据。**注意:**如果 Intent 指定 URI 或 MIME 类型,则数据测试会在
<intent-filter>中没有<data>元素时失败。
4. 例子:intent-filter 同时包含 action、category、data
AndroidManifest
- 如下AndroidManifest ,在1.3的基础上,添加了
<data>
<activity
android:name=".test.ex01_intent.test_ex01_activity1"
android:label="@string/title_activity_test_ex01_activity1">
<intent-filter>
<!-- <action android:name="android.intent.action.MAIN" />-->
<action android:name="myaction.test.ex01.activity1" />/>
<category android:name="android.intent.category.DEFAULT" />
<data android:mimeType="video/mpeg" android:scheme="http" />
</intent-filter>
</activity>
(1)测试代码:启动activity 失败
mBtn01.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
Intent intent = new Intent("myaction.test.ex01.activity1");
startActivity(intent);
}
});
(2)测试代码:启动activity 失败
mBtn01.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
Intent intent = new Intent(mMyAction);
intent.setType("video/mpeg");
startActivity(intent);
}
});
(3)测试代码:启动activity 失败
mBtn01.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
Intent intent = new Intent(mMyAction);
intent.setType("video/mpeg");
intent.setData(Uri.parse("http"));
startActivity(intent);
}
});
原因如下:
- intent.setData(Uri.parse(“http”));语句中scheme部分未识别到
- 若要识别,需改为intent.setData(Uri.parse(“http://”));

(4)测试代码:启动activity 成功
mBtn01.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
Intent intent = new Intent(mMyAction);
intent.setType("video/mpeg");
intent.setData(Uri.parse("http://"));
startActivity(intent);
}
});
总结
从下面几个实验可以看到,当同时包含这三项,则需要三项同时依据各自匹配规则,三项均通过才行