Android中应用与应用通信,以及数据传递

本文详细介绍了如何在Android平台上实现两个应用间的交互,包括从A应用进入B应用的逻辑判断与跳转,B应用退回A应用的处理,以及A应用接收B应用退出时的数据传递。同时,针对安全问题,文章提出了manifest权限控制和activity声明的修改建议,确保应用间交互的安全性和稳定性。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

一、前言

某海外应用A有需求如下:在A应用首页,提供B应用的入口。点击入口时,判断B应用是否已安装,未安装则跳转Google商店安装,已安装则直接进入B应用。当退回A应用时需回传一个参数。

A应用和B应用交互时序图如下:

根据需求和时序图,本文从如下三个方面展开来讲:A应用进入B应用、B应用退回A应用、A应用处理B应用退出。

二、A应用进入B应用方案

点击A应用入口时,判断B应用是否已安装,未安装则跳转Google商店安装,已安装则直接进入B应用;

2.1 判断应用是否已安装

/**
 * 检测某个应用是否已安装
 *
 * @param context 上下文
 * @param pkgName 包名
 * @return 应用是否已安装
 */
public static boolean isAppInstalled(Context context, String pkgName) {
    if (context == null || TextUtils.isEmpty(pkgName)) {
        return false;
    } else {
        try {
            return context.getPackageManager().getApplicationInfo(pkgName, 0).enabled;
        } catch (PackageManager.NameNotFoundException var3) {
            return false;
        }
    }
}

2.2 跳转 Google 商店安装 APK

/**
 * 跳转 Google 商店安装 APK
 *
 * @param context 上下文
 */
public static void gotoGooglePlay(Context context) {
    if (context == null) {
        return;
    }
    try {
        Intent intent = new Intent(Intent.ACTION_VIEW, Uri.parse("market://details?id=" + context.getPackageName()));
        intent.setPackage("com.android.vending");
        intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
        context.startActivity(intent);
    } catch (Exception e) {
        e.printStackTrace();
    }
}

 2.3 进入B应用

/**
 * 进入B应用
 *
 * @param context 上下文
 */
public static void enterGameApp(Context context) {
    if (context == null) {
        return;
    }
    try {
        Intent intent = new Intent("action_name_B");
        intent.setPackage("package_name_B");
        intent.putExtra("extra_tag_B", "value");
        context.startActivity(intent);
    } catch (Exception e) {
        e.printStackTrace();
    }
}

三、B应用退回A应用方案

注:参考Android应用间切换的机制,B应用应该是先主动调用Intent退回到A应用,然后销毁自身应用。

3.1 代码参考

Intent intent = new Intent("action_name_A");
intent.setPackage("package_name_A");
intent.putExtra("extra_tag_A", "value");
startActivity(intent)

3.2 manifest声明使用权限

<uses-permission android:name="app.permission.OPEN_MAIN_PAGE" />

四、A应用处理B应用退出方案

4.1 manifest 设计

对于不同应用间的数据传递需求,首先考虑到Intent携带参数的方式。从B应用退出到A应用时,A应用的返回处理逻辑放在主页是比较好的(和直接退出B应用返回上一个应用的UI交互是一致的)。

所以直接更改manifest的intent-filter如下:

<activity
    android:name=".main.activity.MainActivity"
    android:exported="true"
    android:launchMode="singleTask"
    android:theme="@style/ActivityTheme">
    <intent-filter>
        <action android:name="action_name_A" />
        <category android:name="android.intent.category.DEFAULT" />
    </intent-filter>
</activity>

原来manifest中是这样的:

<activity
    android:name=".main.activity.MainActivity"
    android:launchMode="singleTask"
    android:theme="@style/ActivityTheme" />

但考虑到对Android的组件增加了intent-filter或显示指定exported为true后,会出现安全问题,于是增加了permission控制,在manifest声明了如下permission:

<permission android:name="app.permission.OPEN_MAIN_PAGE" />

同时需要在activity声明中增加permission控制:

android:permission="app.permission.OPEN_MAIN_PAGE"

最终版manifest的声明如下:

<permission android:name="app.permission.OPEN_MAIN_PAGE" />
 
<application android:allowBackup="false">
    <activity
        android:name=".main.activity.MainActivity"
        android:exported="true"
        android:launchMode="singleTask"
        android:permission="app.permission.OPEN_MAIN_PAGE"
        android:theme="@style/ActivityTheme">
        <intent-filter>
            <action android:name="action_name_A" />
            <category android:name="android.intent.category.DEFAULT" />
        </intent-filter>
    </activity>
</application>

4.2 代码设计

退出B应用时,需要使用Intent通过extra传递数据给A应用。A应用此处需要考虑的问题有:

  1. 在Activity的哪个回调方法中获取Intent的extra值?
  2. Activity在后台被销毁了,这种情况是否考虑?
  3. 直接退出B应用,是不需要传递数据给A应用的,是否考虑此种情况?

下面一起来分析一下:

首先,我们知道重复打开一个,启动模式为SingonTask的Activity,声明周期回调顺序如下:onNewIntent、onRestart、onStart、onResume,那么我们在onNewIntent回调中处理数据传递较好,接着讨论第二个问题;

其次,如果Activity在后台被销毁了,就不会走onNewIntent,会重新走onCreat、onStart、onResume,所以需要在onCreate做一个保障措施;

最后,因为正常退出B应用不需要传递数据(比如从Launcher启动B应用),但可能存在任务栈中A应用在B应用的下一级,那么退出B应用还是会回到A应用,此时不会调用onNewIntent方法,会调用onRestart、onStart、onResume。(此处主要想说明在onResume中去处理数据不太合理)。

所以,最后在A应用中处理B应用退出的代码如下:

@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    mPresenter.handleIntent(getIntent());
}
 
@Override
protected void onNewIntent(Intent intent) {
    super.onNewIntent(intent);
    setIntent(intent);
    mPresenter.handleIntent(intent);
}
 
@Override
public void handleIntent(Intent intent) {
    String exerciseTime = intent.getStringExtra("EXTRA_ACTION");
    Log.i(TAG, "handleIntent: exerciseTime = " + exerciseTime);
}

五、小结

         走一段令人留恋的路,做一个不负自己的人。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值