之前一直搞不很明白 AndroidManifest.xml 中 activity 標簽下的 intent-filter 中 data 標簽的屬性含義,今天認真看了 Dev Guide,又在網上查詢了大量相關資料,現把 data 標簽中的屬性含義做一個總結。
一、定義
scheme, host, port, path, pathPrefix, pathPattern 是用來匹配 Intent 中的 Data Uri 的。具體規則如下:
scheme://host:port/path or pathPrefix or pathPattern
這里需要注意的是這里的 scheme 不是 schema,也許你記得 xmlns:android="http://schemas.android.com/apk/res/android" 這段聲明,你就會想起其中的 schema (至少我是這樣想到了...- -!),但這里的 scheme 不是 schema。雖然在寫 AndroidManifest.xml 的時候,有智能提示,但是希望大家還是能注意到。
上面那句最后的 “path or pathPrefix or pathPattern” 是指后面的 path 驗證可以使用 data 屬性中的 android:path、android:pathPrefix 或 pathPattern,你可以添加任意個 data 標簽,由于是 “or” ,因此,只要其中任意一個 data 匹配,系統就會選擇你的 Activity 啟動,當然,如果別的 Activity 也有相同的 data 標簽,系統就會給用戶彈出一個 Chooser Dialog。
mimeType 也是是用來匹配 Intent 的。比如,當你使用 Intent.setType("text/plain") ,那么系統將會匹配到所有注冊 android:mimeType="text/plain" 的 Activity,想獲取更多有關 mimeType 的知識請參考:【轉】備份:Android 常用 mimeType 表。
這里需要十分注意的是 Intent.setType(), Intent.setData,Intent.setDataAndType() 這三個方法!
- setType 調用后設置 mimeType,然后將 data 置為 null;
- setData 調用后設置 data,然后將 mimeType 置為 null;
- setDataAndType 調用后才會同時設置 data 與 mimeType。
另外需要注意的是,如果你在 data 標簽,既設置了 mimeType 又設置了 scheme 之內的,那么你的 Intent 需要同時設置匹配的 data 與 mimeType 即調用 setDataAndType ,系統才能匹配到這個 Activity(即便你 mimeType 設置為 "*/*" 也是如此)。當然,如果你沒有設置 mimeType,那么調用 setData 進行匹配,如果你設置了任何的 mimeType 將不會匹配到該 Activity。
二、區別
這里主要說的區別是 path、pathPrefix、pathPattern 之間的區別
- path 用來匹配完整的路徑,如:http://example.com/blog/abc.html,這里將 path 設置為 /blog/abc.html 才能夠進行匹配;
- pathPrefix 用來匹配路徑的開頭部分,拿上面的 Uri 來說,這里將 pathPrefix 設置為 /blog 就能進行匹配了;
- pathPattern 用表達式來匹配整個路徑,這里需要說下匹配符號與轉義。
匹配符號:
- “*” 用來匹配0次或更多,如:“a*” 可以匹配“a”、“aa”、“aaa”...
- “.” 用來匹配任意字符,如:“.” 可以匹配“a”、“b”,“c”...
- 因此 “.*” 就是用來匹配任意字符0次或更多,如:“.*html” 可以匹配 “abchtml”、“chtml”,“html”,“sdf.html”...
轉義:
因為當讀取 Xml 的時候,“\” 是被當作轉義字符的(當它被用作 pathPattern 轉義之前),因此這里需要兩次轉義,讀取 Xml 是一次,在 pathPattern 中使用又是一次。如:“*” 這個字符就應該寫成 “\\*”,“\” 這個字符就應該寫成 “\\\\”。
三、一些例子
例子1:如果我們想要匹配 http 以 “.pdf” 結尾的路徑,使得別的程序想要打開網絡 pdf 時,用戶能夠可以選擇我們的程序進行下載查看。
我們可以將 scheme 設置為 “http”,pathPattern 設置為 “.*\\.pdf”,整個 intent-filter 設置為:

View Code
1 <intent-filter>
2 <action android:name="android.intent.action.VIEW"></action>
3 <category android:name="android.intent.category.DEFAULT"></category>
4 <data android:scheme="http" android:pathPattern=".*\\.pdf"></data>
5 </intent-filter>
如果你只想處理某個站點的 pdf,那么在 data 標簽里增加 android:host="yoursite.com" 則只會匹配 http://yoursite.com/xxx/xxx.pdf,但這不會匹配 www.yoursite.com,如果你也想匹配這個站點的話,你就需要再添加一個 data 標簽,除了 android:host 改為 “www.yoursite.com” 其他都一樣。
例子2:如果我們做的是一個IM應用,或是其他類似于微博之類的應用,如何讓別人通過 Intent 進行調用分享出現在選擇框里呢?
我們只用注冊 android.intent.action.SEND 與 mimeType 為 “text/plain” 或 “*/*” 就可以了,整個 intent-filter 設置為:

View Code
1 <intent-filter>
2 <action android:name="android.intent.action.SEND" />
3 <category android:name="android.intent.category.DEFAULT" />
4 <data mimeType="*/*" />
5 </intent-filter>
這里設置 category 的原因是,創建的 Intent 的實例默認 category 就包含了 Intent.CATEGORY_DEFAULT ,google 這樣做的原因是為了讓這個 Intent 始終有一個 category。
例子3:如果我們做的是一個音樂播放軟件,當文件瀏覽器打開某音樂文件的時候,使我們的應用能夠出現在選擇框里?
這類似于文件關聯了,其實做起來跟上面一樣,也很簡單,我們只用注冊 android.intent.action.VIEW 與 mimeType 為 “audio/*” 就可以了,整個 intent-filter 設置為:

View Code
1 <intent-filter>
2 <action android:name="android.intent.action.VIEW" />
3 <category android:name="android.intent.category.DEFAULT" />
4 <data android:mimeType="audio/*" />
5 </intent-filter>
參考:
posted on 2011-09-28 12:40
小果子 閱讀(1161)
評論(0) 編輯 收藏 引用