android基础总结篇之九:Intent应用详解

时间:2021-08-13 15:35:03

今天我们来讲一下Android中Intent的原理和应用。

前面我们总结了几个Android中重要组件,相信大家对于这些组件已经有了清晰的认识,我们就来看一下几个常见的操作

启动一个Activity:Context.startActivity(Intent intent);

启动一个Service:Context.startService(Intent service);

绑定一个Service:Context.bindService(Intent service, ServiceConnection conn, int flags);

发送一个Broadcast:Context.sendBroadcast(Intent intent);

我们发现,在这些操作中,都有一个Intent参与其中,看起来像是一个非常重要的组件,那么Intent到底是什么呢?

简单来说,Intent是系统各组件之间进行数据传递的数据负载者。当我们需要做一个调用动作,我们就可以通过Intent告诉Android系统来完成这个过程,Intent就是调用通知的一种操作。Intent为组件的启动提供了一致的编程模型,无论想要启动什么组件,都可以使用intent封装启动的意图。并且,在某些时候,应用程序只是想启动具有某种也正的组件,并不像和某个特定的组件耦合,使用intent可以方便的达到这种高层次解耦的目的。

Intent有几个重要的属性,下面我们将会逐一介绍:

1.action,要执行的动作

action属性是由用户自定义的一个字符串,代表某一种特定的动作。系统预定义了一些action常量,开发者也可以自定义action.自定义的action应该以application的包名作为前缀,然后附加特定的大写字符串。例如:"com.scott.intent.action.TARGET"就是一个命名良好的action.

Intent类的setAction()方法用于设定action, getAction()方法可以获取Intent中封装的action.

对于有如下声明的Activity:

[html]  view plain  copy
  1. <activity android:name=".TargetActivity">  
  2.     <intent-filter>  
  3.         <action android:name="com.scott.intent.action.TARGET"/>  
  4.         <category android:name="android.intent.category.DEFAULT"/>  
  5.     </intent-filter>  
  6. </activity>  
TargetActivity在其<intent-filter>中声明了<action>,即目标action,如果我们需要做一个跳转的动作,就需要在Intent中指定目标的action,如下:

[java]  view plain  copy
  1. public void gotoTargetActivity(View view) {  
  2.     Intent intent = new Intent("com.scott.intent.action.TARGET");  
  3.     startActivity(intent);  
  4. }  

当我们为Intent指定相应的action,然后调用startActivity方法后,系统会根据action跳转到对应的Activity。

除了自定义的action之外,Intent也内含了很多默认的action,随便列举几个:

[java]  view plain  copy
  1. public static final String ACTION_MAIN = "android.intent.action.MAIN";  
  2. public static final String ACTION_VIEW = "android.intent.action.VIEW";  
  3. public static final String ACTION_WEB_SEARCH = "android.intent.action.WEB_SEARCH";  
  4. public static final String ACTION_CALL = "android.intent.action.CALL";  

每一个action都有其特定的用途,下文也会使用到它们。

2.data和extras,即执行动作要操作的数据和传递到目标的附加信息

下面就举一个与浏览器交互的例子:

[java]  view plain  copy
  1. /** 
  2.  * 打开指定网页 
  3.  * @param view 
  4.  */  
  5. public void invokeWebBrowser(View view) {  
  6.     Intent intent = new Intent(Intent.ACTION_VIEW);  
  7.     intent.setData(Uri.parse("http://www.google.com.hk"));  
  8.     startActivity(intent);  
  9. }  
  10.   
  11. /** 
  12.  * 进行关键字搜索 
  13.  * @param view 
  14.  */  
  15. public void invokeWebSearch(View view) {  
  16.     Intent intent = new Intent(Intent.ACTION_WEB_SEARCH);  
  17.     intent.putExtra(SearchManager.QUERY, "android");    //关键字  
  18.     startActivity(intent);  
  19. }  
上面两个方法分别是启动浏览器并打开指定网页、进行关键字搜索,分别对应的action是Intent.ACTION_VIEW和Intent.ACTION_WEB_SEARCH,前者需指定相应的网页地址,后者需指定关键字信息,对于关键字搜索来说,浏览器会按照自己设置的默认的搜索引擎进行搜索。

我们注意到,在打开网页时,为Intent指定一个data属性,这其实是指定要操作的数据,是一个URI的形式,我们可以将一个指定前缀的字符串转换成特定的URI类型,如:“http:”或“https:”表示网络地址类型,“tel:”表示电话号码类型,“mailto:”表示邮件地址类型,等等。例如,我们要呼叫给定的号码,可以这样做:

