1. 程式人生 > >Android理解:顯式和隱式Intent

Android理解:顯式和隱式Intent

Intent是Android初學者比較難理解的一個東西。我這裡以通俗易懂的語言和通俗易懂的程式碼,讓初學者簡單感受一下Intent。

intent就是意圖的意思。Intent分兩種:顯式(Explicit intent)和隱式(Implicit intent)

一、顯式(設定Component)

顯式,即直接指定需要開啟的activity對應的類。
以下多種方式都是一樣的,實際上都是設定Component直接指定Activity類的顯式Intent,由MainActivity跳轉到SecondActivity:
1、構造方法傳入Component,最常用的方式

Intent intent = new Intent(this, SecondActivity.class);
startActivity(intent);

2、setComponent方法
ComponentName componentName = new ComponentName(this, SecondActivity.class);
// 或者ComponentName componentName = new ComponentName(this, "com.example.app016.SecondActivity");
// 或者ComponentName componentName = new ComponentName(this.getPackageName(), "com.example.app016.SecondActivity");

Intent intent = new Intent();
intent.setComponent(componentName);
startActivity(intent);

3、setClass/setClassName方法
Intent intent = new Intent();

intent.setClass(this, SecondActivity.class);
// 或者intent.setClassName(this, "com.example.app016.SecondActivity");
// 或者intent.setClassName(this.getPackageName(), "com.example.app016.SecondActivity");
		
startActivity(intent);

顯式Intent通過Component可以直接設定需要呼叫的Activity類,可以唯一確定一個Activity,意圖特別明確,所以是顯式的。設定這個類的方式可以是Class物件(如SecondActivity.class),也可以是包名加類名的字串(如"com.example.app016.SecondActivity")。這個很好理解,在應用程式內部跳轉介面常用這種方式。

二、隱式


隱式,即不是像顯式的那樣直接指定需要呼叫的Activity,隱式不明確指定啟動哪個Activity,而是設定Action、Data、Category,讓系統來篩選出合適的Activity。篩選是根據所有的<intent-filter>來篩選。

下面以Action為例:

AndroidManifest.xml檔案中,首先被呼叫的Activity要有一個帶有<intent-filter>並且包含<action>的Activity,設定它能處理的Intent,並且category設為"android.intent.category.DEFAULT"。action的name是一個字串,可以自定義,例如我在這裡設成"abcdefg":

        <activity
            android:name="com.example.app016.SecondActivity">
            <intent-filter>
                <action android:name="abcdefg"/>
                <category android:name="android.intent.category.DEFAULT"/>
            </intent-filter>
        </activity>

然後,在MainActivity,才可以通過這個action name找到上面的Activity。下面兩種方式分別通過setAction和構造方法方法設定Action,兩種方式效果相同。

1、setAction方法

Intent intent = new Intent();
intent.setAction("abcdefg");
startActivity(intent);

2、構造方法直接設定Action
Intent intent = new Intent("abcdefg");
startActivity(intent);

通過設定Action字串,表明自己的意圖,即我想幹嘛,需要由系統解析,找到能夠處理這個Intent的Activity並啟動。比如我想打電話,則可以設定Action為"android.intent.action.DIAL"字串,表示打電話的意圖,系統會找到能處理這個意圖的Activity,例如調出撥號面板。

有幾點需要注意:
1、
這個Activity其他應用程式也可以呼叫,只要使用這個Action字串。這樣應用程式之間互動就很容易了,例如手機QQ可以呼叫QQ空間,可以呼叫騰訊微博等。
因為如此,為了防止應用程式之間互相影響,一般命名方式是包名+Action名,例如這裡命名"abcdefg"就很不合理了,就應該改成"com.example.app016.MyTest"。
2、
當然,你可以在自己的程式中呼叫其他程式的Action。
例如可以在自己的應用程式中呼叫撥號面板:

Intent intent = new Intent(Intent.ACTION_DIAL);
// 或者Intent intent = new Intent("android.intent.action.DIAL");
// Intent.ACTION_DIAL是內建常量,值為"android.intent.action.DIAL"
startActivity(intent);


3、一個Activity可以處理多種Action
只要你的應用程式夠牛逼,一個Activity可以看網頁,打電話,發簡訊,發郵件。。。當然可以。
Intent的Action只要是其中之一,就可以開啟這個Activity。

        <activity
            android:name="com.example.app016.SecondActivity">
            <intent-filter>
                <!-- 可以處理下面三種Intent -->
                <action android:name="com.example.app016.SEND_EMAIL"/>
                <action android:name="com.example.app016.SEND_MESSAGE"/>
                <action android:name="com.example.app016.DAIL"/>
                <category android:name="android.intent.category.DEFAULT" />
            </intent-filter>
        </activity>

對於一個Action字串,系統有可能會找到一個Activity能處理這個Action,也有可能找到多個Activity,也可能一個都找不到。
1、找到一個Activity
很簡單,直接開啟這個Activity。這個不需要解釋。
2、找到多個Acyivity
系統會提示從多個activity中選擇一個開啟。
例如我們自己開發一個撥號面板應用程式,可以設定activity的<intent-filter>中Action name為"android.intent.action.DIAL",這樣別的程式呼叫撥號器時,使用者可以從Android自帶的撥號器和我們自己開發的撥號器中選擇。
        <activity
            android:name="com.example.app016.SecondActivity">
            <intent-filter>
                <action android:name="android.intent.action.DIAL"/>
                <category android:name="android.intent.category.DEFAULT" />
            </intent-filter>
        </activity>

這也就是當Android手機裝上UC瀏覽器後,開啟網頁時會彈出選擇Android自帶瀏覽器還是UC瀏覽器,可能都會遇到過。
3、一個Activity都沒找到
一個都沒找到的話,程式就會出錯,會丟擲ActivityNotFoundException。比如隨便寫一個action字串:
Intent intent = new Intent("asasasas");
startActivity(intent);


所以應該注意try catch異常。

Intent intent = new Intent("asasasas");
try
{
	startActivity(intent);
}
catch(ActivityNotFoundException e)
{
	Toast.makeText(this, "找不到對應的Activity", Toast.LENGTH_SHORT).show();
}
或者也可以使用Intent的resolveActivity方法判斷這個Intent是否能找到合適的Activity,如果沒有,則不再startActivity,或者可以直接禁用使用者操作的控制元件。
Intent intent = new Intent(Intent.ACTION_DIAL);
if(intent.resolveActivity(getPackageManager()) == null)
{
    // 設定控制元件不可用
}

注意resolveActivity方法返回值就是顯式Intent上面講到的ComponentName物件,一般情況下也就是系統找到的那個Activity。但是如果有多個Activity可供選擇的話,則返回的Component是com.android.internal.app.ResolverActivity,也就是使用者選擇Activity的那個介面對應的Activity,這裡不再深究。
Intent intent = new Intent(Intent.ACTION_DIAL);
ComponentName componentName = intent.resolveActivity(getPackageManager());
if(componentName != null)
{
    String className = componentName.getClassName();
    Toast.makeText(this, className, Toast.LENGTH_SHORT).show();
}