標籤:
Intent
Android中提供了Intent機制來協助應用間的互動與通訊,Intent負責對應用中一次操作的動作、動作涉及資料、附加資料進行描述,Android則根據此Intent的描述,負責找到對應的組件,將 Intent傳遞給調用的組件,並完成組件的調用。Intent不僅可用於應用程式之間,也可用於應用程式內部的Activity/Service之間的互動。因此,Intent在這裡起著一個媒體中介的作用,專門提供組件互相調用的相關資訊,實現調用者與被調用者之間的解耦。在SDK中給出了Intent作用的表現形式為:
****************************************************************************************************
· 通過Context.startActivity() orActivity.startActivityForResult() 啟動一個Activity;
· 通過 Context.startService() 啟動一個服務,或者通過Context.bindService() 和後台服務互動;
· 通過廣播方法(比如 Context.sendBroadcast(),Context.sendOrderedBroadcast(), Context.sendStickyBroadcast()) 發給broadcast receivers。
******************************************************************************************
Intent屬性的設定,包括以下幾點:(以下為XML中定義,當然也可以通過Intent類的方法來擷取和設定)
(1)Action,使用android:name特性來指定對響應的動作名。動作名必須是獨一無二的字串,所以,一個好的習慣是使用基於Java包的命名方式的命名系統。SDk中定義了一些標準的動作,包括
onstant |
Target component |
Action |
ACTION_CALL |
activity |
Initiate a phone call. |
ACTION_EDIT |
activity |
Display data for the user to edit. |
ACTION_MAIN |
activity |
Start up as the initial activity of a task, with no data input and no returned output. |
ACTION_SYNC |
activity |
Synchronize data on a server with data on the mobile device. |
ACTION_BATTERY_LOW |
broadcast receiver |
A warning that the battery is low. |
ACTION_HEADSET_PLUG |
broadcast receiver |
A headset has been plugged into the device, or unplugged from it. |
ACTION_SCREEN_ON |
broadcast receiver |
The screen has been turned on. |
ACTION_TIMEZONE_CHANGED |
broadcast receiver |
The setting for the time zone has changed. |
當然,也可以自訂動作(自訂的動作在使用時,需要加上包名作為首碼,如"com.example.project.SHOW_COLOR”),並可定義相應的Activity來處理我們的自訂動作。
(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是一系列的可選動作中的一個,這些動作可以在同一塊資料上執行。還有其他的為
Constant |
Meaning |
CATEGORY_BROWSABLE |
The target activity can be safely invoked by the browser to display data referenced by a link — for example, an image or an e-mail message. |
CATEGORY_GADGET |
The activity can be embedded inside of another activity that hosts gadgets. |
CATEGORY_HOME |
The activity displays the home screen, the first screen the user sees when the device is turned on or when the HOME key is pressed. |
CATEGORY_LAUNCHER |
The activity can be the initial activity of a task and is listed in the top-level application launcher. |
CATEGORY_PREFERENCE |
The target activity is a preference panel. |
(5)component(組件),指定Intent的的目標組件的類名稱。通常 Android會根據Intent 中包含的其它屬性的資訊,比如action、data/type、category進行尋找,最終找到一個與之匹配的目標組件。但是,如果 component這個屬性有指定的話,將直接使用它指定的組件,而不再執行上述尋找過程。指定了這個屬性以後,Intent的其它所有屬性都是可選的。
(6)extras(附加資訊),是其它所有附加資訊的集合。使用extras可以為組件提供擴充資訊,比如,如果要執行“寄送電子郵件”這個動作,可以將電子郵件的標題、本文等儲存在extras裡,傳給電子郵件發送組件。
Intent的兩種用法:
(1). 顯式的Intent,即在構造Intent對象時就指定接收者,它一般用在知道目標組件名稱的前提下,一般是在相同的應用程式內部實現的,如下:
Intent it = new Intent(Activity.Main.this, Activity2.class);
startActivity(it);
上面那個intent中, 直接指明了接收者:Activity2
(2).隱式的Intent,即Intent的寄件者在構造Intent對象時,並不知道也不關心接收者是誰,有利於降低寄件者和接收者之間的耦合,它一般用在沒有明確指出目標組件名稱的前提下,一般是用於在不同應用程式之間,如下:
Intent it = new Intent();
it.setAction("com.google.test");
startActivity(it);
上面那個intent, 沒有指明接收者, 只是給了一個action作為接收者的過濾條件。
對於顯式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用法執行個體
1.無參數Activity跳轉
Intent it = new Intent(Activity.Main.this, Activity2.class);
startActivity(it);
2.向下一個Activity傳遞資料(使用Bundle和Intent.putExtras)
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");
3.向上一個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);
4.回調上一個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);
}
}
}
Intent-Filter的定義
IntentFilter就是用於描述intent的各種屬性, 比如action, category等
一些屬性設定的例子:
<action android:name="com.example.project.SHOW_CURRENT" />
<category android:name="android.intent.category.DEFAULT" />
<data android:mimeType="video/mpeg" android:scheme="http" . . . />
<data android:mimeType="image/*" />
<data android:scheme="http" android:type="video/*" />
關於IntentFilter的幾點注意事項:
(1).android.intent.action.MAIN 與 android.intent.category.LAUNCHER
android.intent.action.MAIN決定一個應用程式最先啟動那個組件
android.intent.category.LAUNCHER決定應用程式是否顯示在程式列表裡(說白了就是是否在案頭上顯示一個表徵圖)
這兩個屬性群組合情況:
第一種情況:有MAIN,無LAUNCHER,程式列表中無表徵圖
原因:android.intent.category.LAUNCHER決定應用程式是否顯示在程式列表裡
第二種情況:無MAIN,有LAUNCHER,程式列表中無表徵圖
原因:android.intent.action.MAIN決定應用程式最先啟動的Activity,如果沒有Main,則不知啟動哪個Activity,故也不會有表徵圖出現
所以這兩個屬性一般成對出現。
如果一個應用中有兩個組件intent-filter都添加了android.intent.action.MAIN和
android.intent.category.LAUNCHER這兩個屬性, 則這個應用將會顯示兩個表徵圖, 寫在前面的組件先運行。
(2).關於隱式intent
每一個通過 startActivity() 方法發出的隱式 Intent 都至少有一個 category,就是 "android.intent.category.DEFAULT",所以只要是想接收一個隱式 Intent 的 Activity 都應該包括 "android.intent.category.DEFAULT" category,不然將導致 Intent 匹配失敗.
比如說一個activity組件要想被其他組件通過隱式intent調用, 則其在manifest.xml中的聲明如下:
<activity android:name="com.gesture.QGestureListActivity">
<intent-filter>
<action android:name="com.google.test" />
<category android:name="android.intent.category.DEFAULT" />
</intent-filter>
</activity>
(3).關於intent-filter匹配優先順序
首先查看Intent的過濾器(intent-filter),按照以下優先關係尋找:action->data->category
(4).android.intent.category.LAUNCHER與android.intent.category.HOME的區別
android.intent.category.LAUNCHER:android.intent.category.LAUNCHER決定應用程式是否顯示在程式列表裡,就是android開機後的主程式列表。
android.intent.category.HOME:按住“HOME”鍵,該程式顯示在HOME列表裡。
[轉]Intent和IntentFilter詳解