Java中intentfiler_【Android - 组件】之IntentFilter的匹配规则

我们知道,Activity的启动模式分为两种,分别是显式启动和隐式启动。显式启动需要明确的指定被启动的对象的组件信息,包括包名和类名;而隐式启动需要 Intent 能够匹配目标组件的 IntentFilter 中所设置的过滤信息(原则上,一个Intent不应该既是显式调用又是隐式调用,如果二者共存,则显示调用为主)。

IntentFilter 中的过滤信息包括 action、category 和 data。为了匹配过滤列表,需要同时匹配过滤列表中的action、category和data信息,否则匹配失败。一个Activity中可以有多个IntentFilter,一个Intent只要能匹配任何一组 intent-filter,即可成功启动对应的Activity。以下是一个过滤规则的实例:

action的匹配规则

action是一个字符串。action的匹配规则是:Intent中必须有一个action且必须能够和过滤规则中的某个action匹配(这里说的匹配是指字符串值完全一样)。

【注意】

action中的内容是区分大小写的;

Intent中如果没有指定action,则视为匹配失败。

category的匹配规则

category是一个字符串。category的匹配规则是:Intent中可以没有category,但是如果一旦有category,不管有几个,每个都要能够和过滤规则中的任何一个category匹配。

【注意】:如果想让Activity A隐式启动Activity B,那么需要在Activity B的 intent-filter 中指定 android.intent.category.DEFAULT 这个category,原因是系统在调用 startActivity() 或者 startActivityForResult() 方法的时候会默认为Intent加上 android.intent.category.DEFAULT 这个category。

data的匹配规则

data的匹配规则:Intent中必须含有data数据,并且data数据能够完全匹配过滤规则中的某一个data。

先来看 data 的语法格式:

data由两部分组成: mimeType 和 URI ,URI通过如下格式,包括scheme、host、port、path、pathPrefix和pathPattern。

://:/[||]

具体的参数解释如下:

mimeType :指媒体类型,比如 image/jpeg、audio/mpeg4-generic、vidio/*等,可以表示图片、文本、视频等不同的媒体格式。

scheme :URI的模式,如http、file、content等,如果URI中没有指定scheme,那么整个URI的其他参数无效,这也意味着URI是无效的。

host :URI的主机名,如www.baidu.com,如果host未指定,那么整个URI中的其他参数无效,这也意味着URI是无效的。

port :URI中的端口号,比如80,进档URI中指定了scheme和host参数的时候,port参数才是有意义的。

path :表述路径的完整信息。

pathPrefix :表述路径的前缀信息。

pathPattern :表述路径的完整信息,但它里面可以包含通配符 * ,表示0个或任意字符(如果想要表示真是字符串,则需要转义成 \\* ; \ 要写成 \\\\ )。

我们可以通过 intent.setDataAndType(Uri.parse("URI字符串"), "mimeType字符串") 的格式为Intent设置data。

【注意】

URI可以不设置,但如果设置了,则 scheme 和 host 属性必须要设置;

URI的 scheme 属性有默认值,默认值为 content 或者 file ,因此,就算在 intent-filter 中没有为data设置URI,也需要在匹配的时候设置scheme和host两个属性,且scheme属性的值必须是content或者file;

在为Intent指定data的时候,必须要调用 setDataAndType() 方法,不能先调用 setData() 再调用 setType() ,因为这两个方法会彼此清楚对方的值;

所有有关data的属性可以放在同一个标签中,也可以分作多个标签存放,其效果相同。

总结

1、以下是在JAVA代码中匹配某个Activity的 intent-filter 的代码:

Intent intent = newIntent();

intent.addAction("my.itgungnir.action1");

intent.addCategory("my.itgungnir.category1");

intent.setDataAndType(Uri.parse("file://abc"), "text/plain");

startActivity(intent);

2、在Menifest文件的标签中,action、category和data都可以有多个;在JAVA代码中,一个Intent中只能有一个action和一个data,可以有多个category。

3、我们在通过隐式方式启动一个Activity的时候,可以做以下判断,看有没有Activity能够匹配我们的Intent,具体的判断方法有两种:

(1)使用 PackageManager 或者 Intent 的 resolveActivity() 方法,这个方法会返回最佳匹配的Activity信息,如果没有匹配的Activity,则返回null;

(2)使用 PackageManager 的 queryIntentActivities() 方法,这个方法会返回所有成功匹配的Activity的信息。

public abstract ResolveInfo resolveActivity(Intent intent, intflags);public abstract List queryIntentActivities(Intent intent, int flags);

需要说明的是这两个方法的第二个参数,我们在使用这两个方法的时候,第二个参数都必须是 Intent.MATCH_DEFAULT_ONLY ,这个参数用来匹配那些在 intent-filter 中声明了category为 android.intent.category.DEFAULT 的Activity,避免某些Activity因为没有设置category为DEFAULT而无法接收隐式Intent。

4、如果想将一个Activity标记为应用的入口,可以在其  标签中添加如下两行属性(这两行属性必须同时存在才有用,缺一不可):

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值