[java]  view plain  copy
  1. public void call(View view) {  
  2.     Intent intent = new Intent(Intent.ACTION_CALL);  
  3.     intent.setData(Uri.parse("tel:12345678"));  
  4.     startActivity(intent);  
  5. }  

那么我们如何知道目标是否接受这种前缀呢?这就需要看一下目标中<data/>元素的匹配规则了。

在目标<data/>标签中包含了以下几种子元素,他们定义了url的匹配规则:

android:scheme 匹配url中的前缀,除了“http”、“https”、“tel”...之外,我们可以定义自己的前缀

android:host 匹配url中的主机名部分,如“google.com”,如果定义为“*”则表示任意主机名

android:port 匹配url中的端口

android:path 匹配url中的路径

我们改动一下TargetActivity的声明信息:

[html]  view plain  copy
  1. <activity android:name=".TargetActivity">  
  2.     <intent-filter>  
  3.         <action android:name="com.scott.intent.action.TARGET"/>  
  4.         <category android:name="android.intent.category.DEFAULT"/>  
  5.         <data android:scheme="scott" android:host="com.scott.intent.data" android:port="7788" android:path="/target"/>  
  6.     </intent-filter>  
  7. </activity>  
这个时候如果只指定action就不够了,我们需要为其设置data值,如下:

[java]  view plain  copy
  1. public void gotoTargetActivity(View view) {  
  2.     Intent intent = new Intent("com.scott.intent.action.TARGET");  
  3.     intent.setData(Uri.parse("scott://com.scott.intent.data:7788/target"));  
  4.     startActivity(intent);  
  5. }  
此时,url中的每个部分和TargetActivity配置信息中全部一致才能跳转成功,否则就被系统拒绝。

