这两种方法(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.java的onCreate方法中:
// 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,或者想要标准化的协议 |
总结如下:
如果这两个 App 都要安装在同一台专用设备上(比如工业PDA、特定的E-bike中控屏):
使用方法一(直接唤起)最快。因为环境是可控的,您确定两个 App 都会安装,直接写包名跳过去就行,代码改动最小。如果您希望看起来像一个 App,用户体验更好:
可以把App A 做成一个“壳”(Launcher),桌面只显示 App A 的图标。App B 在 Manifest 里去掉LAUNCHERcategory(这样桌面上看不到 B 的图标)。
然后用户打开 A,点击某个按钮时,A 在后台启动 B 的 Activity。这样用户感觉就像是在同一个软件里操作一样。