1. 程式人生 > >Android隱式啟動Activity匹配詳解:Action,category,data

Android隱式啟動Activity匹配詳解:Action,category,data

關於Android隱式啟動Activity

隱式啟動Activity的intent到底發給哪個activity,需要進行三個匹配,一個是action,一個是category,一個是data,可以是全部或部分匹配

同樣適用於Service和BroadcastReceiver,下面是以Activity為例

MainActivity.java --主Activity

TestActivity.java --需要隱式啟動的Activity

(1) 根據Action和Category來進行匹配

         <activity android:name=".TestActivity"  android:label="TestActivity">

        <intent-filter >

         <action android:name="cc.android/myaction.leo"/>

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

        </intent-filter>

        </activity>

在MainActivity.java裡啟動它:

intent.setAction( "cc.android/myaction.leo");

//不加下面這行也行,因為intent的這個屬性預設值即系Intent.CATEGORY_DEFAULT

intent.addCategory(Intent.CATEGORY_DEFAULT);

startActivity( intent );

總結:

a.在某個Activity裡用startActivity()方法傳送一個intent,這個intent設定了一些條件,比如用方法setAction(),addCategory()設定了兩個屬性,

  傳送了這個intent之後,android會去系統裡儲存的MainManifest.xml清單(假設這個系統存放全部apk清單的檔案為MainManifest.xml)裡查詢符合這兩個屬性的activity,然後啟動它。

  查詢過程是怎樣的呢?

  我猜測:在安裝某個apk的時候,android系統會把這個apk的清單檔案裡內容複製一份至系統的某個清單檔案裡(假如這個系統存放全部apk清單的檔案為MainManifest.xml)

  當 某個Activity用startActivity(intentOther)方法向系統傳送了一個intent(假如為intentOther),那麼 android系統會去查詢這個MainManifest.xml裡註冊的<intent-filter >屬性,

  查詢到符合這個 intentOther 的就啟動這個Activity,如果有多個這樣的Activity符合條件的話,就跳出一個對話方塊讓使用者選擇究竟要啟動哪一個

  上面那個自定義的Action字串("cc.android/myaction.leo",當然也可以寫成這樣"cc.android.myaction.leo",同時AndroidManifest.xml裡也要寫成這樣)是系統唯一的,

  所以系統很容易就能匹配到。

b.任何一個需要隱式啟動的Activity都必須要有這項:<category android:name="android.intent.category.DEFAULT"/>

例外情況是:android.intent.category.MAIN和android.intent.category.LAUNCHER的filter中沒有必要加入android.intent.category.DEFAULT,當然加入也沒有問題

c.假如有兩個Activity,它們的在AndroidManifest.xml裡配置如下:

<activity android:name="MyActivityOne" android:label="@string/activityOne">

<intent-filter>

<action android:name="hello.leo.liao" />

<action android:name="hello.leo.leo" />

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

</intent-filter>

</activity>

<activity android:name=".MyActivityTwo" android:label="@string/activityTwo">

<intent-filter>

<action android:name="hello.leo.liao" />

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

</intent-filter>

</activity>

在MainActivity.java裡傳送一個intent:

intent.setAction( "hello.leo.liao");

//不加下面這行也行,因為intent的這個屬性預設值即系Intent.CATEGORY_DEFAULT

intent.addCategory(Intent.CATEGORY_DEFAULT);

startActivity( intent );

這樣的話,android系統會跳出一個對話方塊讓你選擇啟動哪一個Activity(MyActivityOne還是MyActivityTwo)

如果把上面的intent.setAction( "hello.leo.liao");改為intent.setAction( "hello.leo.leo");的話,就自動匹配到MyActivityOne

就是說如果category和action都相同的話,會跳出一個對話方塊讓使用者選擇要啟動哪一個activity;

如果category相同,而action不相同,就可以匹配到相應的activity

d.單單靠新增addCategory屬性不能匹配,如:

Intent intent = new Intent();

