天天看點

Android Intent 詳解

前言

         歡迎大家我分享和推薦好用的代碼段~~

聲明

         歡迎轉載,但請保留文章原始出處:

         CSDN:http://www.csdn.net

         雨季o莫憂離:http://blog.csdn.net/luckkof

正文

今天我們來講一下Android中Intent的原理和應用。

前面我們總結了幾個Android中重要元件,相信大家對于這些元件已經有了清晰的認識,我們就來看一下幾個常見的操作:

啟動一個Activity:Context.startActivity(Intent intent);

啟動一個Service:Context.startService(Intent service);

綁定一個Service:Context.bindService(Intent service, ServiceConnection conn, int flags);

發送一個Broadcast:Context.sendBroadcast(Intent intent);

我們發現,在這些操作中,都有一個Intent參與其中,看起來像是一個非常重要的元件,那麼Intent到底是什麼呢?

簡單來說,Intent是系統各元件之間進行資料傳遞的資料負載者。當我們需要做一個調用動作,我們就可以通過Intent告訴Android系統來完成這個過程,Intent就是調用通知的一種操作。

Intent有幾個重要的屬性,下面我們将會逐一介紹:

1.action,要執行的動作

對于有如下聲明的Activity:

<activity android:name=".TargetActivity">
			<intent-filter>
				<action android:name="com.scott.intent.action.TARGET"/>
				<category android:name="android.intent.category.DEFAULT"/>
			</intent-filter>
		</activity>
           

TargetActivity在其<intent-filter>中聲明了<action>,即目标action,如果我們需要做一個跳轉的動作,就需要在Intent中指定目标的action,如下:

public void gotoTargetActivity(View view) {
    	Intent intent = new Intent("com.scott.intent.action.TARGET");
    	startActivity(intent);
    }
           

當我們為Intent指定相應的action,然後調用startActivity方法後,系統會根據action跳轉到對應的Activity。

除了自定義的action之外,Intent也内含了很多預設的action,随便列舉幾個:\

public static final String ACTION_MAIN = "android.intent.action.MAIN";
    	public static final String ACTION_VIEW = "android.intent.action.VIEW";
    	public static final String ACTION_WEB_SEARCH = "android.intent.action.WEB_SEARCH";
    	public static final String ACTION_CALL = "android.intent.action.CALL";
           

每一個action都有其特定的用途,下文也會使用到它們。

2.data和extras,即執行動作要操作的資料和傳遞到目标的附加資訊

下面就舉一個與浏覽器互動的例子:

/**
     * 打開指定網頁
     * @param view
     */
    public void invokeWebBrowser(View view) {
    	Intent intent = new Intent(Intent.ACTION_VIEW);
    	intent.setData(Uri.parse("http://www.google.com.hk"));
    	startActivity(intent);
    }
    
    /**
     * 進行關鍵字搜尋
     * @param view
     */
    public void invokeWebSearch(View view) {
    	Intent intent = new Intent(Intent.ACTION_WEB_SEARCH);
    	intent.putExtra(SearchManager.QUERY, "android");	//關鍵字
    	startActivity(intent);
    }
           

上面兩個方法分别是啟動浏覽器并打開指定網頁、進行關鍵字搜尋,分别對應的action是Intent.ACTION_VIEW和Intent.ACTION_WEB_SEARCH,前者需指定相應的網頁位址,後者需指定關鍵字資訊,對于關鍵字搜尋來說,浏覽器會按照自己設定的預設的搜尋引擎進行搜尋。

我們注意到,在打開網頁時,為Intent指定一個data屬性,這其實是指定要操作的資料,是一個URI的形式,我們可以将一個指定字首的字元串轉換成特定的URI類型,如:“http:”或“https:”表示網絡位址類型,“tel:”表示電話号碼類型,“mailto:”表示郵件位址類型,等等。例如,我們要呼叫給定的号碼,可以這樣做:

public void call(View view) {
    	Intent intent = new Intent(Intent.ACTION_CALL);
    	intent.setData(Uri.parse("tel:12345678"));
    	startActivity(intent);
    }
           

那麼我們如何知道目标是否接受這種字首呢?這就需要看一下目标中<data/>元素的比對規則了。

在目标<data/>标簽中包含了以下幾種子元素,他們定義了url的比對規則:

android:scheme 比對url中的字首,除了“http”、“https”、“tel”...之外,我們可以定義自己的字首

android:host 比對url中的主機名部分,如“google.com”,如果定義為“*”則表示任意主機名

android:port 比對url中的端口

android:path 比對url中的路徑

我們改動一下TargetActivity的聲明資訊:

