Android中的Intent Filter匹配规则简介
本文主要介绍了隐式intent匹配目标组件的规则,若有叙述不清晰或是不准确的地方希望大家指出,谢谢大家: )
1. intent简介
intent用于在一个组件(component,如activity、service、broadcast receiver)中打开另一个组件。
intent可分为隐式(implicitly)和显式(explicitly)两种:
explicitly intent:在知道要打开哪个具体的component时使用,通过指定调用者和被调用者即可打开目标component;
implicitly intent:在不确切的知道要打开哪个component的情况下,通过指出action、data、category,系统会寻找到匹配的component。
(1)explicitly intent
当明确知道你想打开哪个component时,它就是你的菜。通常这样使用:
intent intent = new intent(this, mainactivity.class); intent.putextra("key", "value"); startactivity(intent);
执行以上代码会导致目标component(这里是mainactivity)被创建(oncreate等一系列生命周期方法被调用)。在mainacitivity中的相应生命周期方法中通过getintent.getxxxextra(“key”)即可得到随intent一起传过来的数据。
(2)implicitly intent
implicitly intent很好的实现了调用者和被调用者之间的解耦:
调用者通过action、data、category这三个方面描述他的intent,被调用者通过在manifest文件中声明的一系列intent filter来描述自己能够响应哪些意图。如此一来,调用者和被调用者无需互相了解,通过implicitly intent这个联系他们的纽带就能很好的协同工作。
关于intent更加详细的介绍,大家可以参考官方文档,这里主要介绍下implicitly intent的匹配规则。
2.intent filter匹配规则
只有action、data、category三方都匹配,intent才算是匹配成功,进而才能打开相应的component。一个component若声明了多个intent filter,只需要匹配任意一个即可启动该组件。
(1)action的匹配规则
一个intent filter中可声明多个action,intent中的action与其中的任一个action在字符串形式上完全相同(注意,区分大小写),action方面就匹配成功。可通过setaction方法为intent设置action,也可在构造intent时传入action。需要注意的是,隐式intent必须指定action。比如我们在manifest文件中为myactivity定义了如下intent filter:
<intent-filter> <action android:name="android.intent.action.send"/> <action android:name="android.intent.action.send_to"/> </intent-filter>
那么只要intent的action为“send”或“send_to”,那么这个intent在action方面就能和上面那个activity匹配成功。比如我们的intent定义如下:
intent intent = new intent("android.intent.action.send")
...
那么我们的intent在action方面就与myactivity匹配了。
android系统预定义了许多action,这些action代表了一些常见的操作。常见action如下(intent类中的常量):
intent.action_view
intent.action_dial
intent.action_sendto
intent.action_send
intent.action_web_search
(2)data的匹配规则
data可进一步分为uri(由scheme、host、port、path | pathpattern | pathprefix这4部分组成)和mimetype。intent的uri可通过setdata方法设置,mimetype可通过settype方法设置。隐式intent也必须指定data。同action类似,只要intent的data只要与intent filter中的任一个data声明完全相同,data方面就匹配成功。需要注意的是:若intent filter的data声明部分未指定uri,则缺省uri为content或file,intent中的uri的scheme部分需为content或file才能匹配;若要为intent指定完整的data,必须用setdataandtype方法,原因请看setdata和settype方法的源码:
public intent setdata(uri data) { mdata = data; mtype = null; return this; } public intent settype(string type) { mdata = null; mtype = type; return this; }
从以上代码可以看到,setdata会把mimetype置为null,settype会把uri置为null。下面我们来举例说明一下data的匹配。首先我们先来看一下intent filter中指定data的语法:
<data android:scheme="...“ android:host="..." android:port="..." android:path="..." android:pathpattern="..." android:pathprefix="..." android:mimetype="..." />
其中scheme、host等各个部分无需全部指定。假如我们为myactivity的intent filter指定了如下data:
<intent-filter> <data android:mimetype="vidoe/mpeg" android:scheme="http" android:host="www.xxx.com" /> <data android:mimetype="text/plain" android:scheme="http" /> </intent-filter>
那么我们的intent想要匹配,mimetype可以为”text/plain”或“video/mpeg”,scheme必须为”http“,host则没有限制,因为第二个data没有指定host。
(3)category的匹配规则
与action和data不同,intent中的category必须都在intent filter中出现才算匹配成功。intent可以不指定category,若intent中未指定category,系统会自动为它带上“android.intent.category.default”。所以,想要接收implicitly intent的component都必须在manifest文件中的intent filter声明中带上“android.intent.category.default”。我们可以通过addcategory方法为intent添加category。
(4)查询是否有可接收指定intent的component
采用packagemanager的resolveactivity或者intent的resolveactivity方法会获得最适合intent的一个activity;调用packagemanager的queryintentactivities会返回所有成功匹配intent的activity。关于这几个方法的详细定义大家可以参考官方文档,这里不再赘述。
以上就是本文的全部内容,希望对大家的学习有所帮助。
推荐阅读