Android中提供了Intent機制來協助應用間的交互與通訊,Intent負責對應用中一次操作的動作、動作涉及數據、附加數據進行描述,Android則根據此Intent的描述,負責找到對應的組件,將 Intent傳遞給調用的組件,並完成組件的調用。Intent不僅可用於應用程序之間,也可用於應用程序內部的 Activity / Service之間的交互。因此,Intent在這里起着一個媒體中介的作用,專門提供組件互相調用的相關信息,實現調用者與被調用者之間的解耦。
1. Intent作用
Intent是一個將要執行的動作的抽象的描述,一般來說是作為參數來使用,由 Intent來協助完成 Android各個組件之間的通訊。比如說調用startActivity()來啟動一個Activity,或者由broadcaseIntent()來傳遞給所有感興趣的BroadcaseReceiver,再或者由startService() / bindservice()來啟動一個后台的 service。所以可以看出來,Intent 主要是用來啟動其他的 activity 或者 service,所以可以將 intent 理解成 activity 之間的粘合劑。
Intent作用的表現形式為:
- 啟動Activity
通過Context.startActvity() / Activity.startActivityForResult()啟動一個Activity;
- 啟動Service
通過Context.startService()啟動一個服務,或者通過Context.bindService()和后台服務交互;
- 發送Broadcast
通過廣播方法Context.sendBroadcasts() / Context.sendOrderedBroadcast() / Context.sendStickyBroadcast()發給Broadcast Receivers
2. Intent種類
- 顯式Intent
顯式,即直接指定需要打開的activity對應的類。
1)構造方法傳入Component,最常用的方式:
Intent intent = new Intent(this, SecondActivity.class);
startActivity(intent);
2)setComponent方法
Intent intent = new Intent();
intent.setClass(this, SecondActivity.class);
//或者intent.setClassName(this, "com.example.app.SecondActivity");
//或者intent.setClassName(this.getPackageName(),"com.example.app.SecondActivity");
startActivity(intent);
3)setClass / setClassName方法
Intent intent = new Intent();
intent.setClass(this, SecondActivity.class);
//或者intent.setClassName(this, "com.example.app.SecondActivity");
//或者intent.setClassName(this.getPackageName(),"com.example.app.SecondActivity");
startActivity(intent);
顯式Intent通過Component可以直接設置需要調用的Activity類,可以唯一確定一個Activity,意圖特別明確,所以是顯式的。設置這個類的方式可以是Class對象(如SecondActivity.class),也可以是包名加類名的字符串(如"com.example.app.SecondActivity")。
- 隱式Intent
隱式,不明確指定啟動哪個Activity,而是設置Action、Data、Category,讓系統來篩選出合適的Activity。篩選是根據所有的
下面以Action為例:
AndroidManifest.xml文件中,首先被調用的Activity要有一個帶有
<activity
android:name="com.example.app.SecondActivity">
<intent-filter>
<action android:name="mark"/>
<category android:name="android.intent.category.DEFAULT"/>
</intent-filter>
</activity>
然后,在MainActivity,才可以通過這個action name找到上面的Activity。下面兩種方式分別通過setAction和構造方法方法設置Action,兩種方式效果相同。
1)setAction方法
Intent intent = new Intent();
intent.setAction("mark");
startActivity(intent);
2)構造方法直接設置Action
Intent intent = new Intent("mark");
startActivity(intent);
為了防止應用程序之間互相影響,一般命名方式是包名+Action名,例如這里命名"mark"就很不合理了,就應該改成"com.example.app.Test"。
3. Intent屬性
Intent對象大致包括7大屬性:Action(動作)、Data(數據)、Category(類別)、Type(數據類型)、Component(組件)、Extra(擴展信息)、Flag(標志位)。其中最常用的是Action屬性和Data屬性。
- Action:用來表現意圖的行動
一個字符串變量,可以用來指定Intent要執行的動作類別。常見的action有:
Activity Actions:
類型 | 作用 |
---|---|
ACTION_MAIN | 表示程序入口 |
ACTION_VIEW | 自動以最合適的方式顯示Data |
ACTION_EDIT | 提供可以編輯的 |
ACTION_PICK | 選擇一個一條Data,並且返回它 |
ACTION_DAIL | 顯示Data指向的號碼在撥號界面Dailer上 |
ACTION_CALL | 撥打Data指向的號碼 |
ACTION_SEND | 發送Data到指定的地方 |
ACTION_SENDTO | 發送多組Data到指定的地方 |
ACTION_RUN | 運行Data,不管Data是什么 |
ACTION_SEARCH | 執行搜索 |
ACTION_WEB_SEARCH | 執行網上搜索 |
ACRION_SYNC | 執同步一個Data |
ACTION_INSERT | 添加一個空的項到容器中 |
Broadcast Actions:
類型 | 作用 |
---|---|
ACTION_TIME_TICK | 當前時間改變,並即時發送時間,只能通過系統發送。調用格式"android.intent.action.TIME_TICK" |
ACTION_TIME_CHENGED | 設置時間。調用格式"android.intent.action.TIME_SET" |
- Data:表示與動作要操縱的數據
一個URI對象是一個引用的data的表現形式,或是data的MIME類型;data的類型由Intent的action決定。
- Category:用來表現動作的類別
一個包含Intent額外信息的字符串,表示哪種類型的組件來處理這個Intent。任何數量的Category 描述都可以添加到Intent中,但是很多intent不需要category,下面列舉一些常用的category:
類型 | 作用 |
---|---|
CATEGORY_DEFAULT | 把一個組件Component設為可被implicit啟動的 |
CATEGORY_LAUNCHER | 把一個action設置為在頂級執行。並且包含這個屬性的Activity所定義的icon將取代application中定義的icon |
CATEGORY_BROWSABLE | 當Intent指向網絡相關時,必須要添加這個類別 |
CATEGORY_HOME | 使Intent指向Home界面 |
CATEGORY_PREFERENCE | 定義的Activity是一個偏好面板Preference Panel |
- Type:指定數據類型
一般Intent的數據類型能夠根據數據本身進行判定,但是通過設置這個屬性,可以強制采用顯式指定的類型而不再進行推導。
- Component:目的組件
指定Intent的目標組件名稱,當指定了這個屬性后,系統將跳過匹配其他屬性,而直接匹配這個屬性來啟動對應的組件。
- Extra:擴展信息
Intent可以攜帶的額外 key-value 數據,你可以通過調用putExtra()方法設置數據,每一個 key對應一個 value數據。你也可以通過創建 Bundle對象來存儲所有數據,然后通過調用putExtras()方法來設置數據。
類型 | 作用 |
---|---|
EXTRA_BCC | 存放郵件密送人地址的字符串數組 |
EXTRA_CC | 存放郵件抄送人地址的字符串數組 |
EXTRA_EMAIL | 存放郵件地址的字符串數組 |
EXTRA_SUBJECT | 存放郵件主題字符串 |
EXTRA_TEXT | 存放郵件內容 |
EXTRA_KEY_EVENT | 以KeyEvent對象方式存放觸發Intent 的按鍵 |
EXTRA_PHONE_ NUMBER | 存放調用ACTION_CALL 時的電話號碼 |
Flag:期望這個意圖的運行模式
用來指示系統如何啟動一個Activity,可以通過setFlags()或者addFlags()可以把標簽flag用在Intent中。
類型 | 作用 |
---|---|
FLAG_ACTIVITY_CLEAR_TOP | 相當於SingleTask |
FLAGE_ACTIVITY_SINGLE_TOP | 相當於SingleTop |
FLAG_ACTIVITY_NEW_TASK | 類似於SingleInstance |
FLAG_ACTIVITY_NO_HISTORY | 當離開該Activity后,該Activity將被從任務棧中移除 |
4. Intent用法
調用撥號程序
// 調用撥打電話,給10010撥打電話
Uri uri = Uri.parse("tel:10010");
Intent intent = new Intent(Intent.ACTION_DIAL, uri);
startActivity(intent);
// 直接撥打電話,需要加上權限 <uses-permission id="android.permission.CALL_PHONE" />
Uri callUri = Uri.parse("tel:10010");
Intent intent = new Intent(Intent.ACTION_CALL, callUri);
發送短信或彩信
// 給10010發送內容為“Hello”的短信
Uri uri = Uri.parse("smsto:10010");
Intent intent = new Intent(Intent.ACTION_SENDTO, uri);
intent.putExtra("sms_body", "Hello");
startActivity(intent);
// 發送彩信(相當於發送帶附件的短信)
Intent intent = new Intent(Intent.ACTION_SEND);
intent.putExtra("sms_body", "Hello");
Uri uri = Uri.parse("content://media/external/images/media/23");
intent.putExtra(Intent.EXTRA_STREAM, uri);
intent.setType("image/png");
startActivity(intent);
通過瀏覽器打開網頁
// 打開百度主頁
Uri uri = Uri.parse("https://www.baidu.com");
Intent intent = new Intent(Intent.ACTION_VIEW, uri);
startActivity(intent);
發送電子郵件
// 給someone@domain.com發郵件
Uri uri = Uri.parse("mailto:someone@domain.com");
Intent intent = new Intent(Intent.ACTION_SENDTO, uri);
startActivity(intent);
// 給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 intent=new Intent(Intent.ACTION_SEND);
String[] tos = {"1@abc.com", "2@abc.com"}; // 收件人
String[] ccs = {"3@abc.com", "4@abc.com"}; // 抄送
String[] bccs = {"5@abc.com", "6@abc.com"}; // 密送
intent.putExtra(Intent.EXTRA_EMAIL, tos);
intent.putExtra(Intent.EXTRA_CC, ccs);
intent.putExtra(Intent.EXTRA_BCC, bccs);
intent.putExtra(Intent.EXTRA_SUBJECT, "Subject");
intent.putExtra(Intent.EXTRA_TEXT, "Hello");
intent.setType("message/rfc822");
startActivity(intent);
顯示地圖與路徑規划
// 打開Google地圖中國北京位置(北緯39.9,東經116.3)
Uri uri = Uri.parse("geo:39.9,116.3");
Intent intent = new Intent(Intent.ACTION_VIEW, uri);
startActivity(intent);
// 路徑規划:從北京某地(北緯39.9,東經116.3)到上海某地(北緯31.2,東經121.4)
Uri uri = Uri.parse("http://maps.google.com/maps?f=d&saddr=39.9 116.3&daddr=31.2 121.4");
Intent intent = new Intent(Intent.ACTION_VIEW, uri);
startActivity(intent);
播放多媒體
Intent intent = new Intent(Intent.ACTION_VIEW);
Uri uri = Uri.parse("file:///sdcard/foo.mp3");
intent.setDataAndType(uri, "audio/mp3");
startActivity(intent);
Uri uri = Uri.withAppendedPath(MediaStore.Audio.Media.INTERNAL_CONTENT_URI, "1");
Intent intent = new Intent(Intent.ACTION_VIEW, uri);
startActivity(intent);
選擇圖片
Intent intent = new Intent(Intent.ACTION_GET_CONTENT);
intent.setType("image/*");
startActivityForResult(intent, 2);
拍照
// 打開拍照程序
Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
startActivityForResult(intent, 1);
// 取出照片數據
Bundle extras = intent.getExtras();
Bitmap bitmap = (Bitmap) extras.get("data");
獲取並剪切圖片
// 獲取並剪切圖片
Intent intent = new Intent(Intent.ACTION_GET_CONTENT);
intent.setType("image/*");
intent.putExtra("crop", "true"); // 開啟剪切
intent.putExtra("aspectX", 1); // 剪切的寬高比為1:2
intent.putExtra("aspectY", 2);
intent.putExtra("outputX", 20); // 保存圖片的寬和高
intent.putExtra("outputY", 40);
intent.putExtra("output", Uri.fromFile(new File("/mnt/sdcard/temp"))); // 保存路徑
intent.putExtra("outputFormat", "JPEG");// 返回格式
startActivityForResult(intent, 0);
// 剪切特定圖片
Intent intent = new Intent("com.android.camera.action.CROP");
intent.setClassName("com.android.camera", "com.android.camera.CropImage");
intent.setData(Uri.fromFile(new File("/mnt/sdcard/temp")));
intent.putExtra("outputX", 1); // 剪切的寬高比為1:2
intent.putExtra("outputY", 2);
intent.putExtra("aspectX", 20); // 保存圖片的寬和高
intent.putExtra("aspectY", 40);
intent.putExtra("scale", true);
intent.putExtra("noFaceDetection", true);
intent.putExtra("output", Uri.parse("file:///mnt/sdcard/temp"));
startActivityForResult(intent, 0);
打開手機應用市場
// 打開手機應用市場,直接進入該程序的詳細頁面
Uri uri = Uri.parse("market://details?id=" + packageName);
Intent intent = new Intent(Intent.ACTION_VIEW, uri);
startActivity(intent);
安裝程序
String fileName = Environment.getExternalStorageDirectory() + "/myApp.apk";
Intent intent = new Intent(Intent.ACTION_VIEW);
intent.setDataAndType(Uri.fromFile(new File(fileName)),
"application/vnd.android.package-archive");
startActivity(intent);
卸載程序
Uri uri = Uri.parse("package:" + packageName);
Intent intent = new Intent(Intent.ACTION_DELETE, uri);
startActivity(intent);
進入設置界面
// 進入系統設置界面
Intent intent = new Intent(android.provider.Settings.ACTION_SETTINGS);
startActivity(intent);