news 2026/6/10 1:12:37

两个独立 App如何相互唤起

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
两个独立 App如何相互唤起

这两种方法(Intent直接唤起 和 DeepLink)的核心区别在于:它们不需要把代码合并在一起,而是保持两个独立的 APK(App A 和 App B),通过 Android 系统的通信机制让它们互相“对话”。

假设场景:

  • App A (主控端):想要打开另一个 App。
  • App B (被控端):拥有你想要展示的功能页面。

以下是详细教程:


方法一:通过包名/类名直接唤起 (Intent ComponentName)

这种方法最简单直接,适合App B 是你自己的,且你知道它的确切包名和类名的情况。

第一步:配置 App B (被唤起方)

在 App B 的AndroidManifest.xml中,找到你想要让外部启动的那个 Activity(比如MainActivity或者FunctionActivity)。

关键点:必须添加android:exported="true"

<!-- App B 的 AndroidManifest.xml --><manifestpackage="com.ebike.intercept"><!-- 假设这是 App B 的包名 --><application...><!-- 这里的 exported="true" 允许外部 App 启动它 --><activityandroid:name=".MainActivity"android:exported="true"><intent-filter><actionandroid:name="android.intent.action.MAIN"/><categoryandroid:name="android.intent.category.LAUNCHER"/></intent-filter></activity></application></manifest>
第二步:配置 App A (发起方) - 适配 Android 11+

从 Android 11 (API 30) 开始,为了隐私安全,App A 默认看不见App B 是否安装。你需要添加<queries>标签声明“我需要查询 App B”。

在 App A 的AndroidManifest.xml中:

<!-- App A 的 AndroidManifest.xml --><manifestpackage="com.example.appa"><!-- 必须添加这个,否则检测不到 App B 是否安装 --><queries><packageandroid:name="com.ebike.intercept"/><!-- 填 App B 的包名 --></queries><application...>...</manifest>
第三步:在 App A 的 Java 代码中编写跳转逻辑
// 在 App A 的点击事件中publicvoidjumpToAppB(){// 1. 定义 App B 的包名和类名StringpackageName="com.ebike.intercept";StringclassName="com.ebike.intercept.MainActivity";// 必须是完整路径// 2. 检查 App B 是否安装(为了防止 Crash)if(isAppInstalled(context,packageName)){Intentintent=newIntent();// 3. 设置 ComponentNameComponentNamecomponentName=newComponentName(packageName,className);intent.setComponent(componentName);// 4. (可选) 传递数据intent.putExtra("key_from_a","Hello from App A");// 5. 必须加 NEW_TASK 标记(如果在非 Activity 环境下启动)intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);startActivity(intent);}else{Toast.makeText(context,"请先安装 App B",Toast.LENGTH_SHORT).show();}}// 辅助方法:检查 App 是否安装privatebooleanisAppInstalled(Contextcontext,StringpackageName){try{context.getPackageManager().getPackageInfo(packageName,0);returntrue;}catch(PackageManager.NameNotFoundExceptione){returnfalse;}}

方法二:使用 DeepLink (自定义 URL Scheme)

