App相互喚醒的幾種方式

Lebens發表於2019-03-20

下文皆使用Client表示操作的App,Server表示需要被喚起的遠端App,Server的包名為“com.jxx.server”

1. ComponentName

使用ComponentName喚起Server步驟很簡單,需要注意的是Server的Activity需要在manifest配置種設定exported為true

Server的配置如下:

<activity android:name="com.jxx.server.ServerActivity"
    android:exported="true"/>                                                          
複製程式碼

Client呼叫如下:

Intent intent1 = new Intent();                                                                   
ComponentName componentName = new ComponentName("com.jxx.server", "com.jxx.server.ServerActivity");
intent1.setComponent(componentName);                                                             
intent1.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);                                                 
startActivity(intent1);                                                                          
複製程式碼

Intent中新增ComponentName還有另外一種寫法

Intent intent2 = new Intent();                                                                   
intent2.setClassName("jxx.com.server", "jxx.com.server.MainActivity");                                                             
intent2.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);                                                 
startActivity(intent1); 
複製程式碼

只不過setClassName內部幫我們new ComponentName的例項

public @NonNull Intent setClassName(@NonNull String packageName, @NonNull String className) {
    mComponent = new ComponentName(packageName, className);
    return this;
}
複製程式碼

既然是用Intent來喚起Activity,那就能使用Intent的特性,例如使用Bundle傳遞資料

Intent intent1 = new Intent();                                                                   
ComponentName componentName = new ComponentName("com.jxx.server", "com.jxx.server.ServerActivity");
intent1.setComponent(componentName);   

Bundle bundle1 = new Bundle();                     
bundle1.putString("remote_invoke", "from_client"); 
intent1.putExtras(bundle1);                        
                                                          
intent1.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);                                                 
startActivity(intent1);
複製程式碼

在Server端提取資料也很簡單

Bundle bundle = getIntent().getExtras();
複製程式碼

2. 隱式跳轉,Uri

Android中喚起撥號頁面是這樣的

Intent intent =  new Intent(Intent.ACTION_CALL,Uri.parse("tel:" + phoneNumber));
startActivity(intent);
複製程式碼

其實就是用Uri的形式喚起Server,並傳遞資料,我們來自己實現一下。 這種方式下,Server端的配置如下,必須新增要有action、data以及category:

<activity android:name=".SecondActivity">                              
                                                                       
    <intent-filter>                                                    
        <action android:name="com.jxx.server.ServerActivity" />
        <data                                                          
            android:host="com.jxx.server"                      
            android:scheme="ServerActivity" />                                 
        <category android:name="android.intent.category.DEFAULT" />    
    </intent-filter>                                                   
                                                                       
</activity>                                                            
複製程式碼

Client呼叫:

Intent intent2 = new Intent("com.jxx.server.ServerActivity");            
Uri uri = Uri.parse("ServerActivity://com.jxx.server?remote_invoke=from_client");
intent2.setData(uri);                                                            
startActivity(intent2);                                                          
複製程式碼

我們看到uri中?後面還加了"remote_invoke=from_client",這其實是用來給Server傳遞資料用的,我們可以在Server中解析出來

Uri uri = getIntent().getData();
String from = uri.getQueryParameter("remote_invoke");
//from = "from_client"
複製程式碼

這裡還有一個需要注意的點是,如果Client在呼叫時沒有指定Action,同時Server中又有多個Activity註冊了相同的scheme和host,那麼在頁面跳轉時,系統會彈框讓我們選擇跳轉到哪個頁面,如下圖所示:

App相互喚醒的幾種方式

3. 通過PackageManager喚起

只需要知道Server的包名即可

PackageManager packageManager = getPackageManager();                         
Intent intent3 = packageManager.getLaunchIntentForPackage("com.jxx.server");
if (intent3 != null) {                                                       
    startActivity(intent3);                                                  
}                                                                            
複製程式碼

4. 靜態廣播接收者

只需要Server端註冊一個靜態廣播接收者,在廣播接收者中跳轉Activity即可,客戶端只需要傳送一個廣播。

Server定義廣播接收者:

public class ServerBroadCastReceiver extends BroadcastReceiver {

    @Override
    public void onReceive(Context context, Intent intent) {
        Intent intent1 = new Intent(context, MainActivity.class);
        //注意,這裡必須要新增這個flag,
        //原因在於這裡的context並不是一個Activity型別的context,無法直接開啟activity
        intent1.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
        context.startActivity(intent1);
    }
}
複製程式碼

並在manifest中註冊為靜態廣播接收者,並定義action

<receiver                                                       
    android:name=".ServerBroadCastReceiver"                     
    android:enabled="true"                                      
    android:exported="true">                                    
    <intent-filter>                                             
        <action android:name="server.ServerBroadCastReceiver" />
    </intent-filter>                                                                                              
</receiver>                                                     
複製程式碼

Client中傳送廣播即可

Intent intent4 = new Intent("server.ServerBroadCastReceiver"); 
//這裡加上componentName用於解決8.0以上不能喚起的問題                                     
ComponentName componentName = new ComponentName("com.jxx.server", "com.jxx.server.ServerBroadCastReceiver");
intent4.setComponent(componentName);                                                                        
sendBroadcast(intent4);                                                                                     
複製程式碼

5. Service

Android Service詳解(二)中我們介紹瞭如何通過Service實現IPC通訊,這當然也能用來喚起App,這裡就不再過多介紹了,有興趣的同學可以點選檢視。

相關文章