前言
activity的启动分为显式调用和隐式调用。
- 显式调用需要明确指定组件的信息。
- 隐式调用是通过匹配目标组件的IntentFilter
这里是记录一下如何使用匹配目标组建隐式启动Activity
正文
通过组建匹配,可以再不知道应用包名的情况启动对应的Activity(或者说应用),一个组件可以设置多个IntentFilter来匹配。
IntentFilter中的过滤信息有action、category、data组成。
- 只有一个Intent同时匹配action类别,category类别,data类别才算完全匹配,才能启动对应的组件。
- 一个intent只需要匹配任何一组intentFilter就可以启动对应组件。
action的匹配规则
action是一个字符串,一个过滤规则(intentFilter)可以有多个action。一个IntentFilter必须有一个action,如果没有那么则匹配失败。多个action只要匹配成功一个即可匹配成功。
PS:字符串值完全一样,区分大小写
比如: AndroidManifest.xml中Activity的配置
<activity
android:name="com.biumall.media.ImageActivity">
<intent-filter>
<action android:name="com.biumall.action.PLAY_IMAGE_1" />
<category android:name="android.intent.category.DEFAULT" />
</intent-filter>
<intent-filter>
<action android:name="com.biumall.action.PLAY_IMAGE_2" />
<category android:name="android.intent.category.DEFAULT" />
</intent-filter>
</activity>
下面方式两种都可以启动上面的Activity
Intent intent = new Intent();
intent.setAction("com.biumall.action.PLAY_IMAGE_1");
intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
startActivity(intent);
或者
Intent intent = new Intent();
intent.setAction("com.biumall.action.PLAY_IMAGE_2");
intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
startActivity(intent);
category的匹配规则
category是一个字符串,Intent可以没有category,但是如果一旦有category,不管有几个,每一个都要能够和过滤规则中的任何一个category相同。
简而言之:Intent中有N个,AndroidManifest.xml中Activity的匹配规则可以N+1个,但这个N+1个中要[完全匹配]Intent中的N个category。
PS: 系统在调用startActivity或者startActivityForResult时,会默认为Intent添加“android.intent.category.DEFAULT"这个category。
在[action的匹配规则]中展示的,虽然启动代码中没有category,但AndroidManifest.xml中必须添加:
<category android:name="android.intent.category.DEFAULT" />
data的匹配规则
Data一般由两部分组成mimeType和URI。
MimeType
mineType指资源类型包括文本、图片、音视频等等。可以表示图片、文本、音视频等等不同的数据格式。
常见文件格式的mimeType:
<!--常见MimeType-->
<!--音视频-->
<data android:mimeType="video/*" />
<data android:mimeType="audio/*" />
<!--图片-->
<data android:mimeType="image/*" />
<!--文本-->
<data android:mimeType="text/*" />
<!--*、bin、class dms、exe等-->
<!--<data android:mimeType="application/octet-stream" />-->
<!--pdf-->
<data android:mimeType="application/pdf" />
<!--prf-->
<data android:mimeType="application/pics-rules" />
<!--pot pos ppt-->
<data android:mimeType="application/vnd.ms-powerpoint" />
<!--js-->
<data android:mimeType="application/x-javascript" />
<!--zip-->
<data android:mimeType="application/zip" />
<!--rar-->
<data android:mimeType="application/rar" />
<!--rar-->
<data android:mimeType="application/x-rar-compressed" />
<!--jar-->
<data android:mimeType="application/ava-archive" />
<!--tar-->
<data android:mimeType="application/x-tar" />
<!--tgz-->
<data android:mimeType="application/x-compressed" />
<!--zip-->
<data android:mimeType="application/x-zip-compressed" />
<!--ppt-->
<data android:mimeType="application/vnd.ms-powerpoint" />
<!--xls-->
<data android:mimeType="application/vnd.ms-excel" />
<!--doc、dot-->
<data android:mimeType="application/msword" />
<!--docx-->
<data android:mimeType="application/vnd.openxmlformats-officedocument.wordprocessingml.document" />
<!--potx-->
<data android:mimeType="application/vnd.openxmlformats-officedocument.presentationml.template" />
<!--pptx-->
<data android:mimeType="application/vnd.openxmlformats-officedocument.presentationml.presentation" />
<!--xlsx-->
<data android:mimeType="application/vnd.openxmlformats-officedocument.spreadsheetml.sheet" />
<!--xltx-->
<data android:mimeType="application/vnd.openxmlformats-officedocument.spreadsheetml.template" />
<!--ppsx-->
<data android:mimeType="application/vnd.openxmlformats-officedocument.presentationml.slideshow" />
<!--dotx-->
<data android:mimeType="application/vnd.openxmlformats-officedocument.wordprocessingml.template" />
如果不知道指定文件的mimeType的话,可以通过一下方法获取指定文件的mimeType。
private static String getMimeType(String filePath) {
String ext = MimeTypeMap.getFileExtensionFromUrl(filePath);
return MimeTypeMap.getSingleton().getMimeTypeFromExtension(ext);
}
URI
一个IntentFilter中可以有多个data。匹配过程中如果intentFilter含有data那么intent中必须含有data并且只要其中某一个匹配成功即可认为intent匹配成功。
URI具体结构如下:
scheme://host:port/(path|pathPrefix|pathPatten)
示例如下:
<data
android:scheme="string"
android:host="string"
android:port="1"
android:path="/string"
android:pathPattern="string"
android:pathPrefix="/string"
android:mimeType="string" />
对应含义:
- Scheme:URI的模式。即指定的URI的格式。比如http、content、file等,如果没有指定scheme那么后面的参数都无效,即整个URI都无效。
- Host:URI的主机名。如www.baidu.com等,如果host未指定那么这个URI也是无效的。
- port:端口号,仅当指定了scheme和host的时候port才能起作用。
- path:完整的路径信息。
- pathPattern:也表示完整的路径信息,但是可以包含通配符“",”"表示0或多个等任意字符。要注意正则表达式的规范。
- pathPrefix:路径前缀信息。
PS: 如果同时存在URI、miemType调用如下方法,不能单独设置,因为单独设置会只对应的null
intent.setDataAndType(Uri.parse(""),"image/*");
隐式调用注意事项
隐式调用启动组件之前,根据相应的隐式条件信息查询一下能否匹配到对应的组件。防止直接隐式调用启动,发生错误,找不到对应的组件。
判断方法分为两种:
- 通过Intent的intent.resolveActivity(getPackageManager());方法查询。
Intent intent=new Intent();
intent.setDataAndType(Uri.parse("http://demo"),"image/*");
intent.resolveActivity(getPackageManager());
- 通过PackageManager的resolveActivity(Intent intent, @ResolveInfoFlags int flags);方法查询。
Intent intent=new Intent();
intent.setDataAndType(Uri.parse("http://demo"),"image/*");
intent.resolveActivity(getPackageManager());
PackageManager packageManager = getPackageManager();
packageManager.resolveActivity(intent,PackageManager.MATCH_DEFAULT_ONLY);
问题举例
启动代码:
Intent intent = new Intent();
intent.setAction("com.biumall.action.PLAY_IMAGE_1");
intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
intent.setData(uri);
startActivity(intent);
AndroidManifest.xml配置
<intent-filter>
<action android:name="com.biumall.action.PLAY_IMAGE_1" />
<category android:name="android.intent.category.DEFAULT" />
</intent-filter>
错误提示日志:
android.content.ActivityNotFoundException: No Activity found to handle Intent { act=com.biumall.action.PLAY_IMAGE_1 dat=/storage/udisk2/125la_8.jpg (has extras) }
at android.app.Instrumentation.checkStartActivityResult(Instrumentation.java:2007)
at android.app.Instrumentation.execStartActivity(Instrumentation.java:1673)
at android.app.Activity.startActivityForResult(Activity.java:4586)
at androidx.fragment.app.FragmentActivity.startActivityForResult(FragmentActivity.java:792)
at android.app.Activity.startActivityForResult(Activity.java:4544)
at androidx.fragment.app.FragmentActivity.startActivityForResult(FragmentActivity.java:779)
at android.app.Activity.startActivity(Activity.java:4905)
at android.app.Activity.startActivity(Activity.java:4873)
at com.biumall.media.utils.MediaUtils.startImageActivity(MediaUtils.java:84)
上面意思就是没有找到能跟启动代码匹配的Activity。
以下是修改后验证通过的配置。
Intent intent = new Intent();
intent.setAction("com.biumall.action.PLAY_IMAGE_1");
intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
intent.setDataAndType(uri, "image/*");
startActivity(intent);
<intent-filter>
<action android:name="com.image.action.PLAY_FILE" />
<category android:name="android.intent.category.DEFAULT" />
<data android:mimeType="image/*" />
</intent-filter>
参考文章