这是更推荐的现代做法,它的耦合度更低。App A 不需要知道 App B 的具体类名,只需要像打开网页一样打开一个 URL(例如ebike://home),系统会自动找到 App B。

第一步:配置 App B (被唤起方)

你需要给 App B 的某个 Activity 配置一个“暗号”(Scheme)。

在 App B 的AndroidManifest.xml中:

<!-- App B 的 AndroidManifest.xml --><activityandroid:name=".DetailActivity"><!-- 这里的 intent-filter 是关键 --><intent-filter><!-- 必须有 ACTION_VIEW --><actionandroid:name="android.intent.action.VIEW"/><!-- 必须有 DEFAULT --><categoryandroid:name="android.intent.category.DEFAULT"/><!-- 允许浏览器唤起(可选,但推荐) --><categoryandroid:name="android.intent.category.BROWSABLE"/><!-- 定义暗号:ebike://open_detail --><!-- scheme: 协议头,host: 路径/主机名 --><dataandroid:scheme="ebike"android:host="open_detail"/></intent-filter></activity>
第二步:在 App B 中接收参数 (可选)

DetailActivity.javaonCreate方法中:

// App B 的 DetailActivity.java@OverrideprotectedvoidonCreate(BundlesavedInstanceState){super.onCreate(savedInstanceState);setContentView(R.layout.activity_detail);// 获取 Intent 中的数据Intentintent=getIntent();Uridata=intent.getData();if(data!=null){// 比如 URL 是 ebike://open_detail?msg=hello&id=100Stringmsg=data.getQueryParameter("msg");// 获取 "hello"Stringid=data.getQueryParameter("id");// 获取 "100"Log.d("DeepLink","收到参数: "+msg);}}
第三步:在 App A 中编写跳转逻辑

App A 现在根本不需要知道 App B 的包名或类名,只需要访问这个 URL。

// 在 App A 的点击事件中publicvoidjumpByDeepLink(){// 1. 构建 URI,并带上参数Stringurl="ebike://open_detail?msg=来自AppA的问候&id=666";Intentintent=newIntent(Intent.ACTION_VIEW,Uri.parse(url));intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);// 2. 检查系统中是否有 App 能响应这个 Intent// (注意:在 Android 11+ 上,如果 App A 没加 <queries>,这个 resolveActivity 也可能返回 null,// 但直接 startActivity 通常会弹出一个系统选择框,或者直接跳转)try{startActivity(intent);}catch(Exceptione){Toast.makeText(context,"没有检测到已安装的目标应用",Toast.LENGTH_SHORT).show();}}

总结对比

特性方法一:直接唤起 (Intent/ComponentName)方法二:DeepLink (URL Scheme)
原理“我要去某个具体地址找老王”“我要喊一声老王,谁是老王谁答应”
配置难度简单中等(需配置 IntentFilter)
耦合度(需硬编码包名类名,B重构改名则A会挂)(只需约定 URL 字符串)
灵活性只能 App 唤起App 和 网页浏览器都能唤起
参数传递Bundle (类型丰富)URL 字符串 (只能传 String)
适用场景两个 App 都是你开发的,且不做网页跳转需要从 H5 页面跳入 App,或者想要标准化的协议

总结如下:

  1. 如果这两个 App 都要安装在同一台专用设备上(比如工业PDA、特定的E-bike中控屏):
    使用方法一(直接唤起)最快。因为环境是可控的,您确定两个 App 都会安装,直接写包名跳过去就行,代码改动最小。

  2. 如果您希望看起来像一个 App,用户体验更好:
    可以把App A 做成一个“壳”(Launcher),桌面只显示 App A 的图标。App B 在 Manifest 里去掉LAUNCHERcategory(这样桌面上看不到 B 的图标)。
    然后用户打开 A,点击某个按钮时,A 在后台启动 B 的 Activity。这样用户感觉就像是在同一个软件里操作一样。

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/6/9 18:24:57

如何把两个android项目合二为一

将两个独立的 Android 项目合二为一是一个比较复杂的过程&#xff0c;不能简单地复制粘贴。 最推荐、最标准的方法是将其中一个项目作为一个模块 (Module) 导入到另一个主项目 (Main Project) 中。 这里有一个分步指南&#xff0c;假设您有两个项目&#xff1a; 项目 A&#xf…

作者头像 李华
网站建设 2026/6/9 4:04:24

FPGA中的 LUT6

LUT6 的能力远不止实现简单的逻辑门&#xff0c;它极其灵活。我们可以从基础功能到高级应用&#xff0c;再到扩展能力三个层面来全面回答。 一、 基础功能&#xff1a;实现任意6输入逻辑门 核心结论&#xff1a;一个 LUT6 可以实现任意一个 6 输入、1 输出的组合逻辑函数。 这意…

作者头像 李华
网站建设 2026/6/7 15:22:47

金仓智核:强调智能内核驱动一切,是数据库的“智慧核心”

兼容 是对企业历史投资的尊重 是确保业务平稳过渡的基石 然而 这仅仅是故事的起点 在数字化转型的深水区&#xff0c;企业对数据库的需求早已超越“语法兼容”的基础诉求。无论是核心业务系统的稳定运行&#xff0c;还是敏感数据的安全防护&#xff0c;亦或是复杂场景下的性能优…

作者头像 李华
网站建设 2026/6/10 12:35:32

linux项目自动化构建工具(make和makefile)

目录 什么是make和makefile 依赖关系和依赖方法 make的工作原理 什么是make和makefile make是一个构建自动化工具&#xff0c;用于自动编译和构建软件项目 makefile是一个配置文件&#xff0c;告诉make如何构建项目 make是一条指令&#xff0c;makefile是一个文件&#xf…

作者头像 李华