intent.addCategory(Intent.CATEGORY_DEFAULT);

intent.addCategory("android.intent.category.hello");

startActivity(intent);

<activity android:name=".MyActivityTwo" android:label="@string/activityTwo">

<intent-filter>

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

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

</intent-filter>

</activity>

e.當匹配不上任何Activity的話,會發生異常,跳出對話方塊:很抱歉...某某應用程式意外停止,請重試。

f.Service和BroadcastReceiver 同理

(2) 根據Action和Data匹配

<activity android:name=".MyActivityTwo" android:label="@string/activityTwo">

<intent-filter>

             <action android:name="android.intent.action.leo"></action>

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

                <data android:scheme="x-id"></data>

</intent-filter>

</activity>

//Uri uri = Uri.parse("x-id://www.google.com/getDetails?id=123");//這個也可以

//Uri uri = Uri.parse("x-id");//這個不行

//Uri uri = Uri.parse("x-id://");這個可以

Uri uri = Uri.parse("x-id:");//這個可以

            Intent in = new Intent();

            in.setAction("android.intent.action.leo");//去掉這行不行,單靠data不能匹配

            in.addCategory(Intent.CATEGORY_DEFAULT);//可以去掉這行,因為intent的預設category值即系Intent.CATEGORY_DEFAULT

            in.setData(uri);//去掉這行不行

            startActivity(in);

總結:如果在AndroidManifest.xml裡面指定了<data>這行,那麼,需要匹配到它的話,在程式碼裡必須要設定intent的data,如上面的in.setData(uri)    

Data的語法:

<data android:host="string"

      android:mimeType="string"

      android:path="string"

      android:pathPattern="string"

      android:pathPrefix="string"

      android:port="string"

      android:scheme="string" />

Uri的格式:scheme://host:port/path or pathPrefix or pathPattern

如果scheme沒有指定,那其它的屬性均無效;

如果host沒有指定,那麼port,path,pathPrefix,pathPattern均無效;

如果在manifest裡這樣寫:<data android:scheme="something" android:host="project.example.com" />

那麼Uri uri = Uri.parse("something://project.example.com"); 才可以匹配

再如:

<data android:scheme="something" android:host="project.example.com" android:port="80"/>

等同於這樣寫:

<data android:scheme="something"/>

<data android:host="project.example.com"/>

<data android:port="80"/>

那麼Uri uri = Uri.parse("something://project.example.com:80"); 才可以匹配

不知為何,下面這個不行:

<data android:scheme="content" android:host="com.example.project" android:port="200" android:path="folder/subfolder/etc"/>

Uri uri = Uri.parse("content://com.example.project:200/folder/subfolder/etc")

下面這樣也不行

<data android:scheme="content" android:host="com.example.project" android:port="200" android:path="folder"/>

Uri uri = Uri.parse("content://com.example.project:200/folder")

可以有多個data,只需匹配其中一個即可

<activity android:name=".MyActivityTwo" android:label="@string/activityTwo">

<intent-filter>

             <action android:name="android.intent.action.leo"></action>

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

                <data android:scheme="x-id"/>

                <data android:scheme="something"/>

</intent-filter>

</activity>

           Intent in = new Intent();

            in.setAction("android.intent.action.leo");

            in.addCategory(Intent.CATEGORY_DEFAULT);

            in.setData(Uri.parse("something:"));//或者用這個亦可in.setData(Uri.parse("x-id:"));            

            startActivity(in);      

(3)  根據action和data的mimeType屬性匹配          

<activity android:name=".MyActivityTwo" android:label="@string/activityTwo">

<intent-filter>

<action android:name="android.intent.action.VIEW"></action>

<data android:mimeType="vnd.android.cursor.dir/vnd.google.note" />

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

</intent-filter>

</activity>