不过有时候对path限定死了也不太好,比如我们有这样的url:(scott://com.scott.intent.data:7788/target/hello)(scott://com.scott.intent.data:7788/target/hi

这个时候该怎么办呢?我们需要使用另外一个元素:android:pathPrefix,表示路径前缀。

我们把android:path="/target"修改为android:pathPrefix="/target",然后就可以满足以上的要求了。

Intent的Extra属性
通过Intent启动一个component时, 经常需要携带一些额外的数据过去. 携带数据需要调用Intent的putExtra()方法, 该方法存在多个重载方法, 可用于携带基本数据类型及其数组, String类型及其数组, Serializable类型及其数组, Parcelable类型及其数组, Bundle类型等. Serializable和Parcelable类型代表一个可序列化的对象
, Bundle与Map类似,可用于存储键值对.

而在进行搜索时,我们使用了一个putExtra方法,将关键字做为参数放置在Intent中,我们成为extras(附加信息),这里面涉及到了一个Bundle对象。

Bundle和Intent有着密不可分的关系,主要负责为Intent保存附加参数信息,它实现了android.os.Paracelable接口,内部维护一个Map类型的属性,用于以键值对的形式存放附加参数信息。在我们使用Intent的putExtra方法放置附加信息时,该方法会检查默认的Bundle实例为不为空,如果为空,则新创建一个Bundle实例,然后将具体的参数信息放置到Bundle实例中。我们也可以自己创建Bundle对象,然后为Intent指定这个Bundle即可,如下:

[java]  view plain  copy
  1. public void gotoTargetActivity(View view) {  
  2.     Intent intent = new Intent("com.scott.intent.action.TARGET");  
  3.     Bundle bundle = new Bundle();  
  4.     bundle.putInt("id"0);  
  5.     bundle.putString("name""scott");  
  6.     intent.putExtras(bundle);  
  7.     startActivity(intent);  
  8. }  
需要注意的是,在使用putExtras方法设置Bundle对象之后,系统进行的不是引用操作,而是复制操作,所以如果设置完之后再更改bundle实例中的数据,将不会影响Intent内部的附加信息。那我们如何获取设置在Intent中的附加信息呢?与之对应的是,我们要从Intent中获取到Bundle实例,然后再从中取出对应的键值信息:

[java]  view plain  copy
  1. Bundle bundle = intent.getExtras();  
  2. int id = bundle.getInt("id");  
  3. String name = bundle.getString("name");  

当然我们也可以使用Intent的getIntExtra和getStringExtra方法获取,其数据源都是Intent中的Bundle类型的实例对象。

前面我们涉及到了Intent的三个属性:action、data和extras。除此之外,Intent还包括以下属性:

3.category,要执行动作的目标所具有的特质或行为归类

例如:在我们的应用主界面Activity通常有如下配置:

[html]  view plain  copy
  1. <category android:name="android.intent.category.LAUNCHER" />  
代表该目标Activity是该应用所在task中的初始Activity并且出现在系统launcher的应用列表中。

category属性也是一个字符串, 用于指定一些目标组件需要满足的额外条件. Intent对象中可以包含任意多个category属性. Intent类也预定义了一些category常量, 开发者也可以自定义category属性.
Intent类的addCategory()方法为Intent添加Category属性, getCategories()方法用于获取Intent中封装的所有category.
几个常见的category如下:
CATEGORY_HOME--表示目标activity必须是一个显示home screen的activity;
CATEGORY_LAUNCHER--表示目标activity可以作为task栈中的初始activity, 常与ACTION_MAIN配合使用;
CATEGORY_GADGET--表示目标activity可以被作为另一个activity的一部分嵌入.

Intent.CATEGORY_DEFAULT(android.intent.category.DEFAULT) 默认的category

Intent.CATEGORY_PREFERENCE(android.intent.category.PREFERENCE) 表示该目标Activity是一个首选项界面;

Intent.CATEGORY_BROWSABLE(android.intent.category.BROWSABLE)指定了此category后,在网页上点击图片或链接时,系统会考虑将此目标Activity列入可选列表,供用户选择以打开图片或链接。

在为Intent设置category时,应使用addCategory(String category)方法向Intent中添加指定的类别信息,来匹配声明了此类别的目标Activity。

4.type:要执行动作的目标Activity所能处理的MIME数据类型

例如:一个可以处理图片的目标Activity在其声明中包含这样的mimeType:

[html]  view plain  copy
  1. <data android:mimeType="image/*" />  
在使用Intent进行匹配时,我们可以使用setType(String type)或者setDataAndType(Uri data, String type)来设置mimeType。

5.component,目标组件的包或类名称

Intent对象的setComponent(ComponentName comp)方法用于设置Intent的Component属性. ComponentName包含如下几个构造器:
ComponentName(String pkg, String cls)
ComponentName(Context pkg, String cls)
ComponentName(Context pkg, Class<?> cls)
由以上的构造器可知, 创建一个ComponentName对象需要指定包名和类名--这就可以唯一确定一个组件类, 这样应用程序即可根据给定的组件类去启动特定的组件. 例如:

在使用component进行匹配时,一般采用以下几种形式:

[java]  view plain  copy
  1. intent.setComponent(new ComponentName(getApplicationContext(), TargetActivity.class));  
  2. intent.setComponent(new ComponentName(getApplicationContext(), "com.scott.intent.TargetActivity"));  
  3. intent.setComponent(new ComponentName("com.scott.other""com.scott.other.TargetActivity"));  

以上几句代码创建了一个制定了component属性的intent,完全等价于下面的代码:

Intent intent = new Intent(FirstActivity.this, com.scott.other.TargetActivity.class);

除了使用setComponent() 之外, 还可以使用setClass(), setClassName()来显式指定目标组件, 还可以调用getComponent()方法获得Intent中封装的ComponentName对象.

当程序采用这种形式启动组件时, 在Intent中明确的指定了待启动的组件类, 此时的Intent属于显式intent, 显式Intent应用场合比较狭窄, 多用于启动本应用中的component, 因为这种方式需要提前获知目标组件类的全限定名.而隐式Intent则通过Intent中的action, category, data属性指定目标组件需要满足的若干条件, 系统筛选出满足所有条件的component, 从中选择最合适的component或者由用户选择一个component作为目标组件启动.

其中,前两种是用于匹配同一包内的目标,第三种是用于匹配其他包内的目标。需要注意的是,如果我们在Intent中指定了component属性,系统将不会再对action、data/type、category进行匹配。如果Intent中指定了ComponentName属性, 则Intent的其他属性将被忽略.

6Intent的Flag属性
flag属性是一个int值, 用于通知android系统如何启动目标activity, 或者启动目标activity之后应该采取怎样的后续操作. 所有的flag都在Intent类中定义, 部分常用flag如下:
FLAG_ACTIVITY_NEW_TASK--通知系统将目标activity作为一个新task的初始activity;
FLAG_ACTIVITY_NO_HISTORY--通知系统不要将目标activity放入历史栈中;
FLAG_FROM_BACKGROUND--通知系统这个Intent来源于后台操作, 而非用户的直接选择...