Intent 和 Intent 过滤器

Intent是一个消息传递对象,我们一般用它来在组件间进行通信。

一、Intent类型

Intent分为两种类型:
(1) 显式Intent
知道要启动的组件确切名称,通过传入具体组件名称来启动组件。 通常启动自己应用中的组件都是通过显式Intent,因为要启动的 Activity 或服务的类名我们是知道的。
通常这样来使用:

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

(2)隐式Intent
在不确切的知道要打开哪个组件的情况下,通过指出action、data、category等信息,系统会寻找到匹配的组件。 例如,如需在地图上向用户显示位置,则可以使用隐式 Intent,请求另一具有此功能的应用在地图上显示指定的位置。

二、Intent对象

Intent对象携带了 Android 系统用来确定要启动哪个组件的信息(例如,准确的组件名称或应当接收该 Intent 的组件类别),以及收件人组件为了正确执行操作而使用的信息(例如,要采取的操作以及要处理的数据)。Intent 中包含的主要信息如下:

  • 组件名称
      要启动的组件名称。如需在应用中启动特定的组件,则应指定该组件的名称。如下所示,其中SecondActivity.class即为限定的组件名称,可用Intent构造函数或setComponent()、setClass()、setClassName()设置组件名称:
    方法1:在构造函数中传入Component
Intent intent = new Intent(this,SecondActivity.class);
startActivity(intent);

方法2:设置Component

Intent intent = new Intent();  
ComponentName componentName = new ComponentName(this, SecondActivity.class);  
// 或者ComponentName componentName = new ComponentName(this, "com.example.myapplication.SecondActivity");  
// 或者ComponentName componentName = new ComponentName(this.getPackageName(), "com.example.myapplication.SecondActivity");  
intent.setComponent(componentName);  
startActivity(intent); 

方法3:设置Class

Intent intent = new Intent();  
intent.setClass(this, SecondActivity.class);  
// 或者intent.setClassName(this, "com.example.myapplication.SecondActivity");  
// 或者intent.setClassName(this.getPackageName(), "com.example.myapplication.SecondActivity");  
startActivity(intent);
  • Action
      指定要执行的通用操作(例如,“查看”或“选取”)的字符串。可通过setAction方法为Intent设置action,也可在构造Intent时传入action。
      Android系统预定义了许多action,这些action代表了一些常见的操作。例如,我们可以使用ACTION_DIAL来调用拨打电话的页面,如下所示:
Intent intent = new Intent();
intent.setAction(Intent.ACTION_DIAL);
//intent.setAction("android.intent.action.DIAL");
startActivity(intent);

其中Intent.ACTION_DIAL是Android系统的内置常量,值为"android.intent.action.DIAL"。

  • Data
    data可分为两类:
    (1)uri
    uri由scheme、host、port、path | pathPattern | pathPrefix这4部分组成,Intent的uri可通过setData()方法设置。
    (2)mimetype
    指定数据的 MIME 类型有助于 Android 系统找到接收 Intent 的最佳组件,例如,能够显示图像的 Activity 可能无法播放音频文件,即便 URI 格式十分类似时也是如此。mimetype可通过setType()方法设置。
    注意:若要同时设置 URI 和 MIME 类型,需要使用setDataAndType(),而不应该调用 setData()和setType(),因为它们会互相抵消彼此的值
  • Category
      表示一个包含应处理 Intent 组件类型的附加信息的字符串。 一个 Intent 中可以有任意数量的Category,但大多数的 Intent 均不需要Category。可以使用addCategory()来指定类别。例如:
Intent intent = new Intent();
intent.setAction(Intent.ACTION_DIAL);
intent.addCategory(Intent.CATEGORY_DEFAULT);//默认Category,可省略
startActivity(intent);
  • Extra
    Intent可以携带的额外的key-value键值对,是其它所有附加信息的集合。使用extras可以为组件提供扩展信息,比如,如果要执行“发送电子邮件”这个动作,可以将电子邮件的标题、正文等保存在extras里,传给电子邮件发送组件。如下所示:
    方法1:通过调用putExtra()方法设置数据,每一个key对应一个value数据。
// 给someone@domain.com发邮件发送内容为“Hello”的邮件  
Intent intent =new Intent(Intent.ACTION_SEND);
intent.putExtra(Intent.EXTRA_EMAIL,"someone@domain.com");
intent.putExtra(Intent.EXTRA_SUBJECT,"Subject");
intent.putExtra(Intent.EXTRA_TEXT,"Hello");
intent.setType("text/plain");
startActivity(intent);

目标组件可以通过intent.getStringExtra()来读取邮件内容:

Intent intent = getIntent();
String text = intent.getStringExtra(Intent.EXTRA_TEXT);

方法2:通过创建Bundle对象来存储所有数据,然后通过调用putExtras()方法来设置数据。

Intent intent = new Intent(Intent.ACTION_SEND);
Bundle extra = new Bundle();
extra.putString(Intent.EXTRA_EMAIL,"someone@domain.com");
extra.putString(Intent.EXTRA_SUBJECT,"Subject");
extra.putString(Intent.EXTRA_TEXT,"Hello");
intent.putExtras(extra);
startActivity(intent);

目标组件可以通过bundle.getString()来获得邮件的标题和内容:

Intent intent = getIntent();
Bundle bundle = intent.getExtras();
String subject = bundle.getString(Intent.EXTRA_SUBJECT);
String text = bundle.getString(Intent.EXTRA_TEXT);
  • Flag
    标志可以指示 Android 系统如何启动 Activity(例如,Activity 应属于哪个任务),以及启动之后如何处理(例如,它是否属于最近的 Activity 列表),这里涉及到Activity启动的四种模式,例如:
