🔥码云GVP开源项目 12k star Uniapp+ElementUI 功能强大 支持多语言、二开方便! 广告
## 意图(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)