启动Activity分为两种,显示调用和隐式调用。显式调用需要明确的指定被启动对象的组件信息,包括包名和类名,而隐式调用则不需要明确指定组件信息。原则上一个Intent不应该既是显式调用又是隐式调用,如果二者共存的话以显式调用为主。隐式调用需要Intent能够匹配目标组件的IntentFilter中所设置的过滤信息,如果不匹配将无法启动目标Activity,IntentFilter中的过滤信息有action、category,data。
eg1:
![](https://img.haomeiwen.com/i3163032/6b746b10128662a6.png)
给出完全匹配它的Intent:
![](https://img.haomeiwen.com/i3163032/b8510655d7d15416.png)
注释:URI的schema是有默认值的,如果把“intent.setDataAndType(Uri.parse("file://abc"),"text/plain");”这句改成“intent.setDataAndType(Uri.parse("http://abc"),"text/plain");”,打开Activity就会报错,提示无法找到Activity。
![](https://img.haomeiwen.com/i3163032/2a76de358f436955.png)
只有一个Intent同时匹配action类别、category类别、data类别才算完全匹配,只有完全匹配才能成功启动目标Activity。另外一点,一个Activity中可以有多个intent-filter,一个Intent只要能匹配任何一组intent-filter,即可成功启动对应的Activity。下面详细分析各种属性的匹配规则。
1.action:一个过滤规则中可以有多个action,那么只要Intent中的action能够和过滤规则中的任何一个action相同即可匹配成功。如果Intent中没有指定action,那么匹配失败。总结一下,action的匹配要求Intent中的action存在且必须和过滤规则中的其中一个action相同。另外,action区分大小写,大小写不同字符串相同的action会匹配失败。
2.category:category要求Intent可以没有category,但是如果你一旦有category,不管几个,每个都要能够和过滤规则中的任何一个category相同。为什么不设置category也可以匹配成功呢?原因是系统在调用startActivity或者startActivityForResult的时候会默认Intent加上"android.intent.category.DEFAULT"这个category,同时为了我们的Activity能接收隐式调用,就必须在intent-filter中指定”android.intent.category.DEFAULT“这个category。
3.data:data的匹配规则和action类似,如果过滤规则中定义了data,那么Intent中必须也要定义可匹配的data,也就说,它也要求Intent中必须含有data数据,并且data数据能够完全匹配过滤规则中的某一个data。这里的完全匹配指的是过滤规则中出现的data部分也出现在了Intent中的data中。我们先了解下data的结构,如下所示:
![](https://img.haomeiwen.com/i3163032/e2bedf3af07750f1.png)
data由两部分组成,mimeType和URI。mimeType指的是媒体类型,比如image/jpeg、audio/mpeg4-generic和video/*等,可以表示图片、文本、视频等不同的媒体格式,而URI中包含的数据就比较多了,下面是URI的结构:
<scheme>://<host>:<port>/[<path|<pathPattern>|<pathPrefix>]
比如:content://come.example.project:200/folder/subfolder/etc
http://www.baidu.come:80/search/info
1.scheme:URI的模式,比如http、file、content等,如果URI中没有指定scheme,那么整个URI的其他的参数无效,这也意味着URI是无效的。
2.Host:URI的主机名,比如www.baidu.come,如果host未指定,那么整个URI的其他的参数无效,这也意味着URI是无效的。
3.Port:URI中的端口号,比如80,仅当URI中指定了scheme和host参数的时候port参数才有意义。
4.Path\pathPattern\pathPrefix:这三个参数表述路径信息,其中path表示完整的路径信息;pathPattern也表示完整的路径信息,但是它里面可以包含通配符“*”,“*”表示0个或多个任意字符,需要注意的是,由于正则表达式的规范,如果想表示真实的字符串,那么“*”要写成“\\*”,“\”要写成“\\\\";pathPrefix表示路径的前缀信息。
<data android:mimeType="image/*"/>
这种规则指定了媒体类型为所有类型的图片,那么Intent中的mimeType属性必须为”image/*才能匹配,这种情况虽然过滤规则没有指定URI,但是却有默认值,URI的默认值为content和file,也就是说,虽然没有指定URI,但是Intent中的URI部分的schema必须为content或者file才能匹配,这点需要尤其注意的,为了匹配上述过滤器中的规则,我们可以写出如下示例:
intent.setDataAndType(Uri.parse("file://abc"),"image/png");
另外,如果要为Intent指定完整的data,必须要调用setDataAndType方法,不能先调用setData方法在调用setType,因为这两个方法彼此会清除对方的值。关于data还有一个特殊情况需要说明一下,这也是它和action不同的地方,如下两种特殊的写法,它们的作用是一样的:
![](https://img.haomeiwen.com/i3163032/c7d3a357fc13d200.png)
Intent-filter的匹配规则对于Service和BroadcastReceiver也是同样的道理,不过系统对于Service的建议是尽量使用显示调用方式来启动服务。
最后,当我们通过隐式方式启动一个Activity的时候,可以做一下判断,看是否有Activity能够匹配我们的隐式Intent,如果不做判断,就很可能出现找不到Activity错误。判断方法有两种:采用PackageManager的resolveActivity方法或者Intent的resolveActivity方法,如果它们找不到匹配的Activity就会返回null,我们通过判断返回值就可以规避上述的错误了。
![](https://img.haomeiwen.com/i3163032/e7265007eac481a3.png)
另外,PackageManager还提供了queryIntentActivities方法,这个方法和resolveActivity方法不同的是:它不是返回最佳匹配的Activity信息,而是返回所有成功匹配的Activity信息。
public abstract List<ResolveInfo>queryIntentActivities(Intent intent,int flags);
public abstract ResolveInfo queryIntentActivity(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同样提供了类似的方法去获取成功匹配的组件信息。
参考:Android 开发艺术探索
网友评论