《Android开发艺术探索》读书笔记----第一章:IntentFilter匹配规则

Activity 的 Flags

Activity 的 Flags 有很多,这里主要分析一些常用的标记位,标记位的作用有很多,有的标记为可以设置启动模式,有的可以影响 Activity 的运行状态,对于标记位理解即可,有些标记位是系统内部使用的,应用程序不需要手动设置这些标记位以防出现问题。

  • FLAG_ACTIVITY_NEW_TASK
    这个标记位是为 Activity 指定 singleTask 的启动模式,其效果和在XML中指定该启动模式相同。

  • FLAG_ACTIVITY_SINGLE_TOP
    这个标记位的作用是为 Activity 指定 singleTop 启动模式,其效果和在XML中指定该启动模式相同。

  • FLAG_ACTIVITY_CLEAR_TOP
    具有此标记位的 Activity ,当它启动时,在同一个任务栈中所有位于它上面的 Activity 都要出栈,这个标记位一般和 singleTask 一起出现。在这种情况下,被启动的 Activity 实例如果已经存在,那么系统就会调用它的 onNewIntent 方法,如果被启动的 Activity 采用 standard 模式启动,那么它连同它之上的 Activity 都要出栈,系统会创建新的 Activity 实例并放入栈顶,由此可见,singleTask 启动模式默认具有此标记位的效果。

  • FLAG_ACTIVITY_EXCLUDE_FROM_RECENTS
    具有这个标记位的 Activity 不会出现在历史 Activity 的列表中,当某些情况下我们不希望用户通过历史列表回到我们的 Activity 的时候,这个标记比较有用,它等同于在XML中指定 android:exclueFromRecents=”true”

IntentFilter 的匹配规则

简介:
我们知道,启动 Activity 有两种方式,显示调用和隐式调用,显示调用需要明确指定被启动对象的组件信息,包括包名和类名,而隐式调用则需要明确指定组件信息,原则上一个 Intent 不该既是显示又是隐式,如果二者共存的话以显示调用为主。隐式调用需要 Intent 能够匹配目标组件的 IntentFilter 中的所设置的过滤信息,如果不匹配将无法启动目标 Activity ,IntentFilter 中包含 action、category、data。

<intent-filter>
<action android:name="android.intent.action.SEND" />
<action android:name="android.intent.action.SEND_MULTIPLE" />
<category android:name="android.intent.category.APP_BROWSER" />
<data android:mimeType="image/*" />
</intent-filter>


为了匹配过滤列表,需要同时匹配过滤列表中的 action、category、data信息,否则匹配失败,一个过滤列表中 action、category、data可以有多个,所有的 action、category、data构成不同的类别,同一类别的信息共同约束当前类别的匹配过程,当一个 Intent 同时匹配 action类别、category 类别、data 类别才算完全匹配,只有完全匹配才能成功启动。一个 Activity 可以有多个 intent-filter ,一个 Intent只要能匹配任何一组 intent-filter 即可成功启动。


  • action 匹配规则
    action 是一个字符串,系统预定义了一些 action ,同时我们也可以在应用中自己定义 action,action的匹配规则是:Intent 中的action必须和过滤规则中的 action 一致(字符串值完全相等),一个过滤规则中可以有多个 action ,那么只要 Intent 中的action 能够匹配任意一个 action即可匹配成功,如果 Intent中没有指定的 action ,那么匹配失败。

总结:action 的匹配要求 Intent 中必须存在 action 并且必须和 过滤规则中的一个匹配,区分大小写。


  • category 的匹配规则
    category 是一个字符串,系统已经预定义了一些 category,同时我们也可以在应用中自定义我们的 category ,它要求如果Intent 中含有category ,那么所有的 category都必须和匹配规则中的category 相同。Intent 中如果有 category ,不管有几个,对于每一个category都要和过滤规则中的 category相同。Intent 中可以没有 category,如果没有的话会默认加上
    <category android:name="android.intent.category.DEFAULT" />

总结:一个 Intent 可以没有category,但是如果有的话,不论有几个,每一个都必须和 过滤规则中的 category匹配。
一个 Intent 必须要有action,并且必须和 过滤规则中的匹配。


  • data的匹配规则
    data的匹配规则和 action 类似,如果过滤规则中定义了 data 那么 Intent中也必须要有 data属性。
    <data
    android:host="String"
    android:mimeType="String"
    android:path="String"
    android:pathPattern="String"
    android:pathPrefix="String"
    android:port="String"
    android:scheme="String" />

    data 由两部分组成,mimeType 和URI。mimeType是指媒体类型,可以表示 图片、文本、视频等不同的媒体格式。
    URI的结构:
    ://:/[||]
    如:
    content://com.example.project:200/folder/subfolder/etc
    http://www.baidu.com:80/search/info
  • Scheme:URI的模式,比如 :http,file,content,如果没有指定scheme,那么整个URI的其他参数无效,那么整个URI无效。
  • Host:URI的主机名,比如 www.baidu.com,如果host未指定,那么整个URI的其他参数无效,那么整个URI无效。
  • Port:URI中的端口号,比如80 ,仅当指定了 scheme和 host 参数时,port才有意义。
  • Path,pathPattern,pathPrefix:这三个参数表示路径信息,其中path表示完整的路径信息;pathPattern也表示完整的路径信息,但它里面可以包含通配符 “”,“”表示0个或多个任意字符;pathPrefix表示路径的前缀信息。

  • 如果没有URI,那么默认的是content和file。
  • 如果Intent要设置完整的data,必须调用 setDataAndType方法,不能分开调用 setData和setType,因为这两个方法会清除对方的值

判断是否匹配成功

当我们用一个隐式的方式启动一个 Activity的时候,可以做一下判断看是否有Activity 能够匹配Intent,否则如果没哟匹配成功的 Activity仍然启动就会报错。
判断方法有两种:采用 PackageManager 的 resolveActivity 方法或者Intent的 resolveActivity 方法,如果他们找不到匹配的 Activity 会返回null。
另外:PackageManager 还有 queryIntentActivities 方法,它不是返回最佳匹配的 Activity,而是返回所有成功匹配的 Activity的信息。
public abstract ResolveInfo resolveActivity(Intent intent, int flags);
public abstract List<ResolveInfo> queryIntentActivities(Intent intent, int flags);
第一个参数传递一个 Intent实例,第二个参数 需要使用 MATCH_DEFAULT_ONLY 这个标记位,这个标记位的含义是仅仅匹配那些在 intent-filter 中申明了 category 属性为 default 的Activity,如果设置了此标记,那么在结果中只要返回不为null,就一定不会报错,如果没有此标记,有可能匹配出 category 不含 default的activity,而这样的 Activity是无法接受隐式Intent的,会启动失败。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值