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

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>
如果你只想處理某個(gè)站點(diǎn)的 pdf,那么在 data 標(biāo)簽里增加 android:host="yoursite.com" 則只會(huì)匹配 http://yoursite.com/xxx/xxx.pdf,但這不會(huì)匹配 www.yoursite.com,如果你也想匹配這個(gè)站點(diǎn)的話,你就需要再添加一個(gè) data 標(biāo)簽,除了 android:host 改為 “www.yoursite.com” 其他都一樣。
例子2:如果我們做的是一個(gè)IM應(yīng)用,或是其他類似于微博之類的應(yīng)用,如何讓別人通過 Intent 進(jìn)行調(diào)用分享出現(xiàn)在選擇框里呢?
我們只用注冊(cè) android.intent.action.SEND 與 mimeType 為 “text/plain” 或 “*/*” 就可以了,整個(gè) intent-filter 設(shè)置為:

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>
這里設(shè)置 category 的原因是,創(chuàng)建的 Intent 的實(shí)例默認(rèn) category 就包含了 Intent.CATEGORY_DEFAULT ,google 這樣做的原因是為了讓這個(gè) Intent 始終有一個(gè) category。
例子3:如果我們做的是一個(gè)音樂播放軟件,當(dāng)文件瀏覽器打開某音樂文件的時(shí)候,使我們的應(yīng)用能夠出現(xiàn)在選擇框里?
這類似于文件關(guān)聯(lián)了,其實(shí)做起來跟上面一樣,也很簡單,我們只用注冊(cè) android.intent.action.VIEW 與 mimeType 為 “audio/*” 就可以了,整個(gè) intent-filter 設(shè)置為:

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)
評(píng)論(0) 編輯 收藏 引用