欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  移动技术

Android中的Intent Filter匹配规则简介

程序员文章站 2024-02-29 19:49:40
本文主要介绍了隐式intent匹配目标组件的规则,若有叙述不清晰或是不准确的地方希望大家指出,谢谢大家:  ) 1. intent简介 intent用于在一个...

本文主要介绍了隐式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。关于这几个方法的详细定义大家可以参考官方文档,这里不再赘述。

以上就是本文的全部内容,希望对大家的学习有所帮助。