在java程式碼裡這樣寫就可以匹配到這個activity:

            Intent in = new Intent();

            in.setAction("android.intent.action.VIEW");

            in.addCategory(Intent.CATEGORY_DEFAULT);//可去掉,因為Category預設值即系Intent.CATEGORY_DEFAULT

            in.setType("vnd.android.cursor.dir/vnd.google.note");

            startActivity(in); 

  單靠data的mimeType屬性不能匹配,就算這個mimeType是唯一的也不行(比如in.setType("leo.android.cursor.dir/vnd.google.leo");),需要有一個action配合   

  可以有多個mimeType,在java程式碼裡只需匹配其中一個即可:

  <activity android:name=".MyActivityTwo" android:label="@string/activityTwo">

<intent-filter>

  <intent-filter>

<action android:name="android.intent.action.VIEW"></action>

<data android:mimeType="leo.android.cursor.dir/vnd.google.leo" />

<data android:mimeType="leo.android.cursor.dir/vnd.google.liao" />

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

</intent-filter>

</activity>

這樣可以啟動MyActivityTwo這個Activity:

Intent in = new Intent();

            in.setAction("android.intent.action.VIEW");

            in.addCategory(Intent.CATEGORY_DEFAULT);//可去掉,因為Category預設值即系Intent.CATEGORY_DEFAULT

            in.setType("leo.android.cursor.dir/vnd.google.liao");

            startActivity(in);

      或者這樣也可以啟動MyActivityTwo這個Activity:   

      Intent in = new Intent();

            in.setAction("android.intent.action.VIEW");

            in.addCategory(Intent.CATEGORY_DEFAULT);//可去掉,因為Category預設值即系Intent.CATEGORY_DEFAULT

            in.setType("leo.android.cursor.dir/vnd.google.leo");

            startActivity(in);

  scheme和mimeType只能有其中一個,下面這樣通不過

  AndroidManifest.xml裡:

  <data android:scheme="something" android:host="project.example.com"

android:port="80"

android:mimeType="leo.android.cursor.dir/vnd.google.leo" />  

<data android:scheme="something" android:host="project.example.com"

android:port="80" />

<data android:mimeType="leo.android.cursor.dir/vnd.google.leo" />

java程式碼裡:

匹配不上:

Intent in = new Intent();

            in.setAction("android.intent.action.VIEW");

            Uri uri = Uri.parse("something://project.example.com:80");

            in.setData(uri);

            in.setType("leo.android.cursor.dir/vnd.google.leo");

            startActivity(in);                  

            這樣還是匹配不上:

            Intent in = new Intent();

            in.setAction("android.intent.action.VIEW");

//            Uri uri = Uri.parse("something://project.example.com:80");

//            in.setData(uri);

            in.setType("leo.android.cursor.dir/vnd.google.leo");

            startActivity(in);

            這樣還是匹配不上:

            Intent in = new Intent();

            in.setAction("android.intent.action.VIEW");

            Uri uri = Uri.parse("something://project.example.com:80");

            in.setData(uri);

//            in.setType("leo.android.cursor.dir/vnd.google.leo");

            startActivity(in);

(4)   一個Activity裡可以有多對<intent-filter></intent-filter>  只要匹配其中一對,即可啟動這個Activity

<activity android:name=".MyActivityTwo" android:label="@string/activityTwo">

<intent-filter>

<action android:name="android.intent.action.VIEW"></action>

<data android:scheme="something" android:host="project.example.com"

android:port="80" />

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

</intent-filter>

<intent-filter>

<action android:name="android.intent.action.VIEW"></action>

<data android:mimeType="leo.android.cursor.dir/vnd.google.leo" />

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

</intent-filter>

<intent-filter>

<action android:name="hello.hi.liao"></action>

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

</intent-filter>

</activity>

java程式碼裡:

匹配第一對<intent-filter> 可以啟動MyActivityTwo這個Activity:

