[TOC]
# 使用Intent打開第三方應用的方式
## 通過包名打開
**必要條件**
需要第三方應用配置有默認的入口Activity。
**打開方式**
```java
Intent intent = context.getPackageManager().getLaunchIntentForPackage("com.mmbox.xbrowser.pro");
startActivity(intent);
```
當目標應用不存在或沒有默認的入口Activity時,會報如下錯誤:
```plain
Attempt to invoke virtual method 'java.lang.String android.content.Intent.toString()' on a null object reference
```
因此啟動三方應用前應進行非空判斷:
```java
Intent intent = context.getPackageManager().getLaunchIntentForPackage("com.mmbox.xbrowser.pro");
if (intent == null) {
Toast.makeText(context, "找不到該應用", Toast.LENGTH_SHORT).show();
} else {
startActivity(intent);
}
```
## 通過包名和Activity打開
**必要條件**
1、需要知道三方應用的包名和Activity名
2、三方應用清單文件中,目標Activity的屬性`Export`需配置為`true`
**打開方式**
```java
Intent intent = new Intent();
intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
ComponentName componentName = new ComponentName("com.mmbox.xbrowser.pro", "com.mmbox.xbrowser.BrowserActivity");
intent.setComponent(componentName);
startActivity(intent);
```
當目標應用不存在、目標Activity不存在、目標Activity未被設置為`Exported=true`時,會報如下錯誤:
```plain
android.content.ActivityNotFoundException: Unable to find explicit activity class {xx.xx.xx/xx.xx.xxActivity}; have you declared this activity in your AndroidManifest.xml?
```
因此啟動前需進行判斷:
```java
Intent intent = new Intent();
intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
ComponentName componentName = new ComponentName("com.mmbox.xbrowsesr.pro", "com.mmbox.xbrowser.BrowserActivity");
intent.setComponent(componentName);
if (intent.resolveActivityInfo(getPackageManager(), PackageManager.MATCH_DEFAULT_ONLY) != null) {
startActivity(intent);
} else {
Toast.makeText(context, "找不到應用", Toast.LENGTH_SHORT).show();
}
```
此處不能使用Intent.resolveActivty方法判斷,具體可參考后面的源碼部分。
## 隱式啟動
**必要條件**
1、IntentFilter中至少有一個action,至少有一個Category,可沒有Data和Type
2、如果有Data,參數中Data必須符合Data規則
3、Action和Category必須同時匹配Activity中的一個Action和一個Category
**打開方式**
```java
Uri uri = Uri.parse("http://www.baidu.com");
Intent intent = new Intent(Intent.ACTION_VIEW, uri);
startActivity(intent);
```
啟動前需進行判斷:
```java
Uri uri = Uri.parse("http://www.baidu.com");
Intent intent = new Intent(Intent.ACTION_VIEW, uri);
if (intent.resolveActivity(getPackageManager()) != null) {
startActivity(intent);
} else {
Toast.makeText(context, "找不到應用", Toast.LENGTH_SHORT).show();
}
```
## 源碼分析
Intent的resolveActivity方法源碼如下:
```java
public ComponentName resolveActivity(@NonNull PackageManager pm) {
if (mComponent != null) {
return mComponent;
}
ResolveInfo info = pm.resolveActivity(this, PackageManager.MATCH_DEFAULT_ONLY);
if (info != null) {
return new ComponentName(
info.activityInfo.applicationInfo.packageName,
info.activityInfo.name);
}
return null;
}
```
Intent的resolveActivityInfo方法源碼如下:
```java
public ActivityInfo resolveActivityInfo(PackageManager pm, @PackageManager.ComponentInfoFlags int flags) {
ActivityInfo ai = null;
if (mComponent != null) {
ai = pm.getActivityInfo(mComponent, flags);
} else {
ResolveInfo info = pm.resolveActivity(this, PackageManager.MATCH_DEFAULT_ONLY | flags);
if (info != null) {
ai = info.activityInfo;
}
}
return ai;
}
```
resolveActivity和resolveActivityInfo兩個方法名稱相似,但是返回值卻是不同的。前者返回的是ComponentName,后者返回的是ActivityInfo。在打開三方應用指定Activity時,需使用resolveActivityInfo方法進行判斷。
# 總結
1、使用PackageManager.getLaunchIntentForPackage方法時,直接判斷返回的Intent是否為空即可
2、使用Intent.setComponent方法時,需使用`Intent.resolveActivityInfo()`或者`packageManager.queryIntentActivities()`兩種方式判斷
3、隱式啟動時,使用`Intent.resolveActivity()`、`Intent.resolveActivityInfo()`、`packageManager.queryIntentActivities()`三種方式均可
參考文檔:[https://likfe.com/2017/08/30/android-is-intent-available/](https://likfe.com/2017/08/30/android-is-intent-available/)
- 導讀
- Java知識
- Java基本程序設計結構
- 【基礎知識】Java基礎
- 【源碼分析】Okio
- 【源碼分析】深入理解i++和++i
- 【專題分析】JVM與GC
- 【面試清單】Java基本程序設計結構
- 對象與類
- 【基礎知識】對象與類
- 【專題分析】Java類加載過程
- 【面試清單】對象與類
- 泛型
- 【基礎知識】泛型
- 【面試清單】泛型
- 集合
- 【基礎知識】集合
- 【源碼分析】SparseArray
- 【面試清單】集合
- 多線程
- 【基礎知識】多線程
- 【源碼分析】ThreadPoolExecutor源碼分析
- 【專題分析】volatile關鍵字
- 【面試清單】多線程
- Java新特性
- 【專題分析】Lambda表達式
- 【專題分析】注解
- 【面試清單】Java新特性
- Effective Java筆記
- Android知識
- Activity
- 【基礎知識】Activity
- 【專題分析】運行時權限
- 【專題分析】使用Intent打開三方應用
- 【源碼分析】Activity的工作過程
- 【面試清單】Activity
- 架構組件
- 【專題分析】MVC、MVP與MVVM
- 【專題分析】數據綁定
- 【面試清單】架構組件
- 界面
- 【專題分析】自定義View
- 【專題分析】ImageView的ScaleType屬性
- 【專題分析】ConstraintLayout 使用
- 【專題分析】搞懂點九圖
- 【專題分析】Adapter
- 【源碼分析】LayoutInflater
- 【源碼分析】ViewStub
- 【源碼分析】View三大流程
- 【源碼分析】觸摸事件分發機制
- 【源碼分析】按鍵事件分發機制
- 【源碼分析】Android窗口機制
- 【面試清單】界面
- 動畫和過渡
- 【基礎知識】動畫和過渡
- 【面試清單】動畫和過渡
- 圖片和圖形
- 【專題分析】圖片加載
- 【面試清單】圖片和圖形
- 后臺任務
- 應用數據和文件
- 基于網絡的內容
- 多線程與多進程
- 【基礎知識】多線程與多進程
- 【源碼分析】Handler
- 【源碼分析】AsyncTask
- 【專題分析】Service
- 【源碼分析】Parcelable
- 【專題分析】Binder
- 【源碼分析】Messenger
- 【面試清單】多線程與多進程
- 應用優化
- 【專題分析】布局優化
- 【專題分析】繪制優化
- 【專題分析】內存優化
- 【專題分析】啟動優化
- 【專題分析】電池優化
- 【專題分析】包大小優化
- 【面試清單】應用優化
- Android新特性
- 【專題分析】狀態欄、ActionBar和導航欄
- 【專題分析】應用圖標、通知欄適配
- 【專題分析】Android新版本重要變更
- 【專題分析】唯一標識符的最佳做法
- 開源庫源碼分析
- 【源碼分析】BaseRecyclerViewAdapterHelper
- 【源碼分析】ButterKnife
- 【源碼分析】Dagger2
- 【源碼分析】EventBus3(一)
- 【源碼分析】EventBus3(二)
- 【源碼分析】Glide
- 【源碼分析】OkHttp
- 【源碼分析】Retrofit
- 其他知識
- Flutter
- 原生開發與跨平臺開發
- 整體歸納
- 狀態及狀態管理
- 零碎知識點
- 添加Flutter到現有應用
- Git知識
- Git命令
- .gitignore文件
- 設計模式
- 創建型模式
- 結構型模式
- 行為型模式
- RxJava
- 基礎
- Linux知識
- 環境變量
- Linux命令
- ADB命令
- 算法
- 常見數據結構及實現
- 數組
- 排序算法
- 鏈表
- 二叉樹
- 棧和隊列
- 算法時間復雜度
- 常見算法思想
- 其他技術
- 正則表達式
- 編碼格式
- HTTP與HTTPS
- 【面試清單】其他知識
- 開發歸納
- Android零碎問題
- 其他零碎問題
- 開發思路