Activity中Intent的知識
Intent中的四個重要屬性——Action、Data、Category、Extras
Intent作為聯系各Activity之間的紐帶,其作用並不僅僅只限於簡單的數據傳遞。通過其自帶的屬性,其實可以方便的完成很多較為復雜的操作。例如直接調用撥號功能、直接自動調用合適的程序打開不同類型的文件等等。諸如此類,都可以通過設置Intent屬性來完成。
Intent主要有以下四個重要屬性,它們分別為:
Action:Action屬性的值為一個字符串,它代表了系統中已經定義了一系列常用的動作。通過setAction()方法或在清單文件AndroidManifest.xml中設置。默認為:DEFAULT。
Data:Data通常是URI格式定義的操作數據。例如:tel:// 。通過setData()方法設置。
Category:Category屬性用於指定當前動作(Action)被執行的環境。通過addCategory()方法或在清單文件AndroidManifest.xml中設置。默認為:CATEGORY_DEFAULT。
Extras:Extras屬性主要用於傳遞目標組件所需要的額外的數據。通過putExtras()方法設置。
四個屬性各自的常用值如下所示:
Action:
ACTION_MAIN:Android Application的入口,每個Android應用必須且只能包含一個此類型的Action聲明。
ACTION_VIEW:系統根據不同的Data類型,通過已註冊的對應Application顯示數據。
ACTION_EDIT:系統根據不同的Data類型,通過已註冊的對應Application編輯示數據。
ACTION_DIAL:打開系統默認的撥號程序,如果Data中設置了電話號碼,則自動在撥號程序中輸入此號碼。
ACTION_CALL:直接呼叫Data中所帶的號碼。
ACTION_ANSWER:接聽來電。
ACTION_SEND:由用戶指定發送方式進行數據發送操作。
ACTION_SENDTO:系統根據不同的Data類型,通過已註冊的對應Application進行數據發送操作。
ACTION_BOOT_COMPLETED:Android系統在啟動完畢後發出帶有此Action的廣播(Broadcast)。
ACTION_TIME_CHANGED:Android系統的時間發生改變後發出帶有此Action的廣播(Broadcast)。
ACTION_PACKAGE_ADDED:Android系統安裝了新的Application之後發出帶有此Action的廣播(Broadcast)。
ACTION_PACKAGE_CHANGED:Android系統中已存在的Application發生改變之後(如應用更新操作)發出帶有此Action的廣播(Broadcast)。
ACTION_PACKAGE_REMOVED:卸載了Android系統已存在的Application之後發出帶有此Action的廣播(Broadcast)。
Category:
CATEGORY_DEFAULT:Android系統中默認的執行方式,按照普通Activity的執行方式執行。
CATEGORY_HOME:設置該組件為Home Activity。
CATEGORY_PREFERENCE:設置該組件為Preference。
CATEGORY_LAUNCHER:設置該組件為在當前應用程序啟動器中優先級最高的Activity,通常為入口ACTION_MAIN配合使用。
CATEGORY_BROWSABLE:設置該組件可以使用瀏覽器啟動。
CATEGORY_GADGET:設置該組件可以內嵌到另外的Activity中。
Extras:
EXTRA_BCC:存放郵件密送人地址的字符串數組。
EXTRA_CC:存放郵件抄送人地址的字符串數組。
EXTRA_EMAIL:存放郵件地址的字符串數組。
EXTRA_SUBJECT:存放郵件主題字符串。
EXTRA_TEXT:存放郵件內容。
EXTRA_KEY_EVENT:以KeyEvent對象方式存放觸發Intent的按鍵。
EXTRA_PHONE_NUMBER:存放調用ACTION_CALL時的電話號碼。
Data:
tel://:號碼數據格式,後跟電話號碼。
mailto://:郵件數據格式,後跟郵件收件人地址。
smsto://:短息數據格式,後跟短信接收號碼。
content://:內容數據格式,後跟需要讀取的內容。
file://:文件數據格式,後跟文件路徑。
market://search?q=pname:pkgname:市場數據格式,在Google Market裏搜索包名為pkgname的應用。
geo://latitude,longitude:經緯數據格式,在地圖上顯示經緯度指定的位置。
在intent-filter中指定data屬性的實際目的是:要求接收的Intent中的data必須符合intent-filter中指定的data屬性,這樣達到反向限定Intent的作用。
例如:在AndroidManifest.xml 中進行如下設置:
- <activity android:name=".TestActivity">
- <intent-filter>
- <action android:name="com.jony.test"/>
- <data android:scheme="file"/>
- </intent-filter>
- </activity>
那麽啟動該Activity的Intent必須進行如下設置:
- Intent intent = new Intent();
- Uri uri = Uri.parse("file://com.android.test:520/mnt/sdcard");
- intent.setData(uri);
data屬性解析:android:scheme、android:host、android:port、android:path、android:mimeType
data的前四個屬性構成了URI的組成部分,mimeType設置了數據的類型
data元素組成的URI模型如下:
scheme://host:port/path
舉例說明:
URI file://com.android.jony.test:520/mnt/sdcard
scheme-->file:
host-->com.android.jony.test
port-->520
path-->mnt/sdcard
其中host和port為URI的authority,如果沒有指定host,port將被忽略
data的各屬性並不是獨立的,data的各屬性構成了URI的整個組成部分。要使authority(host和port)有意義,必須指定scheme;要使path有意義,必須使scheme和authority(host和port)有意義。
URI和intent-filter匹配:
Intent中URI和intent-filter進行比較的時候只會進行部分的比較:
(1)當intent-filter中只設置了scheme,只會比較URI的scheme部分;
(2)當intent-filter中只設置了scheme和authority,那麽只會匹配URI中的scheme和authority;
(3)當intent-filter中設置了scheme、authority和path,那麽只會匹配URI中的scheme、authority、path;(path可以使用通配符進行匹配)
(4)當intent-filter中設置了mimeType,還會進行數據類型的匹配。
總結:
(1)在AndroidMainfest.xml 中對每一個Activity都做了說明——intent-filter,intent-filter聲明了需要接收怎樣的Intent,當發送的Intent和intent-filter中定義的相符合,就會啟動相應的Activity;
(2)當有多個Activity符合發送的Intent時,Android系統會列出所有滿足Intent的Activity,用戶可以通過選擇進行相關的操作;
(3)在一個Activity的intent-filter中可以有多個action、多個category、多個data,這樣可以有多種組合與Intent進行匹配。註意:如果在一個Activity中有多個Intent進行匹配的時候,建議使用多個intent-filter與Intent進行匹配。猜測:這樣應該可以提高Intent的匹配速度。
(4)data屬性,這是一個進行反向限制Intent的操作,要求Intent的data必須是intent-filter中聲明的數據之一(應為在一個intent-filter中可以設置多個data)。註意:如果要啟動目標Activity,但是指定的data數據類型與Activity中data數據類型不匹配,將會造成ActivityNotFoundException異常。
————————————————————————————————————————————————————————————————————————
1、要弄清楚這個問題,首先需要弄明白什麽是implicit(隱藏) intent什麽是explicit(明確) intent。
Explicit Intent明確的指定了要啟動的Acitivity ,比如以下Java代碼:
- Intent intent= new Intent(this, B.class);
Implicit Intent沒有明確的指定要啟動哪個Activity ,而是通過設置一些Intent Filter來讓系統去篩選合適的Acitivity去啟動。
2、intent到底發給哪個activity,需要進行三個匹配,一個是action,一個是category,一個是data。
理論上來說,如果intent不指定category,那麽無論intent filter的內容是什麽都應該是匹配的。但是,如果是implicit intent,android默認給加上一個CATEGORY_DEFAULT,這樣的話如果intent filter中沒有android.intent.category.DEFAULT這個category的話,匹配測試就會失敗。所以,如果你的 activity支持接收implicit intent的話就一定要在intent filter中加入android.intent.category.DEFAULT。
例外情況是:
- <intent-filter>
- <action android:name="android.intent.action.MAIN" />
- <category android:name="android.intent.category.LAUNCHER" />
- </intent-filter>
中沒有必要加入android.intent.category.DEFAULT,當然加入也沒有問題。這個是應用啟動默認的第一個啟動的activity(每個應用有那麽多activity,總得有一個是第一個啟動的吧)
如果自己定義的某個Activity要通過隱式啟動,在AndroidManifast.xm那麽必須加上android.intent.category.DEFAULT,否則不起作用
除此之外,category的用途還有很多
比如做個桌面,按home鍵時啟動自己做的應用
- <activity
- android:name=".MainActivity"
- android:label="@string/app_name" >
- <intent-filter>
- <action android:name="android.intent.action.MAIN" />
- <category android:name="android.intent.category.LAUNCHER"/>
- <category android:name="android.intent.category.HOME" />
- </intent-filter>
- </activity>
在intent中是如何配置那三個匹配數據的呢,也簡單一說
也就是說,在不直接指定要跳轉的Activity的時候,為Intent提供一些相關的參數,讓其自動去和AndroidManifest.xml中已有的Activity去匹配
IntentFilter在xml中的三個主要的參數:action,categary,data。
我們通過Intent的構造函數或者Intent提供的方法可以指定這個三個參數:
- intent.setAction(action);
- intent.setData(data);
- intent.addCategory(category);
Activity中Intent的知識