多应用+插件架构,代码干净,二开方便,首家独创一键云编译技术,文档视频完善,免费商用码云13.8K 广告
#### Activity调用方式 * 显示调用: 明确指定被启动对象的组件信息,包括包名和类名 * 隐式调用: 不需要明确指定组件信息,需要Intent能够匹配目标组件中的IntentFilter中所设置的过滤信息。 二者共存,以显示调用为主。显示调用很简单,这里简述一下隐式调用,隐式调用 需要intent能够匹配目标组件的IntentFilter 中所设置的过滤信息, 如果不匹配将无法启动目标Activity。IntentFilter中的过滤信息有action、category、data。 关于intent和IntentFilter的详细描述,可参考这里[意图(Intent)](https://www.kancloud.cn/alex_wsc/android/432707) #### 匹配规则 IntentFilter中的过滤信息有action、category、data。 只有一个Intent同时匹配action类别、category类别、data类别才能成功启动目标Activity。 一个Activity可以有多个intent-filter,一个Intent只要能匹配任何一组intent-filter即可成功启动对应的Activity。 1. **action** action是一个字符串,匹配是指与action的字符串完全一样,区分大小写,大小写不同字符串相同的action会匹配失败。 一个intent-filter可以有多个aciton,只要Intent中的action能够和任何一个action相同即可成功匹配。 Intent中如果没有指定action,那么匹配失败。 action的匹配要求Intent中的action存在且必须和过滤规则中的其中一个action相同 2. **category** category是一个字符串。 Intent可以没有category,但是如果你一旦有category,不管有几个,每个都必须与intent-filter中的其中一个category相同。 系统在 startActivity 和 startActivityForResult 的时候,会默认为Intent加上 android.intent.category.DEFAULT 这个category,所以为了我们的activity能够接收隐式调用,就必须在intent-filter中加上 android.intent.category.DEFAULT 这个category。 3. data data的匹配规则与action一样,如果intent-filter中定义了data,那么Intent中必须要定义可匹配的data。 intent-filter中data的语法: ``` <data android:scheme="string" android:host="string" android:port="string" android:path="string" android:pathPattern="string" android:pathPrefix="string" android:mimeType="string"/> ``` Intent中的data有两部分组成:mimeType和URI。mimeType是指媒体类型,比如 image/jpeg、audio/mpeg4-generic和video/等,可以表示图片、文本、视频等不同的媒 体格式。 URI的结构: `<scheme>://<host>:<port>/[<path>|<pathPrefix>|<pathPattern>]` 示例: ``` content://com.example.project:200/folder/subfolder/etc http://www.baidu.com:80/search/info ``` **scheme**:URI的模式,比如http、file、content等,默认值是 file 。 **host**:URI的主机名 **port**:URI的端口号 **path、pathPattern和pathPrefix**:这三个参数描述路径信息。 path、pathPattern可以表示完整的路径信息,其中pathPattern可以包含通配符 * ,表示0个或者多个任意字符。 pathPrefix只表示路径的前缀信息。 过滤规则的uri为空时,有默认值content和file,因此intent设置uri的scheme部分必须为content或file。 Intent指定data时,必须调用 setDataAndType 方法, setData 和 setType 会清除另一方的值。 对于service和BroadcastReceiver也是同样的匹配规则,不过对于service最好使用显式调用。 **隐式调用需注意** * 当通过隐式调用启动Activity时,没找到对应的Activity系统就会抛出 android.content.ActivityNotFoundException 异常,所以需要判断是否有Activity能够匹配我们的隐式Intent。 * 判断方法有2种 * 采用 PackageManager 的 resloveActivity 方法或Intent 的 resloveActivity 方法 ,如果它们找不到匹配的Activity就会返回null,通过判断返回值可以避免上述错误 * PackageManager还提供了queryIntentActivityies方法,来返回所有成功匹配的Activity信息 ``` public abstract List<ResolveInfo> queryIntentActivityies(Intent intent,int flags); public abstract ResolveInfo resloveActivity(Intent intent,int flags); ``` 以上的第二个参数使用 MATCH_DEFAULT_ONLY ,这个标志位的含义是仅仅匹配那些在 intent-filter中声明了 android.intent.category.DEFAULT 这个category的Activity。如果不用这个标记位,就会把不含这个category的Activity匹配出来了,从而导致startActivity失败,因为不含DEFAULT这个category的Activity是无法接受隐式Intent的。 * 下面的action和category比较重要,它们用来表明这是一个入口Activity,并且会出现在系统的应用列表中,二者缺一不可。 ``` <action android:name="android.intent.action.MAIN" /> <category android:name="android.intent.category.LAUNCHER" /> ```