## 意图(Intent)
>[info] 简述:
> Android中提供了Intent机制来协助应用间的交互与通讯,Intent负责对应用中一次操作的动作、动作涉及数据、附加数据进行描述,Android则根据此Intent的描述,负责找到对应的组件,将Intent传递给调用的组件,并完成组件的调用。Intent不仅可用于应用程序之间,也可用于应用程序内部的Activity/Service之间的交互。因此,Intent在这里起着一个媒体中介的作用,专门提供组件互相调用的相关信息,实现调用者与被调用者之间的解耦。在SDK中给出了 Intent作用的表现形式为:
* 通过**Context.startActivity()** or**Activity.startActivityForResult(**) 启动一个Activity;
* 通过**Context.startService()** 启动一个服务,或者通过**Context.bindService(**) 和后台服务交互;
* 通过广播方法(比如 **Context.sendBroadcast()**,**Context.sendOrderedBroadcast()**, **Context.sendStickyBroadcast**()) 发给broadcast receivers
### 1,Intent的属性
**(1)Action**,也就是**要执行的动作**
**(2)Data**,也就是执行动作要操作的数据
Android 中采用指向数据的一个URI来表示,如在联系人应用中,一个指向某联系人的URI可能为:`content://contacts/1 `。 对于不同的动作,其 URI数据的类型是不同的(可以设置type属性指定特定类型数据),如ACTION_EDIT指定Data为文件URI,打电话为tel:URI , 访问 网络为http:URI ,而由content provider提供的数据则为content: URIs。
**(3)type(数据类型)**,显式指定Intent的数据类型(MIME)。一般Intent的数据类型能够根据数据本身进行判定,但是通过设置这个属性,可以强制采用显式指定的类型而不再进行推导。
**(4)category(类 别)**,被执行动作的附加信息。例如 LAUNCHER_CATEGORY 表示Intent 的接受者应该在Launcher中作为顶级应用出现;而ALTERNATIVE_CATEGORY表示当前的Intent是一系列的可选动作中的一个,这 些动作可以在同一块数据上执行。
**(5)component(组 件)**,指定Intent的的目标组件的类名称。通常 Android会根据Intent 中包含的其它属性的信息,比如action、data/type、category进行查找,最终找到一个与之匹配的目标组件。但是,**如果 component这个属性有指定的话,将直接使用它指定的组件,而不再执行上述查找过程。指定了这个属性以后,Intent的其它所有属性都是可选的。**
**(6)extras(附加信息)**,是其它所有附加信息的集合。使用extras可以为组件提供扩展信息,比如,如果要执行“发送电子邮件”这个动作,可以将电子邮件的标题、正文等保存在extras里,传给电子邮件发送组件。
### 2,Intent 的基本用法
- **显式的Intent**
即在构造Intent对象时就指定接收者;
eg:
~~~
//显式意图 必须指定动作的执行者
Intent intent = new Intent();
//显式的指定了要激活的组件的包名和类名
intent.setClassName( "com.android.calculator2", "com.android.calculator2.Calculator" );
startActivity(intent);
~~~
- **隐式的Intent**,
即Intent的发送者在构造Intent对象时,并不知道也不关心接收者是谁,有利于降低发送者和接收者之间的耦合。
eg:
~~~
//隐式意图 只需要指定动作,解耦合
Intent intent = new Intent();
intent.setAction(Intent.ACTION_VIEW);
intent.setData(Uri.parse("http://www.baidu.com" ));
startActivity(intent);
~~~
### 3,使用自定义Intent(隐式意图)
自定义隐式意图
~~~
<activity
android:name="com.it360.mobilesafe.activity.LockOneKeyActivity"
android:label="@string/title_activity_lock_one_key" >
<intent-filter >
<action android:name="com.it360.mobilesafe.LOCKONEKEY" />
<category android:name="android.intent.category.DEFAULT" />
</intent-filter >
</activity >
~~~
使用自定义意图
~~~
Intent shortcutIntent = new Intent();
shortcutIntent.setAction("com.it360.mobilesafe.LOCKONEKEY");
shortcutIntent.addCategory(Intent.CATEGORY_DEFAULT);
~~~
### 4,Intent 的解析机制(隐式意图)
对于显式Intent,Android不需要去做解析,因为目标组件已经很明确,Android需要解析的是那些隐式Intent,通过解析,将 Intent映射给可以处理此Intent的Activity、IntentReceiver或Service。
**Intent 解析机制主要是通过查找已注册在AndroidManifest.xml中的所有IntentFilter及其中定义的Intent,最终找到匹配的 Intent**。在这个解析过程中,**Android是通过Intent的action、type、category这三个属性来进行判断的**,判断方法如下:
* 如果Intent指明定了action,则目标组件的IntentFilter的action列表中就必须包含有这个action,否则不能匹配;
* 如果Intent没有提供type,系统将从data中得到数据类型。和action一样,目标组件的数据类型列表中必须包含Intent的数据类型,否则不能匹配。
* 如果Intent中的数据不是content: 类型的URI,而且Intent也没有明确指定它的type,将根据Intent中数据的scheme (比如 http: 或者mailto:) 进行匹配。同上,Intent 的scheme必须出现在目标组件的scheme列表中。
* 如果Intent指定了一个或多个category,这些类别必须**全部**出现在组建的类别列表中。比如Intent中包含了两个类别:LAUNCHER_CATEGORY 和 ALTERNATIVE_CATEGORY,解析得到的目标组件必须至少包含这两个类别。
**Intent-Filter的定义**
完整的实例
~~~
<activity
android:name="NotesList"
android:label="@string/title_notes_list">
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
<intent-filter>
<action android:name="android.intent.action.VIEW" />
<action android:name="android.intent.action.EDIT" />
<action android:name="android.intent.action.PICK" />
<category android:name="android.intent.category.DEFAULT" />
<data android:mimeType="vnd.android.cursor.dir/vnd.google.note" />
</intent-filter>
<intent-filter>
<action android:name="android.intent.action.GET_CONTENT" />
<category android:name="android.intent.category.DEFAULT" />
<data android:mimeType="vnd.android.cursor.item/vnd.google.note" />
</intent-filter>
</activity>
~~~
### 5,Intent 用法实例
**①、无参数Activity跳转**
~~~
Intent it = new Intent(Activity.Main.this, Activity2.class);
startActivity(it);
~~~
**②、向下一个Activity传递数据(使用Bundle和Intent.putExtras)**
>[info] 注意:Activity间的数据通信,对于数据量比较大的,避免使用 Intent+Parcelable的方式,可以考虑EventBus等替代方案,以免造成TransactionTooLargeException。
~~~
Intent it = new Intent(Activity.Main.this, Activity2.class);
Bundle bundle=new Bundle();
bundle.putString("name", "This is from MainActivity!");
it.putExtras(bundle); // it.putExtra(“test”, "shuju”);
startActivity(it); // startActivityForResult(it,REQUEST_CODE);
~~~
对于数据的获取可以采用:
~~~
Bundle bundle=getIntent().getExtras();
String name=bundle.getString("name");
~~~
**③、向上一个Activity返回结果(使用setResult,针对startActivityForResult(it,REQUEST_CODE)启动的Activity)**
~~~
Intent intent=getIntent();
Bundle bundle2=new Bundle();
bundle2.putString("name", "This is from ShowMsg!");
intent.putExtras(bundle2);
setResult(RESULT_OK, intent);
~~~
**④、回调上一个Activity的结果处理函数(onActivityResult)**
~~~
@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
// TODO Auto-generated method stub
super.onActivityResult(requestCode, resultCode, data);
if (requestCode==REQUEST_CODE){
if(resultCode==RESULT_CANCELED)
setTitle("cancle");
else if (resultCode==RESULT_OK) {
String temp=null;
Bundle bundle=data.getExtras();
if(bundle!=null) temp=bundle.getString("name");
setTitle(temp);
}
}
}
~~~
### 6,Intent的Component属性(显示意图)
前面讲到,component(组件),指定Intent的的目标组件的类名称。**通常 Android会根据Intent 中包含的其它属性的信息,比如action、data/type、category进行查找,最终找到一个与之匹配的目标组件。但是,如果 component这个属性有指定的话,将直接使用它指定的组件,而不再执行其他查找过程。指定了这个属性以后,Intent的其它所有属性都是可选的。**
**Component 是用来查找组件的,我们可以根据setComponent()、setClass()、setClassName()来设置组件的名称。通过getComponet(ComponetName componetName)来获取组件。**
方法、一:
~~~
Intent intent = new Intent();
intent.setClassName(<package name>, <class name>); //简写
startActivity(intent);
~~~
方法、二:
~~~
Intent i=new Intent;
ComponentName com= new ComponentName(<Package Name> , <Calss Name>);
i.setComponent(com);
startActivity(i);
~~~
**CompentName对象**
这是一个可以注册系统组件的对象,,可以注册四大组件
eg:
**启动htmlviewer,并打开指定的一个文件**
~~~
Intent intent = new Intent();
ComponentName cn = new ComponentName("com.android.htmlviewer",
"com.android.htmlviewer.HTMLViewerActivity");
intent.setComponent(cn);
Uri uri = Uri.fromFile(new File("/sdcard/demo.txt"));
intent.setDataAndType(uri, "text/plain");
startActivity(intent);
~~~
### 7,[PendingIntent](https://developer.android.com/reference/android/app/PendingIntent.html) 延迟意象
**PendingIntent这个类用于处理即将发生的事情。比如在通知Notification中用于跳转页面,但不是马上跳转**。
**Intent 是及时启动,intent 随所在的activity 消失而消失。**
PendingIntent 可以看作是对intent的包装,**通常通过getActivity,getBroadcast ,getService来得到pendingIntent的实例**,当前activity并不能马上启动它所包含的intent,而是在外部执行 pendingIntent时,调用intent的。正由于pendingIntent中 保存有当前App的Context,使它赋予外部App一种能力,使得外部App可以如同当前App一样的执行pendingIntent里的Intent, 就算在执行时当前App已经不存在了,也能通过存在pendingIntent里的Context照样执行Intent。另外还可以处理intent执行后的操作。常和Alermanger 和NotificationManager一起使用。
Intent一般是用作Activity、Sercvice、BroadcastReceiver之间传递数据,而**Pendingintent,一般用在 Notification上,可以理解为延迟执行的intent,PendingIntent是对Intent一个包装。**
1. GSM网络中android发送短信示例
~~~
String msg ="你好,美女";
String number = "135****6784";
SmsManager sms = SmsManager.getDefault();
PendingIntent pi = PendingIntent.getBroadcast(SmsActivity.this,0,new Intent(...),0);
sms.sendTextMessage(number, null, msg, pi, null);
Toast.makeText(SmsActivity.this,"发送成功",Toast.LENGHT_LONG).show();
~~~
- 代码解释
- PendingIntent就是**一个Intent的描述**,我们可以把这个描述交给别的程序,别的程序根据这个描述在后面的别的时间做你安排做的事情 (By giving a PendingIntent to another application, you are granting it the right to perform the operation you have specified as if the other application was yourself,就相当于PendingIntent代表了Intent)。本例中别的程序就是发送短信的程序,短信发送成功后要把intent广播出去 。
~~~
SmsManager.sendTextMessage(String destinationAddress, String scAddress, String text,
PendingIntent sentIntent, PendingIntent deliveryIntent)
~~~
- 参数解释:
- 1)PendingIntent **sentIntent**:当短信发出时,成功的话sendIntent会把其内部的描述的intent广播出去,否则产生错误代码并通过android.app.PendingIntent.OnFinished进行回调,这个参数最好不为空,否则会存在资源浪费的潜在问题;
- 2)PendingIntent **deliveryIntent**:是当消息已经传递给收信人后所进行的PendingIntent广播。
查看PendingIntent 类可以看到许多的Send函数,就是PendingIntent在进行被赋予的相关的操作。
实例:
~~~
public class NotificationActivity extends Activity {
private static final int NOTIFICATION_ID = 1;
private NotificationManager myNotificationManager;
public void onCreate(Bundle saveIntenceState) {
……
myNotificationManager =
(NotificationManager) getSystemService(NOTIFICATION_SERVICE);
int icon = R.drawable.smallIcon;
String tickerText = “Get New Message”;
String expandTitle = “NewMessage”;
String expandContent = “The notification content abstract when it expanded. ”;
Context context = getApplicationContext();
Notification notifiMessage =
new Notification(icon,tickerText,System.currentTimeMillis());
Intent requestIntent = new Intent(OrginalActivity.class, DestinationActivity.class);
PendingIntent pendIntent = PendingIntent.getActivity(context, 0, requestIntent, 0);
notifiMessage.setLatestEventInfo(context, expandTitle, expandContent, pendIntent);
myNotificationManager.notify(NOTIFICATION_ID, notifiMessage);
}
}
~~~
这里实例化一个 Notification对象,之后通过NotificationManager将通知显示出来。需要注意通知管理器需要将每一个管理的通知都设置一个唯一的ID,这个ID由开发者自行指定。对于显示的通知调用setLatestEventInfo()方法设置通知展开后将要显示的标题和内容,将显式 Intent 传入 getActivity()方法获得 PendingIntent对象实例作为参数传入设置单击事件处理消息。当用户点击该通知时,系统会使用相应的 Intent 消息加载并启动目标组件。
### 8,Intent 常见应用
**显示网页**
~~~
Uri uri = Uri.parse("http://google.com");
Intent it = new Intent(Intent.ACTION_VIEW, uri);
startActivity(it);
~~~
**显示地图**
~~~
Uri uri = Uri.parse("geo:38.899533,-77.036476");
Intent it = new Intent(Intent.ACTION_VIEW, uri);
startActivity(it);
~~~
- 其他 geo URI 範例
1. //geo:latitude,longitude
2. //geo:latitude,longitude?z=zoom
3. //geo:0,0?q=my+street+address
4. //geo:0,0?q=business+near+city
5. //google.streetview:cbll=lat,lng&cbp=1,yaw,,pitch,zoom&mz=mapZoom
**路径规划**
~~~
Uri uri = Uri.parse("http://maps.google.com/maps?
f=d&saddr=startLat%20startLng&daddr=endLat%20endLng&hl=en");
Intent it = new Intent(Intent.ACTION_VIEW, uri);
startActivity(it);
~~~
//where startLat, startLng, endLat, endLng are a long with 6 decimals like: 50.123456
**打电话**
- **①、叫出拨号程序**
~~~
Uri uri = Uri.parse("tel:0800000123");
Intent it = new Intent(Intent.ACTION_DIAL, uri);
startActivity(it);
~~~
- **②、直接打电话出去**
~~~
Uri uri = Uri.parse("tel:0800000123");
Intent it = new Intent(Intent.ACTION_CALL, uri);
startActivity(it);
~~~
> 用这个,要在 AndroidManifest.xml 中,加上 `<uses-permission id="android.permission.CALL_PHONE" />`
**传送SMS/MMS**
- ①、调用短信程序
~~~
Intent it = new Intent(Intent.ACTION_VIEW, uri);
it.putExtra("sms_body", "The SMS text");
it.setType("vnd.android-dir/mms-sms");
startActivity(it);
~~~
- ②、传送消息
~~~
Uri uri = Uri.parse("smsto://0800000123");
Intent it = new Intent(Intent.ACTION_SENDTO, uri);
it.putExtra("sms_body", "The SMS text");
startActivity(it);
~~~
- ③、传送 MMS
~~~
Uri uri = Uri.parse("content://media/external/images/media/23");
Intent it = new Intent(Intent.ACTION_SEND);
it.putExtra("sms_body", "some text");
it.putExtra(Intent.EXTRA_STREAM, uri);
it.setType("image/png");
startActivity(it);
~~~
**传送 Email**
①、
~~~
Uri uri = Uri.parse("mailto:xxx@abc.com");
Intent it = new Intent(Intent.ACTION_SENDTO, uri);
startActivity(it);
~~~
②、
~~~
Intent it = new Intent(Intent.ACTION_SEND);
it.putExtra(Intent.EXTRA_EMAIL, "me@abc.com");
it.putExtra(Intent.EXTRA_TEXT, "The email body text");
it.setType("text/plain");
startActivity(Intent.createChooser(it, "Choose Email Client"));
~~~
③、
~~~
Intent it=new Intent(Intent.ACTION_SEND);
String[] tos={"me@abc.com"};
String[] ccs={"you@abc.com"};
it.putExtra(Intent.EXTRA_EMAIL, tos);
it.putExtra(Intent.EXTRA_CC, ccs);
it.putExtra(Intent.EXTRA_TEXT, "The email body text");
it.putExtra(Intent.EXTRA_SUBJECT, "The email subject text");
it.setType("message/rfc822");
startActivity(Intent.createChooser(it, "Choose Email Client"));
~~~
④、传送附件
~~~
Intent it = new Intent(Intent.ACTION_SEND);
it.putExtra(Intent.EXTRA_SUBJECT, "The email subject text");
it.putExtra(Intent.EXTRA_STREAM, "file:///sdcard/mysong.mp3");
sendIntent.setType("audio/mp3");
startActivity(Intent.createChooser(it, "Choose Email Client"));
~~~
**播放多媒体**
~~~
Uri uri = Uri.parse("file:///sdcard/song.mp3");
Intent it = new Intent(Intent.ACTION_VIEW, uri);
it.setType("audio/mp3");
startActivity(it);
Uri uri = Uri.withAppendedPath(MediaStore.Audio.Media.INTERNAL_CONTENT_URI, "1");
Intent it = new Intent(Intent.ACTION_VIEW, uri);
startActivity(it);
~~~
**Market 相关**
1. 寻找某个应用
~~~
Uri uri = Uri.parse("market://search?q=pname:pkg_name");
Intent it = new Intent(Intent.ACTION_VIEW, uri);
startActivity(it);
~~~
> where pkg_name is the full package path for an application
2. 显示某个应用的相关信息
~~~
Uri uri = Uri.parse("market://details?id=app_id");
Intent it = new Intent(Intent.ACTION_VIEW, uri);
startActivity(it);
~~~
> where app_id is the application ID, find the ID by clicking on your application on Market home page, and notice the ID from the address bar
- Uninstall 应用程序
~~~
Uri uri = Uri.fromParts("package", strPackageName, null);
Intent it = new Intent(Intent.ACTION_DELETE, uri);
startActivity(it);
~~~
### 9,Android Intent Action 一览表
![intent action1](https://box.kancloud.cn/55504127a9d135b9c8ab4c449d0185d9_1106x749.jpg)
![intent action2](https://box.kancloud.cn/2616e10c7a2b80b2decd8ebc1c86660a_1105x744.jpg)
![intent action3](https://box.kancloud.cn/4b56432b8a99eb91508a9c02bbd27573_1104x817.jpg)
![intent action4](https://box.kancloud.cn/5614e46a5bc0b92bdf9044a79032d61b_1107x818.jpg)
![intent action5](https://box.kancloud.cn/2af279ba8d6ef0cad92fa189db1f0fb6_1107x746.jpg)
![intent action6](https://box.kancloud.cn/46abbf4bb67ae4519a43623569b6169f_1105x850.jpg)
![intent action7](https://box.kancloud.cn/6cea482166b86aa61f28098f2f1087f4_1101x727.jpg)
![intent action8](https://box.kancloud.cn/71cd827fec5ec7c9f76801a658abb8e6_1101x475.jpg)
- 前言
- Android系统的体系结构
- Dalvik VM 和 JVM 的比较
- Android 打包应用程序并安装的过程
- Android ADB工具
- Android应用开发
- Android UI相关知识总结
- Android 中window 、view、 Activity的关系
- Android应用界面
- Android中的drawable和bitmap
- AndroidUI组件adapterView及其子类和Adapter的关系
- Android四大组件
- Android 数据存储
- SharedPreference
- Android应用的资源
- 数组资源
- 使用Drawable资源
- Material Design
- Android 进程和线程
- 进程
- 线程
- Android Application类的介绍
- 意图(Intent)
- Intent 和 Intent 过滤器(Google官网介绍)
- Android中关于任务栈的总结
- 任务和返回栈(官网译文)
- 总结
- Android应用安全现状与解决方案
- Android 安全开发
- HTTPS
- 安卓 代码混淆与打包
- 动态注入技术(hook技术)
- 一、什么是hook技术
- 二、常用的Hook 工具
- Xposed源码剖析——概述
- Xposed源码剖析——app_process作用详解
- Xposed源码剖析——Xposed初始化
- Xposed源码剖析——hook具体实现
- 无需Root也能Hook?——Depoxsed框架演示
- 三、HookAndroid应用
- 四、Hook原生应用程序
- 五、Hook 检测/修复
- Android 应用的逆向与加固保护技术
- OpenCV在Android中的开发
- Android高级开发进阶
- 高级UI
- UI绘制流程及原理
- Android新布局ConstraintLayout约束布局
- 关键帧动画
- 帧动画共享元素变换
- Android异步消息处理机制完全解析,带你从源码的角度彻底理解
- Android中为什么主线程不会因为Looper.loop()里的死循环卡死?
- 为什么 Android 要采用 Binder 作为 IPC 机制?
- JVM 中一个线程的 Java 栈和寄存器中分别放的是什么?
- Android源码的Binder权限是如何控制?
- 如何详解 Activity 的生命周期?
- 为什么Android的Handler采用管道而不使用Binder?
- ThreadLocal,你真的懂了吗?
- Android屏幕刷新机制