IntentFilter的匹配规则

注:内容来自任玉刚的《Android艺术探索》

启动Activity分为显示调用和隐式调用,隐式调用需要Intent能够匹配目标的IntentFilter中所设置的过滤信息,如果不匹配将无法启动目标Activity。IntentFilter的过滤信息有action、categor、data,例如:

--------------------------------------------------------------------------------------------------------------

<activity

android:name="com.xxx.XXXActivity"

android:configChanges="screenLayout"

android:label="@string/app_name"

android:launchMode="singleTask"

android:taskAffinity="com.xxx.task1" >

<intent-filter>

<action android:name="com.xxx.chapter_1.c"/>

<action android:name="com.xxx.chapter_1.d"/>

<category android:name="com.xxx.category.c"/>

<category android:name="com.xxx.category.d"/>

<category android:name="android.intent.category.DEFAULT"/>

<data android:mimeType="text/plain"/>

</intent-filter>

</activity>

------------------------------------------------------------------------------------------------------------


为了匹配过滤列表,需要同时匹配过滤列表的action、category、data信息,否则匹配失败。一个过滤列表的action、category、data可以有多个,所有的action、category、data分别构成不同的类别,同一类别的信息共同约束当前类别的匹配过程。只有一个Intent同时匹配action、category、data才算完全匹配,只有完全匹配才能成功启动Activity。另外一点,一个Activity中可以有多个IntentFilter,一个Intent只要能匹配任何一组intent-filter即可成功启动对应的Activity,例如:

--------------------------------------------------------------------------------------------------------------

<activity android:name="ShareActivity" >

<intent-filter>

<action android:name="android.intent.action.SEND"/>

<category android:name="android.intent.category.DEFAULT"/>

<data android:mimeType="text/plain"/>

</intent-filter>

<intent-filter>

<action android:name="android.intent.action.SEND"/>

<action android:name="android.intent.action.SEND_MULTIPLE"/>

<category android:name="android.intent.category.DEFAULT"/>

<data android:mimeType="application/vnd.google.panorama360+jpg"/>

<data android:mimeType="image/*"/>

<data android:mimeType="video/*"/>

</intent-filter>

</activity>

------------------------------------------------------------------------------------------------------------

1,action的匹配规则

action是一个字符串,系统预定义了一些action,同时我们也可以在应用中定义自己的action。action的匹配原则是Intent中的action必须能够和过滤规则中的action匹配,这里说的匹配是指和action的字符串值完全一样。一个过滤规则中可以有多个action,只要Intent的action能够和过滤规则中的任一action相同即可匹配成功。需要注意和category匹配的不同,action区分大小写。

2,category的匹配规则

category是一个字符串,系统预定义了一些category,同时我们也可以在应用中定义自己的category。category匹配规则和action不同,它要求Intent中如果含有category,那么所有的category都必须和过滤规则中的其中一个category相同,换句话说,Intent中如果有category过滤规则,不管有几个,这些过滤规则必须包含在某个过滤规则的category里面,否则匹配不成功。Intent也可以没有过滤规则,没有过滤规则也可以匹配成功。和action不同的是,action要求必须至少有一个cation跟过滤规则的action相同才能匹配成功,category没有这个要求,一旦Intent有了category那么这个category必须包含在过滤规则的category里面。Intent通过intent.addCategoty("xxx.xxx.xxxcategory")或者Intent.addCategory("xxx.xxx.xxxcategory")来添加category,也可以不添加,因为系统会默认添加上“android.intent.category.DEFAULT”这个category,所以这个category就可以匹配前面的过滤规则中的<category android:name="android.intent.category.DEFAULT"/>。同时为了我们的Activity能够接收隐式的调用,就必须在intent-filter里面指定<category android:name="android.intent.category.DEFAULT"/>的category。

3,data的匹配规则

data的匹配规则和action类似,如果过滤规则中定义了data,那么Intent中必须也要定义可匹配的data。data的语法如下:

------------------------------------------------------------------------------------------------------------

<data android:scheme="string"

android:host="string"

android:port="string"

android:path="string"

android:pathParttern="string"

android:pathPrefix="string"

android:mimeType="string" />

------------------------------------------------------------------------------------------------------------

