關於Android隱式啓動Activity

 

更多例子請參考:http://hi.baidu.com/wishwingliao/blog/item/0a38ccfce06f39e8fc037f85.html

 

隱式啓動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>即可。

 

9.剛參考了一下packages\apps\HTMLViewer\AndroidManifest.xml,第4和第5條應該是不衝突纔對,但是實際測試中卻是衝突,暫時未到找原因。匹配方式請看:用於打開HTML文件的intent

 

 

 在被啓動的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

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章