首先廢話(huà)一下哼拔,如果要寫(xiě)東西棚饵,csdn和簡(jiǎn)書(shū)還是老老實(shí)實(shí)選簡(jiǎn)書(shū)送讲。不管什么辙芍,就沖csdn不會(huì)自動(dòng)保存踏志,簡(jiǎn)書(shū)會(huì)自動(dòng)保存就夠了撮珠。因?yàn)楸酒乙呀?jīng)寫(xiě)了兩遍(時(shí)間和精力都浪費(fèi)了琉兜。攀操。院仿。),一遍在csdn速和,一遍在這歹垫,只因在csdn寫(xiě)的時(shí)候,手賤颠放,沒(méi)有保存就不小心把頁(yè)面關(guān)閉了排惨,最蛋疼但是之前寫(xiě)好的一篇也不在了,改天得重寫(xiě)碰凶。
前言
Activity的啟動(dòng)方式分為兩種:顯式啟動(dòng)和隱式啟動(dòng)暮芭。
二者的區(qū)別:顯式啟動(dòng)要明確地指定被啟動(dòng)對(duì)象的組件信息,包括包名和類(lèi)名痒留,而隱式啟動(dòng)則不需要明確指定組件信息谴麦,但是需要Intent能夠匹配目標(biāo)組件的IntentFilter中所設(shè)置的過(guò)濾信息(IntentFilter中的過(guò)濾信息有action、category伸头、data)匾效,如果不匹配則無(wú)法啟動(dòng)目標(biāo)Activity。原則上一個(gè)Intent不應(yīng)該既是顯式啟動(dòng)又是隱式啟動(dòng)恤磷,如果共存的話(huà)面哼,則以顯式啟動(dòng)啟動(dòng)為主野宜。
顯式啟動(dòng)
一個(gè)簡(jiǎn)單的顯式啟動(dòng)啟動(dòng)如下:
// 此處A、B兩個(gè)Activity在同一個(gè)包下魔策,如果不在同一包下匈子,且沒(méi)有導(dǎo)入B相應(yīng)的包,則B需要帶上完整的包名
Intent intent = new Intent(A.this, B.class);
startActivity(intent);
需要注意的是:顯式啟動(dòng)只能啟動(dòng)本應(yīng)用內(nèi)的Activity闯袒,而不能跨應(yīng)用啟動(dòng)虎敦。
隱式啟動(dòng)
如果一個(gè)Activity要能夠被隱式啟動(dòng),則其intent-filter中必須要有一個(gè)action和一個(gè)android.intent.category.DEFAULT這樣的category政敢,否則無(wú)法被啟動(dòng)其徙。而如果要啟動(dòng)一個(gè)能夠被隱式啟動(dòng)的Activity,則Intent中必須要有一個(gè)目標(biāo)Activity中intent-filter中已經(jīng)定義的action喷户,Intent中可以不傳android.intent.category.DEFAULT的category唾那,系統(tǒng)會(huì)自動(dòng)添加android.intent.category.DEFAULT的category。
下面分析action褪尝、category闹获、data過(guò)濾信息的匹配規(guī)則。
1.action匹配規(guī)則
action是一個(gè)字符串河哑,系統(tǒng)預(yù)定義了一些action避诽,同時(shí)我們也可以自定義action(action區(qū)分大小寫(xiě),大小寫(xiě)不同灾馒,字符串相同的action會(huì)匹配失斁ビ谩)。
action的匹配規(guī)則:如果Intent只有一個(gè)action睬罗,intent-filter有多個(gè)action轨功,只要Intent中的action能夠和intent-filter中的任何一個(gè)action相同即可匹配成功(需要注意的是,隱式啟動(dòng)中容达,Intent中必須要有action古涧,如果沒(méi)有指定action,那么匹配失敾ㄑ巍)羡滑;如果Intent存在多個(gè)action,則這些action必須要在目標(biāo)Activity的intent-filter中已經(jīng)定義的算芯,否則也會(huì)匹配失敗柒昏。
例如,SecondActivity有如下過(guò)濾規(guī)則:
<activity android:name=".SecondActivity" >
<intent-filter>
<action android:name="com.example.test1"/>
<action android:name="com.example.test2"/>
<category android:name="android.intent.category.DEFAULT"/>
</intent-filter>
</activity>
以下3個(gè)方式都可以正常啟動(dòng)SecondActivity熙揍。
// 方式一 intent-filter中存在com.example.test1的action
Intent intent = new Intent();
intent.setAction("com.example.test1");
startActivity(intent);
// 方式二 intent-filter中存在com.example.test2的action
Intent intent = new Intent();
intent.setAction("com.example.test2");
startActivity(intent);
// 方式三 intent-filter中存在com.example.test1的action和com.example.test2的action
Intent intent = new Intent();
intent.setAction("com.example.test1");
intent.setAction("com.example.test2");
startActivity(intent);
以下是action無(wú)法匹配成功的導(dǎo)致無(wú)法正常啟動(dòng)SecondActivity职祷。
// 方式四 SecondActivity的intent-filter沒(méi)有配置com.example.test3的action
Intent intent = new Intent();
intent.setAction("com.example.test3");
startActivity(intent);
// 方式五 SecondActivity的intent-filter雖然配置了com.example.test1的action,但是沒(méi)有配置com.example.test3的action
Intent intent = new Intent();
intent.setAction("com.example.test1");
intent.setAction("com.example.test3");
startActivity(intent);
// 方式六 SecondActivity的intent-filter配置的com.example.test1的action為小寫(xiě)的test1,而Intent中的com.example.TEST1的action為大寫(xiě)的test1
Intent intent = new Intent();
intent.setAction("com.example.tesT1");
startActivity(intent);
2.category匹配規(guī)則
同action一樣有梆,category也是一個(gè)字符串是尖,系統(tǒng)也預(yù)定義了一些category,同時(shí)我們也可以自定義category(category也是區(qū)分大小寫(xiě)泥耀,大小寫(xiě)不同饺汹,字符串相同的category也會(huì)匹配失敗)痰催。
category的匹配規(guī)則和action基本一樣兜辞,區(qū)別在于:Intent中可以不傳category,系統(tǒng)會(huì)默認(rèn)添加上android.intent.category.DEFAULT這個(gè)category夸溶,但是一旦Intent中傳入了category弦疮,則傳入的所有category必須為目標(biāo)Activity中intent-filter已經(jīng)定義的category,否則匹配失敗蜘醋。
例如,SecondActivity有如下過(guò)濾規(guī)則:
<activity android:name=".SecondActivity" >
<intent-filter>
<action android:name="com.example.test1"/>
<category android:name="com.example.category1"/>
<category android:name="com.example.category2"/>
<category android:name="android.intent.category.DEFAULT"/>
</intent-filter>
</activity>
以下3個(gè)方式都可以正常啟動(dòng)SecondActivity咏尝。
// 方式一 Intent默認(rèn)添加android.intent.category.DEFAULT的category
Intent intent = new Intent();
intent.setAction("com.example.test1");
startActivity(intent);
// 方式二 intent-filter中存在com.example.category1的category
Intent intent = new Intent();
intent.setAction("com.example.test1");
intent.addCategory("com.example.category1");
startActivity(intent);
// 方式三 intent-filter中存在com.example.category2的category
Intent intent = new Intent();
intent.setAction("com.example.test1");
intent.addCategory("com.example.category2");
startActivity(intent);
以下由于category匹配不成功導(dǎo)致無(wú)法正常啟動(dòng)SecondActivity压语。
// 方式四 SecondActivity的intent-filter沒(méi)有配置com.example.category3的category
Intent intent = new Intent();
intent.setAction("com.example.test1");
intent.addCategory("com.example.category3");
startActivity(intent);
// 方式五 SecondActivity的intent-filter雖然配置了com.example.category1的category,但是沒(méi)有配置com.example.category3的category
Intent intent = new Intent();
intent.setAction("com.example.test1");
intent.addCategory("com.example.category1");
intent.addCategory("com.example.category3");
startActivity(intent);
// 方式六 SecondActivity的intent-filter配置的com.example.category1的category為小寫(xiě)的test1编检,而Intent中的com.example..CATEGORY1的category為大寫(xiě)的category1
Intent intent = new Intent();
intent.setAction("com.example.test1");
intent.addCategory("com.example.CATEGORY1");
startActivity(intent);
3.data匹配規(guī)則
首先胎食,認(rèn)識(shí)data匹配規(guī)則之前得先了解data的的結(jié)構(gòu)。
data語(yǔ)法如下:
<data android:scheme="string"
android:host="string"
android:port="string"
android:path="string"
android:pathPattern="string"
android:pathPrefix="string"
android:mimeType="string" />
data由兩部分組成允懂,mimeType和URI厕怜。mimeType指媒體類(lèi)型,比如image/jpeg蕾总、audio/mpeg4-generic和video/*等粥航,可以表示圖片、文本生百、視頻等不同的媒體格式递雀,而URI中包含的數(shù)據(jù)就比較多了,下面是URI的結(jié)構(gòu):
<scheme>://<host>:<port>/[<path>|<pathPrefix>|<pathPattern>]
如:
content://com.example.project:200/folder/subfolder/etc
http://www.baidu.com:80/search/info
每個(gè)數(shù)據(jù)的含義如下:
scheme:URI的模式蚀浆,比如http缀程、file、content等市俊,如果URI中沒(méi)指定scheme杨凑,那么整個(gè)URI的其他參數(shù)無(wú)效,也就意味著URI無(wú)效摆昧。
host:URI的主機(jī)名撩满,比如www.baidu.com,如果host未指定,那么整個(gè)URI的其他參數(shù)無(wú)效鹦牛,也就意味著URI無(wú)效搞糕。
port:URI中的端口號(hào),比如80曼追,僅當(dāng)URI中指定了scheme和host參數(shù)的時(shí)候窍仰,port參數(shù)才有意義。
path礼殊、pathPattern驹吮、pathPrefix:這三個(gè)參數(shù)表述的是路徑信息,其中path表示完整的路徑信息晶伦;pathPattern也表示完整的路徑信息碟狞,但是它里面可以包含通配符“*”,“*”表示0個(gè)或多個(gè)任意字符婚陪,需要注意的是族沃,由于正則表達(dá)式的規(guī)范,如果想表示真實(shí)的字符串泌参,那么“*”要寫(xiě)成“\\*”脆淹,“\”要寫(xiě)成“\\\\”,pathPrefix表示路徑的前綴信息沽一。
了解data數(shù)據(jù)格式后盖溺,現(xiàn)在可以說(shuō)下data的匹配規(guī)則。data匹配規(guī)則和action類(lèi)似铣缠,如果intent-filter中存在data信息烘嘱,則Intent中必須要有data數(shù)據(jù),并且data數(shù)據(jù)能夠完全匹配過(guò)濾規(guī)則中的某個(gè)data蝗蛙。這里分幾種情況說(shuō)明:
(1)有如下過(guò)濾規(guī)則:
<intent-filter>
...
<data android:mimeType="image/*" />
</intent-filter>
這種規(guī)則指定了媒體類(lèi)型為所有類(lèi)型的圖片蝇庭,那么Intent中mimeType屬性必須為“image/*”才能匹配,雖然沒(méi)有指定URI歼郭,但卻有默認(rèn)值遗契,URI的默認(rèn)值為content和file。因此病曾,Intent中URI部分的scheme必須為content或者file牍蜂。
為了匹配上面?zhèn)€的規(guī)則,可以寫(xiě)出如下示例:
intent.setDataAndType(Uri.parse("file://abc"), "image/png");
或
intent.setDataAndType(Uri.parse("content://abc"), "image/png");
如果要為Intent指定完整的data泰涂,必須要用setDataAndType方法鲫竞,不能先調(diào)用setData再調(diào)用setType,因?yàn)檫@兩個(gè)方法會(huì)互相清除對(duì)方逼蒙。
(2)有如下過(guò)濾規(guī)則:
<intent-filter>
...
<data android:mimeType="video/mpeg" android:scheme="http"/>
<data android:mimeType="audio/mpeg" android:scheme="http"/>
</intent-filter>
這種規(guī)則指定了兩組data規(guī)則从绘,且每個(gè)data都指定了完整的屬性值,既有URI又有mimeType。為了匹配如上規(guī)則僵井,可以寫(xiě)出如下示例:
intent.setDataAndType(Uri.parse("http://abc"), "video/mpeg");
或者
intent.setDataAndType(Uri.parse("http://abc"), "audio/mpeg");
data規(guī)則基本和action的類(lèi)似陕截,不同之處,在于data有如下兩種寫(xiě)法批什,但是它們作用是一樣的:
// 方式一
<intent-filter>
...
<data android:host="www.baidu.com" android:scheme="http"/>
</intent-filter>
// 方式二
<intent-filter>
...
<data android:scheme="http"/>
<data android:host="www.baidu.com"/>
</intent-filter>
最后
IntentFilte的匹配規(guī)則已經(jīng)基本完結(jié)农曲,另外,IntentFilter的匹配規(guī)則也適用于Service和BroadcastReceiver驻债,不過(guò)系統(tǒng)對(duì)于Service的建議是盡量用顯式啟動(dòng)方式來(lái)啟動(dòng)服務(wù)乳规。
在action和category中,如下兩個(gè)比較重要:
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER"/>
這二者共同表明這是一個(gè)入口Activity并且會(huì)出現(xiàn)在系統(tǒng)的應(yīng)用列表中合呐,少了任何一個(gè)都沒(méi)有實(shí)際意義暮的,也無(wú)法出現(xiàn)在系統(tǒng)的應(yīng)用列表中。
同時(shí)淌实,對(duì)于隱式啟動(dòng)Activity的時(shí)候冻辩,可以先做個(gè)判斷,看看是否有Activity能夠匹配Intent拆祈。PackageManager提供了兩種方法微猖,一個(gè)是resolveActivity方法,如果匹配不成功就返回null缘屹;另一個(gè)是queryIntentActivities,返回所有成功匹配的Activity信息侠仇。
兩個(gè)方法原型如下:
public abstract ResolveInfo resolveActivity(Intent intent, @ResolveInfoFlags int flags);
public abstract List<ResolveInfo> queryIntentActivities(Intent intent, @ResolveInfoFlags int flags);
上述兩個(gè)方法的第一個(gè)參數(shù)就是要就是要啟動(dòng)的intent對(duì)象轻姿,第二個(gè)參數(shù)則必須為PackageManager.MATCH_DEFAULT_ONLY這個(gè)標(biāo)記位,這個(gè)標(biāo)記位的含義是僅僅匹配那些在intent-filter中聲明了 <category android:name="android.intent.category.DEFAULT"/>這個(gè)category的Activity逻炊。使用這個(gè)標(biāo)記位的意義在于互亮,只要上述兩個(gè)方法不返回null,那么startActivity一定可以成功余素。如果不用這個(gè)標(biāo)記位豹休,那么就可以把intent-filter中不包含該DEFAULT的category的Activity匹配出來(lái),導(dǎo)致startActivity失敗桨吊。因?yàn)椴话珼EFAULT的category的Activity是無(wú)法接收隱式Intent的威根。針對(duì)Service和BroadcastReceiver,PackageManager也提供了類(lèi)似的方法用于判斷是否有匹配成功的組件信息视乐。