data由两部分组成,mimeType和URI。mimeType指媒体类型,比如image/jpeg、audio/mpeg4-generic和video/*等,可以表示图片、文本、视频等不同类型的媒体格式,而URI中包含的数据就比较多了,下面是URI的数据结构:

------------------------------------------------------------------------------------------------------------

<scheme>://<host>:<port>/[<path>|<pathPrefix>|<pathPattern>]

例如:

contet://com.example.project:200/folder/subfolder/etc

http://www.baidu.com:80/search/info

------------------------------------------------------------------------------------------------------------

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

host:URI的主机名,比如www.baidu.com,如果host未指定,那么整个RUI中的其他参数无效,也就是URI无效

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

path、pathPrefix、pathPattern:这三个参数描述路径信息,其中path表示完整的路径;pathPattern也表示完整的路径,但是pathPattern里面可以包含通配符“ * ”,“ * ”表示0个或多个任意字符,需要注意的是由于正则表达式的规范,如果要想表达真实的字符串,那么“ * ”要写成“ \\* ”,“ \ ”要写成“ \\\\ ”;pathPrefix表示路径的前缀信息。

data过滤规则:

(1)如下:

------------------------------------------------------------------------------------------------------------

<intent-filter>

<data android:mimeType="image/*"/>

...

</intent-filter>

------------------------------------------------------------------------------------------------------------

这种规则指定了媒体类型为所有图片类型,那么Intent的mimeType必须为“image/*”才能匹配,这种情况下虽然没有指定URI,但是却有默认值,URI默认值为content和file。也就是说,虽然没有指定URI,但是Intent中的URI部分的scheme必须为content或者file才能匹配,这点是需要尤其注意的。例如:

intent.setDataAndType(Uri.parse("file://abc"), "image/png");

另外,如果要为Intent指定完整的data,必须要调用setDataAndType方法,不能先用setData再用setType,因为这两个方法彼此会清除对方的值,setData的源码:

public Intent setData(Uri data){

mData = data;

mType = null;

return this;

}

(2)如下:

------------------------------------------------------------------------------------------------------------

<intent-filter>

<data android:mimeType="video/mpeg" android:scheme="http" .../>

<data android:mimeType="audio/mpeg" android:scheme="http" .../>

...

</intent-filter>

------------------------------------------------------------------------------------------------------------

示例:

intent.setDataAndType(Uri.parse("http://abc"), "video/mpeg");

或者

intent.setDataAndType(Uri.parse("http://abc"), "audio/mpeg");

跟action不同的地方,下面两种写法作用是一样的:

<intent-filter ...>

<data android:scheme="file" android:host="www.baidu.com"/>

...

</intent-filter>

<intent-filter ...>

<data android:scheme="file"/>

<data android:host="www.baidu.com"/>

...

</intent-filter>


现在给出一个文章开头intent-filter示例匹配的Intent:

Intent intent = new Intent("com.xxx.chapter_1.c");

intent.addCategory("com.xxx.category.c");

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

startActivity(intent);

如果把Uri.parse("file://abc")改成Uri.parse("http://abc")就会报错,提示无法找到Activity。另外,Intent-Filter的匹配规则对于Service和BroadCase也是同样的道理,不过系统对于Service的建议是尽量使用显示调用来启动服务。


最后,当隐式启动一个Activity的时候,可以先判断一下是否有Activity能够匹配我们的隐式Intent。

第一种方法:采用PackageManager的resolveActivity方法或者Intent的resolveActivity方法,如果找不到匹配的Activity会返回null。

第二种方法:采用PackageManager的queryIntentActivities方法,这个方法和resolveActivity方法不同的是,它不是返回最佳匹配的Activity信息而是返回所有成功匹配的Activity信息。

public abstract List<ResolveInfo> queryIntentActivities(Intent intent, int flags) ;

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

两个方法的第一个参数比较好理解,第二个参数要使用MATCH_DEFAULT_ONLY这个标记位,含义是仅仅匹配那么在Intent-filter中声明了<category android:name="android.intent.category.DEFAULT"/>这个category的Activity。使用这个标记位的意义在于,只要上述两个方法不返回null,那么startActivity一定能成功。如果不使用这个标记位,就可以把intent-filter中category不含DEFAULT的那些Activity给匹配出来,从而导致startActivity可能失败。因为不含DEFAULT的这个category的Activity是无法接收隐式Intent的。在action和category中,有一类action和category比较重要,他们是:

<action android:name="android.intent.action.MAIN"/>

<category android:name="android.intent.category.LAUNCHER"/>

这二者共同的作用是用来表明这是一个入口Activity并且会出现在系统的应用列表中,少了任何一个都没有实际意义,也无法出现在系统的应用列表中,也就是二者缺一不可。另外针对Service和BroadCastReceiver,PackageManager同样提供了类似的方法去获取成功匹配的组件信息。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值