Intent in = new Intent();

            in.setAction("android.intent.action.VIEW");

            in.addCategory(Intent.CATEGORY_DEFAULT);//可去掉,因為Category預設值即系Intent.CATEGORY_DEFAULT

            Uri uri = Uri.parse("something://project.example.com:80");

            in.setData(uri);

            startActivity(in);

   匹配第二對<intent-filter> 也可以啟動MyActivityTwo這個Activity:    

   Intent in = new Intent();

   in.addCategory(Intent.CATEGORY_DEFAULT);//可去掉,因為Category預設值即系Intent.CATEGORY_DEFAULT

            in.setAction("android.intent.action.VIEW");

            in.setType("leo.android.cursor.dir/vnd.google.leo");

            startActivity(in);    

  匹配第三對<intent-filter> 也可以啟動MyActivityTwo這個Activity:      

  Intent in = new Intent();

            in.setAction("hello.hi.liao");

            in.addCategory(Intent.CATEGORY_DEFAULT);//可去掉,因為Category預設值即系Intent.CATEGORY_DEFAULT

            startActivity(in);      

全部總結:

1.  <action/>包含在 <intent-filter></intent-filter> 標籤對裡,而且是必不可少的!不管以哪一種方式來匹配,都不可缺少這個<action/> ,可以有多個,至少要有一個。

如有多個的,話只需要匹配其中一個即可找到這個activity

<action>裡的屬性值大多數是在Intent裡定義的,比如<action android:name="android.intent.action.VIEW"/>裡的屬性值就等於 Intent.ACTION_VIEW,

在這個Intent類裡以ACTION開頭定義的常量都是。當然,也可以自定義。  

2.  任何一個需要隱式啟動的Activity都必須要有這項:<category android:name="android.intent.category.DEFAULT"/>

例外情況是:android.intent.category.MAIN和android.intent.category.LAUNCHER的filter中沒有必要加入android.intent.category.DEFAULT,當然加入也沒有問題  

<category> 裡的屬性值大多數是在Intent裡定義的,比如  <category android:name="android.intent.category.DEFAULT"/>裡的屬性值就等於 Intent.CATEGORY_DEFAULT,

在這個Intent類裡以CATEGORY開頭定義的常量都是。當然,也可以自定義。  

3.一個Activity裡可以有多對<intent-filter></intent-filter>  只要匹配其中一對,即可啟動這個Activity

4.在<intent-filter></intent-filter>裡可以有多個<data android:mimeType="xxxx"/>,只需匹配其中一個即可.注意:不可以同時出現第5點的標籤對,即下面這條。

5. 在<intent-filter></intent-filter>裡可以有多個<data android:scheme="xxxx" android:host="yyyy" android:port="uuu"/>,只需匹配其中一個即可。

語法:

<data android:host="string"

      android:mimeType="string"

      android:path="string"

      android:pathPattern="string"

      android:pathPrefix="string"

      android:port="string"

      android:scheme="string" />

      可以分開寫,如:

      <data android:scheme="something" android:host="project.example.com" android:port="80"/>

等同於這樣寫:

<data android:scheme="something"/>

<data android:host="project.example.com"/>

<data android:port="80"/>

在java程式碼裡,Uri的格式:scheme://host:port/path or pathPrefix or pathPattern

注意:不可以同時出現第4點的標籤對,即上面那條。

6.在<intent-filter></intent-filter>裡可以有多個<action android:name="xxxx"> ,只需匹配其中一個即可。

7.當匹配不上任何Activity的話,會發生異常,跳出對話方塊:很抱歉...某某應用程式意外停止,請重試。

8. 上面所說的全部適用於Service和BroadcastReceiver,只需把<activity ...></activity>換成<service ...></service>或<receiver ...></receiver>即可。

 在被啟動的Activity(本例為MyActivityTwo)裡接收資料:

Intent intent = getIntent();

String intentCategories = intent.getCategories()

String intentType = intent.getType();

Uri uri = intent.getData();

String uriScheme = uri.getScheme();

String uriPath = uri.getPath();

String uriHost = uri.getHost();

String uriEncodedPath = uri.getEncodedPath();

請參考:packages\apps\HTMLViewer\src\com\android\htmlviewer\HTMLViewerActivity.java