参考:
Android PackageManager 详解
Android应用间跳转
Android随笔之——PackageManager详解
Android系统为我们提供了很多服务管理的类,包括ActivityManager、PowerManager(电源管理)、AudioManager(音频管理)等。除此之外,还提供了一个PackageManger管理类,它的主要职责是管理应用程序包。 通过PackageManager,我们就可以获取应用程序信息。
提到PackageManager,就得提一下AndroidManifest.XML文件了。AndroidManifest.xml是Android应用程序中最重要的文件之一。它是Android程序的全局配置文件,是每个 android程序中必须的文件。它位于我们开发的应用程序的根目录下,描述了package中的全局数据,包括package中暴露的组件 (activities, services, 等等),以及他们各自的实现类,各种能被处理的数据和启动位置等重要信息。
因此,该文件提供了Android系统所需要的关于该应用程序的必要信息,即在该应用程序的任何代码运行之前系统所必须拥有的信息。
PackageManager获取的信息即来自AndroidManifest.XML。为了便于理解,从网上找了一张AnroidManifest.xml文件节点说明图:
PackageManager的功能:
1、安装,卸载应用
2、查询permission相关信息
3、查询Application相关信息(application,activity,receiver,service,provider及相应属性等)
4、查询已安装应用
5、增加,删除permission
6、清除用户数据、缓存,代码段等
PackageManager相关类和方法介绍:
1、PackageManager类
说明: 获得已安装的应用程序信息 。可以通过getPackageManager()方法获得。
常用方法:
public abstract PackageManager getPackageManager()
功能:获得一个PackageManger对象
public abstract Drawable getApplicationIcon(String packageName)
参数: packageName 包名
功能:返回给定包名的图标,否则返回null
public abstract ApplicationInfo getApplicationInfo(String packageName, int flags)
参数:
packagename 包名
flags 该ApplicationInfo是此flags标记,通常可以直接赋予常数0即可
功能:返回该ApplicationInfo对象
public abstract List<ApplicationInfo> getInstalledApplications(int flags)
参数:
flag为一般为GET_UNINSTALLED_PACKAGES,那么此时会返回所有ApplicationInfo。我们可以对ApplicationInfo
的flags过滤,得到我们需要的。
功能:返回给定条件的所有PackageInfo
public abstract List<PackageInfo> getInstalledPackages(int flags)
参数如上
功能:返回给定条件的所有PackageInfo
public abstract ResolveInfo resolveActivity(Intent intent, int flags)
参数:
intent 查寻条件,Activity所配置的action和category
flags: MATCH_DEFAULT_ONLY :Category必须带有CATEGORY_DEFAULT的Activity,才匹配
GET_INTENT_FILTERS :匹配Intent条件即可
GET_RESOLVED_FILTER :匹配Intent条件即可
功能 :返回给定条件的ResolveInfo对象(本质上是Activity)
public abstract List<ResolveInfo> queryIntentActivities(Intent intent, int flags)
参数同上
功能 :返回给定条件的所有ResolveInfo对象(本质上是Activity),集合对象
public abstract ResolveInfo resolveService(Intent intent, int flags)
参数同上
功能 :返回给定条件的ResolveInfo对象(本质上是Service)
public abstract List<ResolveInfo> queryIntentServices(Intent intent, int flags)
参数同上
功能 :返回给定条件的所有ResolveInfo对象(本质上是Service),集合对象
2、PackageItemInfo类
说明: AndroidManifest.xml文件中所有节点的基类,提供了这些节点的基本信息:label、icon、 meta-data。
它并不直接使用,而是由子类继承然后调用相应方法。
3、ApplicationInfo类 继承自 PackageItemInfo类
说明:获取一个特定引用程序中<application>节点的信息。
字段说明:
flags字段: FLAG_SYSTEM 系统应用程序
FLAG_EXTERNAL_STORAGE 表示该应用安装在sdcard中
常用方法继承至PackageItemInfo类中的loadIcon()和loadLabel()
String appName = resolveInfo.loadLabel(pm).toString();// 获取应用名称
String packageName = resolveInfo.activityInfo.packageName;// 包名
String className = resolveInfo.activityInfo.name;// 入口类名
4、ActivityInfo类 继承自 PackageItemInfo类
说明: 获得应用程序中<activity/>或者 <receiver/>节点的信息 。我们可以通过它来获取我们设置的任何属性,包括theme 、launchMode、launchmode等
常用方法继承至PackageItemInfo类中的loadIcon()和loadLabel()
5、ServiceInfo类 继承自 PackageItemInfo类
说明:与ActivityInfo类似,代表<service>节点信息
6、ResolveInfo类
说明:根据<intent>节点来获取其上一层目录的信息,通常是<activity>、<receiver>、<service>节点信息。
常用方法有loadIcon(PackageManager pm)和loadLabel(PackageManager pm)
java代码:
package com.example.administrator.mydb;
import android.app.Activity;
import android.content.ComponentName;
import android.content.Intent;
import android.content.pm.ApplicationInfo;
import android.content.pm.PackageInfo;
import android.content.pm.PackageManager;
import android.content.pm.ResolveInfo;
import android.os.Bundle;
import android.util.Log;
import android.view.View;
import android.widget.Button;
import android.widget.TextView;
import java.util.ArrayList;
import java.util.List;
public class MainActivity extends Activity {
private Intent intent = null;
private TextView textView;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
textView = ((TextView) this.findViewById(R.id.tv));
loadApps();
/*跳转微信界面*/
((Button) this.findViewById(R.id.btn_1)).setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
startActivity(getPackageManager().getLaunchIntentForPackage("com.tencent.mm"));
}
});
/*跳转其他应用,指定包名,不指定类名*/
((Button) this.findViewById(R.id.btn_main)).setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
//不知道类名,跳转到指定的包
PackageManager pm = getPackageManager();
PackageInfo pi = null;
try {
pi = pm.getPackageInfo("com.example.administrator.downloadimgdemo", 0);
} catch (PackageManager.NameNotFoundException e) {
e.printStackTrace();
}
Intent resolveIntent = new Intent(Intent.ACTION_MAIN, null);
resolveIntent.addCategory(Intent.CATEGORY_LAUNCHER);
resolveIntent.setPackage(pi.packageName);
//找到匹配intent的所有Activity,这个方法可以在intent跳转前调用用来检测是否有符合条件的activity
List<ResolveInfo> apps = pm.queryIntentActivities(resolveIntent, 0);
ResolveInfo ri = apps.iterator().next();
if (ri != null) {
String className = ri.activityInfo.name;
intent = new Intent(Intent.ACTION_MAIN);
intent.addCategory(Intent.CATEGORY_LAUNCHER);
ComponentName cn = new ComponentName("com.example.administrator.downloadimgdemo", className);
intent.setComponent(cn);
}
}
});
/*跳转其他应用,指定类名*/
((Button) this.findViewById(R.id.btn_jump)).setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
//知道要跳转应用的包名、类名
ComponentName comp = new ComponentName("com.example.administrator.downloadimgdemo",
"com.example.administrator.downloadimgdemo.GetAllImg");
final Intent intent = new Intent();
intent.setComponent(comp);
startActivity(intent);
}
});
}
// 列出普通应用程序
private void loadApps() {
List<PackageInfo> customApps = new ArrayList<PackageInfo>(); // 普通应用程序列表
List<PackageInfo> systemApps = new ArrayList<PackageInfo>(); // 系统应用程序列表
// 得到PackageManager对象
PackageManager pm = this.getPackageManager();
// 得到系统安装的所有程序包的PackageInfo对象
List<PackageInfo> packages = pm.getInstalledPackages(0);
StringBuffer sb = new StringBuffer();
for (PackageInfo pi : packages) {
// 列出普通应用
if ((pi.applicationInfo.flags & ApplicationInfo.FLAG_SYSTEM) <= 0) {
customApps.add(pi);
sb.append("》》》》普通pi======" + pi + "\n");
}
// 列出系统应用,总是感觉这里设计的有问题,希望高手指点
if ((pi.applicationInfo.flags & ApplicationInfo.FLAG_SYSTEM) > 0) {
systemApps.add(pi);
sb.append("《《《《系统pi======" + pi + "\n");
}
}
textView.setText(sb.toString());
}
}
布局:
<?xml version="1.0" encoding="utf-8"?>
<ScrollView
xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:id="@+id/activity_main"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:paddingLeft="@dimen/activity_horizontal_margin"
android:paddingRight="@dimen/activity_horizontal_margin"
android:paddingTop="@dimen/activity_vertical_margin"
android:paddingBottom="@dimen/activity_vertical_margin"
tools:context="com.example.administrator.mydb.MainActivity">
<LinearLayout android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical">
<Button
android:id="@+id/btn_jump"
android:text="跳转其他应用(不是启动页面的activity)"
android:layout_width="wrap_content"
android:layout_height="wrap_content"/>
<Button
android:id="@+id/btn_main"
android:text="跳转其他应用(启动页面的activity)"
android:layout_width="wrap_content"
android:layout_height="wrap_content"/>
<Button
android:id="@+id/btn_1"
android:text="跳转微信"
android:layout_width="wrap_content"
android:layout_height="wrap_content"/>
<TextView
android:id="@+id/tv"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="Hello World!"/>
</LinearLayout>
</ScrollView>
遇到问题:
java.lang.SecurityException: Permission Denial: starting Intent { act=android.intent.action.VIEW cmp=.SnsTimeLineUI } from …… not exported from uid 10092….
参考:
通过startActivity启动第三方应用的Activity时崩溃Permission Denial(比如打开微信朋友圈)
问题分析及解决:
只有当Activity向外声明了自己是可以处理某些intent action时,第三方app才能通过intent去启动它。通过配置文件中的intent过滤器来声明:
<activity android:name=".GetAllImg">
<intent-filter >
<action android:name="android.intent.action.VIEW"/>
</intent-filter>
</activity>
当知道了package name和class name,可以这样去启动第三方的activity:
ComponentName comp = new ComponentName("com.example.administrator.downloadimgdemo","com.example.administrator.downloadimgdemo.GetAllImg");
Intent intent = new Intent();
intent.setComponent(comp);
startActivity(intent);
这样做是没问题的。但假如没有在manifest中声明,startActivity将导致崩溃。