@读书笔记
我们知道,启动Activity分为两种方式,显示调用和隐式调用。显示调用需要明确的指定被启动对象的组件信息,包括包名和类名,而隐式调用需要Intent能够匹配目标组件的IntentFilter中所设置的过滤信息,如果不匹配将无法启动目标Activity. IntentFilter中的过滤信息有
- action
- category
- data
过滤规则的示例如下:
<activity android:name="com.simon.activity.SecondActivity"
android:launchMode="singleTask"
>
<intent-filter>
<action android:name="com.simon.activity1"/>
<action android:name="com.simon.activity2"/>
<category android:name="com.simon.activity3"/>
<category android:name="com.simon.activity4"/>
<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中可以有多个intent_filter. 一个Intent只要能够匹配任何一组intent-filter即可成启动对应的Activity.
1.action的匹配规则:
action是一个字符串,系统预定了一些action,同时我们也可以在应用中定义自己的action. action的匹配规则是Intent中的action必须能够和过滤规则中的action匹配,这里说的匹配是指action的字符串的值完全一样。一个过滤规则可以有多个action,那么只要Intent中的action能够和过滤规则中的任何一个action相同即可匹配成功。 针对上面的过滤规则,只要我们的Intent中的action值为”com.simon.activity1”或“com.simon.activity2”都能成功匹配。 需要注意的是,Intent中如果没有指定action,那么匹配失败。
总结一下:
action匹配要求Intent中的action存在且必须和过滤规则中的其中一个action相同,这里需要注意它和category匹配规的不同。另外,action区分大小写,大小不同,字符相同会匹配失败。
2.category的匹配规则:
category是一个字符串,系统预定了一些category,同时我们也可以在应用中定义自己的category. category的匹配规则和Action不同,它要求Intent中如果含有category,那么所有的category都必须和过滤规则中的其中一个category相同。换句话说,Intent中如果出现了category,不管有几个category,对于每个category来说,它必须是过滤规则中已定义的category。 当然Intent中可以没有category,如果没有category的话,按照上面的描述,这个Intent仍然可以匹配成功。 这里要注意下它和action匹配过程的不同,action是要求Intent中必须有一个action且必须能够和过滤规则中的某个action相同,而category要求Intent可以没有category,但是如果有cageroy,不管有几个,每个都要能够和过滤规则中的任何一个category相同。为了匹配前面过滤规则中的category, 我们可以在Intent中添加 intent.addcategory(“com.simon.activity3”)或者 intent.addcategory(“com.simon.activity4”)亦或者不设置category.为什么不设置category也可以匹配呢? 原因是系统在调用startActivity或者startActivityForResult的时候会默认为Intent加上“android.intent.category.DEFAULT”这个category,所以这个category就可以匹配前面的过滤规则中的第三个category.同时,为了我们的activity能够接受隐式调用,就必须在Intent-filter中指定“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:pathPattern="string"
android:pathPrefix="string"
android:mimeType="string"
/>
data由两部分组成,mimeType和URI。mineType指媒体类型,比如image/jpeg, audio/mepeg4-generic 和 video/*等。可以表示图片,文本,视频等不同的媒体格式,而URI中包含的数据就比较多了。下面是URI的结构:
<scheme>://<host>:<port>/[path]|[pathPrefix]|[pathPattern]
实例:
content://com.example.project:200/folder/subfolder/etc
http://www.baidu.com:80/search/info
- Schema: URI的模式,比如http,file,content等,如果URI中没有指定scheme,那么整个URI的参数无效。这也意味着URI无效。
- Host:URI的主机名,比方www.baidu.com,如果host未指定,那么整个URI中的其他参数无效,这也意味着URI是无效的。
- Port: URI中的端口号,比如80,仅当URI中指定了scheme和host参数的是port参数才是有意义的。
path,pathPattern和pathPrefix:这三个参数表述路径信息,其中path表示完整的路径信息,但是它里面可以包含通配符“ * ”,“ * ”表示0个或者多个任意字符,需要注意的是,由于正则表达式的规范,如果想表示真实的字符串,那么“ * ”要写成“ \* ”,“\”要写成“\\”; pathPrefix表示路径的前缀信息。
介绍完data的数据格式后,来看一下data的匹配规则。 data的匹配规则和action类似,它也要求Intent中必须含有data数据,并且data数据能够完全匹配过滤规则中的某一个data,这里的完全匹配是指完全匹配是指过滤规则中出现的data部分也出现在了Intent中的data中。
如下面这个过滤规则:
<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。
再例如下面这个过滤规则:
<intent-filter>
<data android:mimeType="video/mpeg" android:scheme="http"/>
<data android:mimeType="audio/mpeg" android:scheme="http"/>
</intent-filter>
这个filter指定了两组data数据,且每个data都指定了完整的属性值,既有URI又有mimeType.为了匹配上面的这个规则,我们可以写出如下代码去匹配:
intent.setDataAndType(Uri.parse("http://abc)"),"video/mpeg");
或者 intent.setDataAndType(Uri.parse("http://abc)"),"audio/mpeg");
到这里,我们写开始的那个intentFilter的匹配规则,就很简单了:
如下:
intent.setAction("com.simon.activity1");
intent.addCategory("com.simon.activity3");
intent.setDataAndType(Uri.parse("content://abc)"),"text/plain");
隐式启动应用的时候,判断是否存在能够匹配的Activity
需要注意的是,我们使用隐式启动Activity的时候,需要先判断是否有能够匹配的Activity存在,如果不判断就直接调用的话,如果没有能够匹配的activity就会报错!
android.content.ActivityNotFoundException:NoActivity found to handle Intent
判断的方式有两种:
1.采用PackageManager的resolveActivity方法或者Intent的resolveActivity方法,如果它们找不到匹配的Activity就会返回null, 我们通过判断返回值就可以规避上述错误了。另外,PackageManager还提供了queryIntentActivities方法,这个方法和resolveActivity方法不同的是:它不是返回最佳匹配的Activity而是返回所有成功匹配的Activity信息。我们看一下queryIntentActivities和resolveActivity的方法原型:
public abstract ResolveInfo resolveActivity(Intent intent, @ResolveInfoFlags int flags);
public abstract List<ResolveInfo> queryIntentActivities(Intent intent,
@ResolveInfoFlags int flags);
第一个方法的参数是需要启动的Intent,第二个参数我们需要使用MATCH_DEFAULT_ONLY这个标记位,这个标记位的含义是仅仅匹配那些在Intent-filter中声明了
<category android:name="android.intent.category.DEFAULT"/>
这个category的activity.使用这个标记位的意义在于,只要上述两个方法不返回null,那么startActivity一定可以成功。如果不用这个标记位,就可以把intent-filter中的category不含DEDFAUlT的那些Activity也匹配出来,从而导致startActivity可能失败。