Intent是一个用来从一个App组件启动其他组件的消息对象。这里有3种基本的用法:
- 启动activity: 传递Intent,使用startActivity()或者startActivityForResult(),后者可以传回结果。
- 启动service: 传递Intent,使用startService()启动一个没有用户界面的service,如果service被设计为C/S结构,可以使用bindService()绑定一个service。
- 传递broadcast: 可以向其他app传递信息通过Intent对象使用sendBroadcast()、 sendOrderedBroadcast()或sendStickyBroadcast()。
Intent类型
- 显式Intent 显式的Intent类型就是在启动的时候指定组件的类型,这个一般适用于自己的App。
- 隐式Intent 隐式的Intent类型一般是指指明action,然后传入Intent,适用于非己App。
当你创建一个隐式的Intent对象,安卓系统会通过Intent过滤器查找manifest文件寻找合适的App打开之。如果很多应用都有action,那么会出现一个对话框,让用户选择想要的程序。
下面这张图显示了一个Activity如何通过Intent来打开另外一个Activity。
注意:为了保证你的app的安全性,请一直使用显式的Intent来开启一个service,并且不要打开Intent过滤器,因为通过过滤器打开service不能保证启动者的身份。
建立一个Intent
Intent对象包含了安卓系统决定开启那些应用的信息并且包含了一些传递信息。Intent对象包含了以下内容:
Component name: 要开启的组件的名称。显式Intent所必须的,如果没有该项,则为隐式Intent。对于service必须指定该项。你可以通过 setComponent(), setClass(), setClassName()等函数设置该项的内容,比如com.example.ExampleActivity
Action 一个指定特定的行为的字符串,比如view、pick。
你可以使用setAction()函数来指定Intent的Action,也可以通过以方式自己定义action:
static final String ACTION_TIMETRAVEL = "com.example.action.TIMETRAVEL";
Data: URI对象可以作为MIME类型的数据。数据的类型由Intent的Action决定,例如,如果动作是ACTION_EDIT ,数据应包含编辑文档的URI。
设置Data URI需要调用setData()函数,如果仅仅设置MIME类型,则需要调用setType()函数,如果有必要,你可以同时设置两者通过调用setDataAndType()函数(这种情况下setData()、setType()函数将不起作用)。
Category:这个内容包含了额外的信息,这些信息需要下一个组件处理相关的信息,不是必要的。这儿有一些例子:
CATEGORY_BROWSABLE 目标Activity允许自身通过web浏览器来显示由链接诸如图像或电子邮件消息中引用的数据来启动。
CATEGORY_LAUNCHER 这个Activity是系统app启动器中的任务栏里启动的Activity。
你可以通过调用addCategory()函数来指定category。
Extras:键值对保存的信息。可以通过putExtra()存入,也可以建立一个Bundle对象保存所有的Extra数据,然后通过putExtras()函数插入到Intent中。
Flags : Intent类中定义了Flags函数作为元数据。这个参数可以禁止安卓系统如何运行activity并且接下来如何执行。相关函数为setFlags()
一个例子:
// Executed in an Activity, so 'this' is the Context
// The fileUrl is a string URL, such as "http://www.example.com/image.png"
Intent downloadIntent = new Intent(this, DownloadService.class);
downloadIntent.setData(Uri.parse(fileUrl));
startService(downloadIntent);
这个例子开启了一个下载服务,并将传入的fileUrl内容下载下来。
再来一个例子:
// Create the text message with a string
Intent sendIntent = new Intent();
sendIntent.setAction(Intent.ACTION_SEND);
sendIntent.putExtra(Intent.EXTRA_TEXT, textMessage);
sendIntent.setType(HTTP.PLAIN_TEXT_TYPE); // "text/plain" MIME type
// Verify that the intent will resolve to an activity
if (sendIntent.resolveActivity(getPackageManager()) != null) {
startActivity(sendIntent);
}
这个例子分享了一些文字信息。
强制使用App选择器
有时候需要不直接打开默认应用,而是每次都要求用户选择打开的app,这可以创建一个createChooser() Intent对象,然后将其传入startActivity(),如下:
Intent sendIntent = new Intent(Intent.ACTION_SEND);
...
// Always use string resources for UI text.
// This says something like "Share this photo with"
String title = getResources().getString(R.string.chooser_title);
// Create intent to show the chooser dialog
Intent chooser = Intent.createChooser(sendIntent, title);
// Verify the original intent will resolve to at least one activity
if (sendIntent.resolveActivity(getPackageManager()) != null) {
startActivity(chooser);
}
接收一个隐式的Intent
需要使用intent-filter,前面有说,就不废话了,直接上例子:
<activity android:name="MainActivity">
<!-- This activity is the main entry, should appear in app launcher -->
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
<activity android:name="ShareActivity">
<!-- This activity handles "SEND" actions with text data -->
<intent-filter>
<action android:name="android.intent.action.SEND"/>
<category android:name="android.intent.category.DEFAULT"/>
<data android:mimeType="text/plain"/>
</intent-filter>
<!-- This activity also handles "SEND" and "SEND_MULTIPLE" with media data -->
<intent-filter>
<action android:name="android.intent.action.SEND"/>
<action android:name="android.intent.action.SEND_MULTIPLE"/>
<category android:name="android.intent.category.DEFAULT"/>
<data android:mimeType="application/vnd.google.panorama360+jpg"/>
<data android:mimeType="image/*"/>
<data android:mimeType="video/*"/>
</intent-filter>
</activity>
这是一个多意图的过滤器。第一个Activity,MainActivity,是app的主要入口,通过点击图标进入,这个是由"android.intent.action.MAIN"决定的。 "android.intent.category.LAUNCHER"决定这个Activity是在系统app启动器中启动的。如果没有指定icon属性,则图标为默认。
第二个Activity,ShareActivity,为了便于共享文本和媒体内容,可以通过ShareActivity从其他app中进入,当然需要通过过滤器。
如果你想只有自己访问自己的过滤器而不是让其他app访问的话,可以设置exported属性为false。
使用Pending Intent
pending Intent是Intent的一种包装,主要的目的是将权限授予其他程序申请使用包含的Intent,就像执行自己的程序一样。主要的用途包括:
- 申明一个Intent,当用户平台接收到你的通知后执行。
- 申明一个Intent,当用户使用widget时执行。
- 申明一个Intent,在未来规定的时间内执行。
这里有三个方法:
- PendingIntent.getActivity() for an Intent that starts an Activity.
- PendingIntent.getService() for an Intent that starts a Service.
- PendingIntent.getBroadcast() for a Intent that starts an BroadcastReceiver.
具体的pending Intent官方解释在这儿。
Intent解决方案
当系统接收到一个隐式的Intent后,会通过action、data、category决定使用哪个app。
Action test
<intent-filter>
<action android:name="android.intent.action.EDIT" />
<action android:name="android.intent.action.VIEW" />
...
</intent-filter>
Category test
<intent-filter>
<category android:name="android.intent.category.DEFAULT" />
<category android:name="android.intent.category.BROWSABLE" />
...
</intent-filter>
Data test
<intent-filter>
<data android:mimeType="video/mpeg" android:scheme="http" ... />
<data android:mimeType="audio/mpeg" android:scheme="http" ... />
...
</intent-filter>
Intent匹配
queryIntentActivities()可以返回那些组件可以接受你的Intent,具体请查看这里
P.S:每天写一篇感觉好累啊……