Intent intent = new Intent(MainActivity.this,SecondActivity.class);
intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);//相当于singleTask
//intent.setFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);//相当于singleTop 
startActivity(intent);

三、Intent匹配

相比于显式Intent直接指明组件名称,隐式Intnet则含蓄了许多,它并不明确指出我们想要启动哪一个组件,而是指定一系列更为抽象的action和category等信息,然后交由系统去分析这个Intent,并帮我们找出合适的组件去启动。这时就需要将intent与清单文件中注册的组件的intent-filter进行匹配,来帮我们筛选出合适的组件来启动。
  下图是从网上找的一个intent-filter的匹配示意图,从中我们可以看出,只有action、data、category三方都匹配,Intent才算是匹配成功,进而才能打开相应的组件。

Intent Filter 的匹配过程

一个组件若在清单文件中声明了多个Intent Filter,则只需要匹配任意一个即可启动该组件。
通过比较解析隐式 Intent 时主要进行以下三层过滤:行为过滤 (action)、数据过滤 (data/type)、类别过滤 (category)。
(1)action匹配
   一个Intent Filter中可声明多个action,Intent中的action与其中的任一个action在字符串形式上完全相同(注意,区分大小写),action方面就匹配成功。
注意:隐式Intent必须指定action(如不指定action则必须指定data或mimetype。这种情况下,只要IntentFilter至少含有一个action就可以匹配)
例如,我们定义如下Intent:

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

则在清单文件中定义的组件的<intent-filter>中包含名为"android.intent.action.SEND"的action即为匹配成功,如下所示,该组件在action方面就与intent匹配了:

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

(2)data/type匹配
  同action类似,只要Intent的data只要与Intent Filter中的任一个data声明完全相同,data方面就匹配成功。若Intent Filter的data声明部分未指定uri,则缺省uri为content或file,Intent中的uri的scheme部分需为content或file才能匹配;
  type属性用于明确指定data属性的数据类型或MIME类型,但是通常来说,当Intent不指定data属性时,type属性才会起作用,否则Android系统将会根据data属性值来分析数据的类型,所以无需指定type属性。
  通过setData方法会把type属性设置为null,设置setType方法会把data设置为null,如果想要两个属性同时设置,要使用Intent.setDataAndType()方法。
  如果Intent对象中既包含Uri又包含Type,那么,在<intent-filter>中也必须二者都包含才能通过测试。

(3)category匹配
与action和data不同,Intent中的category必须都在Intent Filter中出现才算匹配成功。Intent可以不指定category,若Intent中未指定category,系统会自动为它带上“android.intent.category.DEFAULT”。所以,想要接收隐式Intent的组件都必须在manifest文件中的Intent Filter声明中带上“android.intent.category.DEFAULT”。

三、Intent匹配失败处理

在上面intent-filter的匹配过程示意图中我们可以看到,若一个组件都没有匹配成功的话,程序就会出错,抛出异常,一般有三种方式可以处理:
(1)使用try-catch捕获异常
例如我们随意启动一个并不存在的Activity--"abcdefg",程序抛出抛出ActivityNotFoundException的异常,可以使用如下try-catch语句来捕获:

Intent intent = new Intent("abcdefg");  
try  
{  
    startActivity(intent);  
}  
catch(ActivityNotFoundException e)  
{  
    Toast.makeText(this, "找不到对应的Activity", Toast.LENGTH_SHORT).show();  
}  

(2)预先使用resolveActivity方法判断
我们可以在startActivity()之前使用PackageManager的resolveActivity或者Intent的resolveActivity方法判断这个Intent是否能找到合适的Activity,如果没有,则不再startActivity,或者可以直接禁用用户操作的控件。

Intent intent = new Intent("abcdefg");  
if(intent.resolveActivity(getPackageManager()) == null)  
{  
    // 设置控件不可用  
}  

resolveActivity方法的返回值就是上面讲到的ComponentName对象,一般情况下也就是系统找到的那个Activity。但是如果有多个Activity可供选择的话,则返回的Component是com.android.internal.app.ResolverActivity,也就是用户选择Activity的那个界面对应的Activity:

Intent intent = new Intent(Intent.ACTION_DIAL);  
ComponentName componentName = intent.resolveActivity(getPackageManager());  
if(componentName != null)  
{  
    String className = componentName.getClassName();  
    Toast.makeText(this, className, Toast.LENGTH_SHORT).show();  
}  

(3)预先使用queryIntentActivities方法判断
若我们想查看所有匹配的Activity的信息,可以调用PackageManager的queryIntentActivities,该方法会返回所有成功匹配Intent的Activity的信息,我们可以根据该方法返回值判断是否有匹配的Activity。

public boolean isIntentAvailable(Context context, String action) 
{ 
    final PackageManager packageManager = context.getPackageManager();
    final Intent intent = new Intent(action);
    List<ResolveInfo> resolveInfo = packageManager.queryIntentActivities(
                     intent, PackageManager.MATCH_DEFAULT_ONLY);
     if (resolveInfo.size() > 0) 
      {
           return true; 
      }
           return false; 
}

一些常见的通用Intent的使用例子可以查看我的博客:通用Intent

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 202,905评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,140评论 2 379
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 149,791评论 0 335
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,483评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,476评论 5 364
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,516评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,905评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,560评论 0 256
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,778评论 1 296
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,557评论 2 319
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,635评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,338评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,925评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,898评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,142评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,818评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,347评论 2 342

推荐阅读更多精彩内容