<activity android:name=".TargetActivity">
			<intent-filter>
				<action android:name="com.scott.intent.action.TARGET"/>
				<category android:name="android.intent.category.DEFAULT"/>
				<data android:scheme="scott" android:host="com.scott.intent.data" android:port="7788" android:path="/target"/>
			</intent-filter>
		</activity>
           

這個時候如果隻指定action就不夠了,我們需要為其設定data值,如下:

public void gotoTargetActivity(View view) {
    	Intent intent = new Intent("com.scott.intent.action.TARGET");
    	intent.setData(Uri.parse("scott://com.scott.intent.data:7788/target"));
    	startActivity(intent);
    }
           

此時,url中的每個部分和TargetActivity配置資訊中全部一緻才能跳轉成功,否則就被系統拒絕。

不過有時候對path限定死了也不太好,比如我們有這樣的url:(scott://com.scott.intent.data:7788/target/hello)(scott://com.scott.intent.data:7788/target/hi)

這個時候該怎麼辦呢?我們需要使用另外一個元素:android:pathPrefix,表示路徑字首。

我們把android:path="/target"修改為android:pathPrefix="/target",然後就可以滿足以上的要求了。

而在進行搜尋時,我們使用了一個putExtra方法,将關鍵字做為參數放置在Intent中,我們成為extras(附加資訊),這裡面涉及到了一個Bundle對象。

Bundle和Intent有着密不可分的關系,主要負責為Intent儲存附加參數資訊,它實作了android.os.Paracelable接口,内部維護一個Map類型的屬性,用于以鍵值對的形式存放附加參數資訊。在我們使用Intent的putExtra方法放置附加資訊時,該方法會檢查預設的Bundle執行個體為不為空,如果為空,則新建立一個Bundle執行個體,然後将具體的參數資訊放置到Bundle執行個體中。我們也可以自己建立Bundle對象,然後為Intent指定這個Bundle即可,如下:

public void gotoTargetActivity(View view) {
    	Intent intent = new Intent("com.scott.intent.action.TARGET");
    	Bundle bundle = new Bundle();
    	bundle.putInt("id", 0);
    	bundle.putString("name", "scott");
    	intent.putExtras(bundle);
    	startActivity(intent);
    }
           

需要注意的是,在使用putExtras方法設定Bundle對象之後,系統進行的不是引用操作,而是複制操作,是以如果設定完之後再更改bundle執行個體中的資料,将不會影響Intent内部的附加資訊。那我們如何擷取設定在Intent中的附加資訊呢?與之對應的是,我們要從Intent中擷取到Bundle執行個體,然後再從中取出對應的鍵值資訊:

Bundle bundle = intent.getExtras();
    	int id = bundle.getInt("id");
    	String name = bundle.getString("name");
           

當然我們也可以使用Intent的getIntExtra和getStringExtra方法擷取,其資料源都是Intent中的Bundle類型的執行個體對象。

前面我們涉及到了Intent的三個屬性:action、data和extras。除此之外,Intent還包括以下屬性:

3.category,要執行動作的目标所具有的特質或行為歸類

例如:在我們的應用主界面Activity通常有如下配置:

<category android:name="android.intent.category.LAUNCHER" />
           

代表該目标Activity是該應用所在task中的初始Activity并且出現在系統launcher的應用清單中。

幾個常見的category如下:

Intent.CATEGORY_DEFAULT(android.intent.category.DEFAULT) 預設的category

Intent.CATEGORY_PREFERENCE(android.intent.category.PREFERENCE) 表示該目标Activity是一個首選項界面;

Intent.CATEGORY_BROWSABLE(android.intent.category.BROWSABLE)指定了此category後,在網頁上點選圖檔或連結時,系統會考慮将此目标Activity列入可選清單,供使用者選擇以打開圖檔或連結。

在為Intent設定category時,應使用addCategory(String category)方法向Intent中添加指定的類别資訊,來比對聲明了此類别的目标Activity。

4.type:要執行動作的目标Activity所能處理的MIME資料類型

例如:一個可以處理圖檔的目标Activity在其聲明中包含這樣的mimeType:

<data android:mimeType="image/*" />
           

在使用Intent進行比對時,我們可以使用setType(String type)或者setDataAndType(Uri data, String type)來設定mimeType。

5.component,目标元件的包或類名稱

在使用component進行比對時,一般采用以下幾種形式:

intent.setComponent(new ComponentName(getApplicationContext(), TargetActivity.class));
    	intent.setComponent(new ComponentName(getApplicationContext(), "com.scott.intent.TargetActivity"));
    	intent.setComponent(new ComponentName("com.scott.other", "com.scott.other.TargetActivity"));
           

其中,前兩種是用于比對同一包内的目标,第三種是用于比對其他包内的目标。需要注意的是,如果我們在Intent中指定了component屬性,系統将不會再對action、data/type、category進行比對。