天天看點

android 互相喚醒 v2ex,App互相喚醒的幾種方式

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

1. ComponentName

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

Server的配置如下:

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:

android:host="com.jxx.server"

android:scheme="ServerActivity" />

複制代碼

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,那麼在頁面跳轉時,系統會彈框讓我們選擇跳轉到哪個頁面,如下圖所示:

android 互相喚醒 v2ex,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

android:name=".ServerBroadCastReceiver"

android:enabled="true"

android:exported="true">

複制代碼

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,這裡就不再過多介紹了,有興趣的同學可以點選檢視。