Android中的Intent Filter匹配规则介绍

http://www.cnblogs.com/absfree/p/5371097.html

本文主要介绍了隐式Intent匹配目标组件的规则,若有叙述不清晰或是不准确的地方希望大家指出,谢谢大家: )

1. Intent简介

    Intent用于在一个组件(Component,如Activity、Service、Broadcast Receiver)中打开另一个组件。

    Intent可分为隐式(implicitly)和显式(explicitly)两种:

  • Explicitly Intent:在知道要打开哪个具体的Component时使用,通过指定调用者和被调用者即可打开目标Component;
  • Implicitly Intent:在不确切的知道要打开哪个Component的情况下,通过指出action、data、category,系统会寻找到匹配的Component。

(1)Explicitly Intent

    当明确知道你想打开哪个Component时,它就是你的菜。通常这样使用:

Intent intent = new Intent(this, MainActivity.class);
intent.putExtra("key", "value");
startActivity(intent);

    执行以上代码会导致目标Component(这里是MainActivity)被创建(onCreate等一系列生命周期方法被调用)。在MainAcitivity中的相应生命周期方法中通过getIntent.getXxxExtra(“key”)即可得到随Intent一起传过来的数据。 

(2)Implicitly Intent

    Implicitly Intent很好的实现了调用者和被调用者之间的解耦:

    调用者通过action、data、category这三个方面描述他的Intent,被调用者通过在manifest文件中声明的一系列Intent Filter来描述自己能够响应哪些意图。如此一来,调用者和被调用者无需互相了解,通过Implicitly Intent这个联系他们的纽带就能很好的协同工作。

   

    关于Intent更加详细的介绍,大家可以参考官方文档或是相关博文,这里主要介绍下Implicitly Intent的匹配规则。

 

2.Intent Filter匹配规则

    只有action、data、category三方都匹配,Intent才算是匹配成功,进而才能打开相应的Component。一个Component若声明了多个Intent Filter,只需要匹配任意一个即可启动该组件。

(1)action的匹配规则

    一个Intent Filter中可声明多个action,Intent中的action与其中的任一个action在字符串形式上完全相同(注意,区分大小写),action方面就匹配成功。可通过setAction方法为Intent设置action,也可在构造Intent时传入action。需要注意的是,隐式Intent必须指定action。比如我们在Manifest文件中为MyActivity定义了如下Intent Filter:

<intent-filter>
    <action android:name="android.intent.action.SEND"/>
    <action android:name="android.intent.action.SEND_TO"/>
</intent-filter>

    那么只要Intent的action为“SEND”或“SEND_TO”,那么这个Intent在action方面就能和上面那个Activity匹配成功。比如我们的Intent定义如下:

Intent intent = new Intent("android.intent.action.SEND")
...

 

   那么我们的Intent在action方面就与MyActivity匹配了。

   Android系统预定义了许多action,这些action代表了一些常见的操作。常见action如下(Intent类中的常量):

  • Intent.ACTION_VIEW
  • Intent.ACTION_DIAL
  • Intent.ACTION_SENDTO
  • Intent.ACTION_SEND
  • Intent.ACTION_WEB_SEARCH

 

(2)data的匹配规则

    data可进一步分为uri(由scheme、host、port、path | pathPattern | pathPrefix这4部分组成)和mimetype。Intent的uri可通过setData方法设置,mimetype可通过setType方法设置。隐式Intent也必须指定data。同action类似,只要Intent的data只要与Intent Filter中的任一个data声明完全相同,data方面就匹配成功。需要注意的是:若Intent Filter的data声明部分未指定uri,则缺省uri为content或file,Intent中的uri的scheme部分需为content或file才能匹配;若要为Intent指定完整的data,必须用setDataAndType方法,原因请看setData和setType方法的源码:

复制代码
public Intent setData(Uri data) {
    mData = data;
    mType = null;
    return this;
}

public Intent setType(String type) {
    mData = null;
    mType = type;
    return this;
}
复制代码

 

    从以上代码可以看到,setData会把mimeType置为null,setType会把uri置为null。下面我们来举例说明一下data的匹配。首先我们先来看一下Intent Filter中指定data的语法:

复制代码
<data android:scheme="...“ 
          android:host="..."
          android:port="..."
          android:path="..."
          android:pathPattern="..."
          android:pathPrefix="..."
          android:mimeType="..." />
复制代码

 

    其中scheme、host等各个部分无需全部指定。假如我们为MyActivity的Intent Filter指定了如下data:

<intent-filter>
    <data android:mimeType="vidoe/mpeg" android:scheme="http" android:host="www.xxx.com" />
    <data android:mimeType="text/plain" android:scheme="http" />
</intent-filter>

 

 

    那么我们的Intent想要匹配,mimeType可以为”text/plain"或“video/mpeg",scheme必须为”http“,host则没有限制,因为第二个data没有指定host。

 

(3)category的匹配规则

    与action和data不同,Intent中的category必须都在Intent Filter中出现才算匹配成功。Intent可以不指定category,若Intent中未指定category,系统会自动为它带上“android.intent.category.DEFAULT”。所以,想要接收Implicitly Intent的Component都必须在manifest文件中的Intent Filter声明中带上“android.intent.category.DEFAULT”。我们可以通过addCategory方法为Intent添加category。

 

(4)查询是否有可接收指定Intent的Component

   采用PackageManager的resolveActivity或者Intent的resolveActivity方法会获得最适合Intent的一个Activity;调用PackageManager的queryIntentActivities会返回所有成功匹配Intent的Activity。关于这几个方法的详细定义大家可以参考官方文档,这里不再赘述。

 

3. 参考资料

(1)《Android开发艺术探索》

(2)Android docs

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值