### 1.3 IntentFilter的匹配规则
我们知道,启动Activity分为两种,显式调用和隐式调用。二者的区别这里就不多说了,**显式调用需要明确地指定被启动对象的组件信息,包括包名和类名,而隐式调用则不需要明确指定组件信息**。原则上一个Intent不应该既是显式调用又是隐式调用,如果二者共存的话**以显式调用为主**。显式调用很简单,这里主要介绍一下隐式调用。
**隐式调用需要Intent能够匹配目标组件的IntentFilter中所设置的过滤信息,如果不匹配将无法启动目标Activity**。IntentFilter中的过滤信息有action、category、data,下面是一个过滤规则的示例:
<activity
android:name="com.ryg.chapter_1.ThirdActivity"
android:configChanges="screenLayout"
android:label="@string/app_name"
android:launchMode="singleTask"
android:taskAffinity="com.ryg.task1" >
<intent-filter >
<action android:name="com.ryg.charpter_1.c"/>
<action android:name="com.ryg.charpter_1.d"/>
<category android:name="com.ryg.category.c"/>
<category android:name="com.ryg.category.d"/>
<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**,如下所示。
<activity android:name="ShareActivity">
<! -- This activity handles "SEND" actions with text data -->
<intent-filter>
<action android:name="android.intent.action.SEND"/>
<category android:name="android.intent.category.DEFAULT"/>
<data android:mimeType="text/plain"/>
</intent-filter>
<! -- This activity also handles "SEND" and "SEND_MULTIPLE" with media
data -->
<intent-filter>
<action android:name="android.intent.action.SEND"/>
<action android:name="android.intent.action.SEND_MULTIPLE"/>
<category android:name="android.intent.category.DEFAULT"/>
<data android:mimeType="application/vnd.google.panorama360+jpg"/>
<data android:mimeType="image/*"/>
<data android:mimeType="video/*"/>
</intent-filter>
</activity>
下面详细分析各种属性的匹配规则。
#### 1.action的匹配规则
action是一个字符串,**系统预定义了一些action,同时我们也可以在应用中定义自己的action。action的匹配规则是Intent中的action必须能够和过滤规则中的action匹配,这里说的匹配是指action的字符串值完全一样。一个过滤规则中可以有多个action,那么只要Intent中的action能够和过滤规则中的任何一个action相同即可匹配成功**。针对上面的过滤规则,只要我们的Intent中action值为“`com.ryg.charpter_1.c`”或者“`com.ryg.charpter_ 1.d`”都能成功匹配。
需要注意的是,Intent中如果没有指定action,那么匹配失败。总结一下,**action的匹配要求Intent中的action存在且必须和过滤规则中的其中一个action相同,这里需要注意它和category匹配规则的不同**。另外,**action区分大小写,大小写不同字符串相同的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,但是如果你一旦有category,不管有几个,每个都要能够和过滤规则中的任何一个category相同**。为了匹配前面的过滤规则中的category,我们可以写出下面的Intent, `intent.addcategory (“com.ryg.category.c”)`或者`Intent. addcategory (“com.ryg. category.d”)`亦或者不设置category。
**为什么不设置category也可以匹配呢?原因是系统在调用startActivity或者startActivityForResult的时候会默认为Intent加上“`android.intent. category.DEFAULT`”这个category,所以这个category就可以匹配前面的过滤规则中的第三个category。同时,为了我们的activity能够接收隐式调用,就必须在intent-filter中指定“`android.intent.category.DEFAULT`”这个category,原因刚才已经说明了,也就是说,不含有DEFAULT这个category的Activity是无法接收隐式Intent的**。
#### 3.[data](https://developer.android.google.cn/guide/topics/manifest/data-element)的匹配规则
data的匹配规则和action类似,**如果过滤规则中定义了data,那么Intent中必须也要定义可匹配的data**。在介绍data的匹配规则之前,我们需要先了解一下data的结构,因为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。向 Intent 过滤器(过滤规则)添加数据规范。该规范可以是只有数据类型([mimeType](https://developer.android.google.cn/guide/topics/manifest/data-element#mime)属性),可以是只有 URI,也可以是既有数据类型又有 URI。**
**mimeType指媒体类型,比如image/jpeg、audio/mpeg4-generic和video/等,可以表示图片、文本、视频等不同的媒体格式**,而**URI中包含的数据就比较多了,下面是URI的结构**:
**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等,**如果URI中没有指定scheme,那么整个URI的其他参数无效,这也意味着URI是无效的**。
* **Host:URI的主机名**,比如`www.baidu.com`,**如果host未指定,那么整个URI中的其他参数无效,这也意味着URI是无效的**。
* **Port:URI中的端口号**,比如80,**仅当URI中指定了scheme和host参数的时候port参数才是有意义的**。
* Path、pathPattern和pathPrefix:这三个参数表述路径信息,其中
* path表示**完整的路径信息**;
* **pathPattern也表示完整的路径信息**,但是它里面可以包含通配符“`*`”, “`*`”表示0个或多个任意字符,需要注意的是,由于正则表达式的规范,如果想表示真实的字符串,那么“`*`”要写成“`\\*`”, “`\`”要写成“`\\\\`”;
* pathPrefix表示路径的前缀信息。
介绍完data的数据格式后,我们要说一下data的匹配规则了。前面说到,**data的匹配规则和action类似,它也要求Intent中必须含有data数据,并且data数据能够完全匹配过滤规则中的某一个data**.这里的**完全匹配是指过滤规则中出现的data部分也出现在了Intent中的data中**。下面分情况说明。
(1)如下过滤规则:
<intent-filter>
<data android:mimeType="image/*" />
...
</intent-filter>
**这种规则指定了媒体类型为所有类型的图片**,那么**Intent中的mimeType属性必须为“image/*”才能匹配**,这种情况下**虽然过滤规则没有指定URI,但是却有默认值,URI的默认值为content和file。也就是说,虽然没有指定URI,但是Intent中的URI部分的schema必须为content或者file才能匹配**,这点是需要尤其注意的。为了匹配(1)中规则,我们可以写出如下示例:
```
intent.setDataAndType(Uri.parse("file://abc"), "image/png")
```
另外,**如果要为Intent指定完整的data,必须要调用setDataAndType方法,不能先调用setData再调用setType,因为这两个方法彼此会清除对方的值**,这个看源码就很容易理解,比如setData:
public Intent setData(Uri data) {
mData = data;
mType = null;
return this;
}
可以发现,**setData会把mimeType置为null,同理setType也会把URI置为null**。
(2)如下过滤规则:
<intent-filter>
<data android:mimeType="video/mpeg" android:scheme="http" ... />
<data android:mimeType="audio/mpeg" android:scheme="http" ... />
...
</intent-filter>
**这种规则指定了两组data规则,且每个data都指定了完整的属性值,既有URI又有mimeType**。为了匹配(2)中规则,我们可以写出如下示例:
```
intent.setDataAndType(Uri.parse("http://abc"), "video/mpeg")
```
或者
```
intent. setDataAndType (Uri.parse("http://abc"), "audio/mpeg")
```
通过上面两个示例,读者应该已经明白了data的匹配规则,关于data还有一个特殊情况需要说明下,这也是**它和action不同的地方,如下两种特殊的写法(写法不同),但它们的作用是一样的**:
<intent-filter . . . >
<data android:scheme="file" android:host="www.baidu.com" />
. . .
</intent-filter>
<intent-filter . . . >
<data android:scheme="file" />
<data android:host="www.baidu.com" />
. . .
</intent-filter>
到这里我们已经把IntentFilter的过滤规则都讲解了一遍,还记得本节前面给出的一个intent-filter的示例吗?现在我们给出完全匹配它的Intent:
Intent intent = new Intent("com.ryg.charpter_1.c");
intent.addCategory("com.ryg.category.c");
intent.setDataAndType(Uri.parse("file://abc"), "text/plain");
startActivity(intent);
还记得URI的schema是有默认值的吗?如果把上面的`intent.setDataAndType (Uri.parse("file://abc"), "text/plain")`这句改成`intent.setDataAndType(Uri.parse("http://abc"),"text/plain")`,打开Activity的时候就会报错,提示无法找到Activity,如图1-10所示。另外一点,**Intent-filter的匹配规则对于Service和BroadcastReceiver也是同样的道理,不过系统对于Service的建议是尽量使用显式调用方式来启动服务,而且在Android8.0系统之后,所有隐式广播都不允许使用静态注册的方式来接收了。隐式广播指的是那些没有具体制定发送给哪个应用程序的广播,大多数系统广播属于隐式广播,但是极少数特殊的系统广播目前仍然允许使用静态注册的方式来接收。详见——[隐式广播例外情况](https://developer.android.google.cn/guide/components/broadcast-exceptions)**。
:-: ![](https://img.kancloud.cn/c3/0b/c30b5016d9ed7d619024166ccc08feab_1437x500.png)
图1-10 系统日志
最后,当我们**通过隐式方式启动一个Activity的时候,可以先做一下判断,看是否有Activity能够匹配我们的隐式Intent,如果不做判断就有可能出现上述的错误了。判断方法有两种:
①、采用PackageManager的resolveActivity方法或者Intent的resolveActivity方法,如果它们找不到匹配的Activity就会返回null,我们通过判断返回值就可以规避上述错误了。
②、另外,PackageManager还提供了queryIntentActivities方法,这个方法和resolveActivity方法不同的是:它不是返回最佳匹配的Activity信息而是返回所有成功匹配的Activity信息**。
我们看一下queryIntentActivities和resolveActivity的方法原型:
```
public abstract List<ResolveInfo> queryIntentActivities(Intent intent, int flags);
public abstract ResolveInfo resolveActivity(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同样提供了类似的方法去获取成功匹配的组件信息。
- 前言
- 第1章 Activity的生命周期和启动模式
- 1.1 Activity的生命周期全面分析
- 1.1.1 典型情况下的生命周期分析
- 1.1.2 异常情况下的生命周期分析
- 1.2 Activity的启动模式
- 1.2.1 Activity的LaunchMode
- 1.2.2 Activity的Flags
- 1.3 IntentFilter的匹配规则
- 第2章 IPC机制
- 2.1 Android IPC简介
- 2.2 Android中的多进程模式
- 2.2.1 开启多进程模式
- 2.2.2 多进程模式的运行机制
- 2.3 IPC基础概念介绍
- 2.3.1 Serializable接口
- 2.3.2 Parcelable接口
- 2.3.3 Binder
- 2.4 Android中的IPC方式
- 2.4.1 使用Bundle
- 2.4.2 使用文件共享
- 2.4.3 使用Messenger
- 2.4.4 使用AIDL
- 2.4.5 使用ContentProvider
- 2.4.6 使用Socket
- 2.5 Binder连接池
- 2.6 选用合适的IPC方式
- 第3章 View的事件体系
- 3.1 View基础知识
- 3.1.1 什么是View
- 3.1.2 View的位置参数
- 3.1.3 MotionEvent和TouchSlop
- 3.1.4 VelocityTracker、GestureDetector和Scroller
- 3.2 View的滑动
- 3.2.1 使用scrollTo/scrollBy
- 3.2.2 使用动画
- 3.2.3 改变布局参数
- 3.2.4 各种滑动方式的对比
- 3.3 弹性滑动
- 3.3.1 使用Scroller7
- 3.3.2 通过动画
- 3.3.3 使用延时策略
- 3.4 View的事件分发机制
- 3.4.1 点击事件的传递规则
- 3.4.2 事件分发的源码解析
- 3.5 View的滑动冲突
- 3.5.1 常见的滑动冲突场景
- 3.5.2 滑动冲突的处理规则
- 3.5.3 滑动冲突的解决方式
- 第4章 View的工作原理
- 4.1 初识ViewRoot和DecorView
- 4.2 理解MeasureSpec
- 4.2.1 MeasureSpec
- 4.2.2 MeasureSpec和LayoutParams的对应关系
- 4.3 View的工作流程
- 4.3.1 measure过程
- 4.3.2 layout过程
- 4.3.3 draw过程
- 4.4 自定义View
- 4.4.1 自定义View的分类
- 4.4.2 自定义View须知
- 4.4.3 自定义View示例
- 4.4.4 自定义View的思想
- 第5章 理解RemoteViews
- 5.1 RemoteViews的应用
- 5.1.1 RemoteViews在通知栏上的应用
- 5.1.2 RemoteViews在桌面小部件上的应用
- 5.1.3 PendingIntent概述
- 5.2 RemoteViews的内部机制
- 5.3 RemoteViews的意义
- 第6章 Android的Drawable
- 6.1 Drawable简介
- 6.2 Drawable的分类
- 6.2.1 BitmapDrawable2
- 6.2.2 ShapeDrawable
- 6.2.3 LayerDrawable
- 6.2.4 StateListDrawable
- 6.2.5 LevelListDrawable
- 6.2.6 TransitionDrawable
- 6.2.7 InsetDrawable
- 6.2.8 ScaleDrawable
- 6.2.9 ClipDrawable
- 6.3 自定义Drawable
- 第7章 Android动画深入分析
- 7.1 View动画
- 7.1.1 View动画的种类
- 7.1.2 自定义View动画
- 7.1.3 帧动画
- 7.2 View动画的特殊使用场景
- 7.2.1 LayoutAnimation
- 7.2.2 Activity的切换效果
- 7.3 属性动画
- 7.3.1 使用属性动画
- 7.3.2 理解插值器和估值器 /
- 7.3.3 属性动画的监听器
- 7.3.4 对任意属性做动画
- 7.3.5 属性动画的工作原理
- 7.4 使用动画的注意事项
- 第8章 理解Window和WindowManager
- 8.1 Window和WindowManager
- 8.2 Window的内部机制
- 8.2.1 Window的添加过程
- 8.2.2 Window的删除过程
- 8.2.3 Window的更新过程
- 8.3 Window的创建过程
- 8.3.1 Activity的Window创建过程
- 8.3.2 Dialog的Window创建过程
- 8.3.3 Toast的Window创建过程
- 第9章 四大组件的工作过程
- 9.1 四大组件的运行状态
- 9.2 Activity的工作过程
- 9.3 Service的工作过程
- 9.3.1 Service的启动过程
- 9.3.2 Service的绑定过程
- 9.4 BroadcastReceiver的工作过程
- 9.4.1 广播的注册过程
- 9.4.2 广播的发送和接收过程
- 9.5 ContentProvider的工作过程
- 第10章 Android的消息机制
- 10.1 Android的消息机制概述
- 10.2 Android的消息机制分析
- 10.2.1 ThreadLocal的工作原理
- 10.2.2 消息队列的工作原理
- 10.2.3 Looper的工作原理
- 10.2.4 Handler的工作原理
- 10.3 主线程的消息循环
- 第11章 Android的线程和线程池
- 11.1 主线程和子线程
- 11.2 Android中的线程形态
- 11.2.1 AsyncTask
- 11.2.2 AsyncTask的工作原理
- 11.2.3 HandlerThread
- 11.2.4 IntentService
- 11.3 Android中的线程池
- 11.3.1 ThreadPoolExecutor
- 11.3.2 线程池的分类
- 第12章 Bitmap的加载和Cache
- 12.1 Bitmap的高效加载
- 12.2 Android中的缓存策略
- 12.2.1 LruCache
- 12.2.2 DiskLruCache
- 12.2.3 ImageLoader的实现446
- 12.3 ImageLoader的使用
- 12.3.1 照片墙效果
- 12.3.2 优化列表的卡顿现象
- 第13章 综合技术
- 13.1 使用CrashHandler来获取应用的crash信息
- 13.2 使用multidex来解决方法数越界
- 13.3 Android的动态加载技术
- 13.4 反编译初步
- 13.4.1 使用dex2jar和jd-gui反编译apk
- 13.4.2 使用apktool对apk进行二次打包
- 第14章 JNI和NDK编程
- 14.1 JNI的开发流程
- 14.2 NDK的开发流程
- 14.3 JNI的数据类型和类型签名
- 14.4 JNI调用Java方法的流程
- 第15章 Android性能优化
- 15.1 Android的性能优化方法
- 15.1.1 布局优化
- 15.1.2 绘制优化
- 15.1.3 内存泄露优化
- 15.1.4 响应速度优化和ANR日志分析
- 15.1.5 ListView和Bitmap优化
- 15.1.6 线程优化
- 15.1.7 一些性能优化建议
- 15.2 内存泄露分析之MAT工具
- 15.3 提高程序的可维护性