news 2026/5/1 7:24:16

两个独立 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/4/23 13:55:28

NVIDIA正式推出Nemotron 3:用开放模型重塑AI多智能体时代

NVIDIA正式推出了全新的Nemotron 3系列开放模型家族&#xff0c;包括Nano、Super和Ultra三种不同规模的选择2025年12月15日&#xff0c;NVIDIA正式推出了全新的Nemotron 3系列开放模型家族&#xff0c;包括Nano、Super和Ultra三种不同规模的选择&#xff0c;旨在为全球开发者提…

作者头像 李华
网站建设 2026/4/28 2:28:05

PaperReding:《LLaMA: Open and Efficient Foundation Language Models》

论文链接: https://arxiv.org/pdf/2302.13971 这篇论文核心是 Meta AI 团队推出了一套叫LLaMA的开源大语言模型&#xff0c;简单说就是 “好用又好获取” 的 AI 模型&#xff0c;普通人或研究者不用依赖大公司的专属资源也能用上、研究它。总结为以下几点&#xff1a; 1. 核心…

作者头像 李华
网站建设 2026/4/30 14:21:20

10、Linux TCP/IP防火墙配置全解析

Linux TCP/IP防火墙配置全解析 1. 网络安全与防火墙的重要性 在当今数字化时代,网络安全对于企业和个人都至关重要。互联网为我们提供了强大的信息传播和获取工具,但同时也带来了诸如计算机犯罪、信息盗窃和恶意破坏等潜在风险。一个未经授权的人若进入计算机系统,可能会猜…

作者头像 李华
网站建设 2026/4/18 6:45:10

21、Linux网络新闻系统C News深度解析

Linux网络新闻系统C News深度解析 1. 网络新闻概述 网络新闻(Netnews),也就是Usenet新闻,至今仍是计算机网络中极为重要且备受重视的服务之一。尽管有人认为它充斥着大量未经请求的商业邮件和色情内容,但它依然保留了许多高质量的讨论组,在互联网普及之前,这些讨论组是…

作者头像 李华
网站建设 2026/4/27 15:40:45

Tustin 离散化及相关核心技术全解析

目录 1. 引言 2. Tustin 离散化基础 2.1 核心定义 2.2 核心映射公式 2.3 核心特性 3. 关键问题&#xff1a;频率混叠与非线性频率映射 3.1 频率混叠&#xff08;Frequency Aliasing&#xff09; 3.1.1 定义 3.1.2 成因与数学本质 3.1.3 工程影响与解决措施 3.2 Tust…

作者头像 李华
网站建设 2026/4/29 0:50:41

[MoeCTF 2022]nyanyanya

下载后用010发现存在LSB&#xff0c;一般线索就是在开头或结尾。 用Zsteg做&#xff0c;‌zsteg是一款用于检测PNG和BMP图片中隐藏数据的工具。 先解释一下LSB。LSB指的是一个二进制数中影响数值最小的那一位&#xff0c;一般是二进制数中最右边的位&#xff0c;其对数值的影响…

作者头像 李华