android得到已安装和未安装apk的信息

         在获取apk资源时候最重要的一个类就是PackageManager,我们可以通过这个类得到各种想要的东西,首先是得到已经安装的apk的基本信息,包括label,和图标等资源:

PackageManager pm = getPackageManager();
List<PackageInfo> apkInfos = pm.getInstalledPackages(0);
ArrayList<String> infos_name =  new ArrayList<String>() ;//用来存放label
ArrayList<Drawable> infos_icon = newArrayList<Drawable>();//用来存储apk启动图标资源
String name = "";
   Drawable icon;
   PackageInfo apk;
   for (int i = 0; i < apkInfos.size(); )
   {
      apk = apkInfos.get(i);
      i++;
      name = (String) pm.getApplicationLabel(apk.applicationInfo);
      icon = pm.getApplicationIcon(apk.applicationInfo);
      infos_icon.add(icon);
      infos_name.add(name);
   }
        接下来我们利用PackageManager获得已安装apk中的activity,和service等

List<PackageInfo>packagesInfo =  pm.getInstalledPackages(0);
for (PackageInfo packageInfo : packagesInfo) {
	Log.d("TAG","packageInfo NAME IS :"+packageInfo.packageName);
}


PackageManager pm = getPackageManager();
try {   
//获得com.example.pertest包中的activity,service和broadcastreceiver
PackageInfo packageInfo = pm.getPackageInfo("com.example.pertest",
				PackageManager.GET_PERMISSIONS
						| PackageManager.GET_SERVICES
						| PackageManager.GET_RECEIVERS);
	ActivityInfo[] activities = packageInfo.activities;

	if (activities != null) {
		for (ActivityInfo activityInfo : activities) {
			Log.d("TAG", "the activity " + activityInfo.toString()
					+ "==" + packageInfo.versionCode + "==="
					+ packageInfo.versionName);
		}
	} else {
		Log.d("TAG", "the activity is null");
	}
	ServiceInfo[] serviceInfos = packageInfo.services;
	for (ServiceInfo serviceInfo : serviceInfos) {
		Log.d("TAG", "service name is :" + serviceInfo.name);
	}

	activities = packageInfo.receivers;
	for (ActivityInfo activity : activities) {
		Log.d("TAG", "receiver is :" + activity.name);
	}
} catch (NameNotFoundException e) {
	// TODO Auto-generated catch block
	e.printStackTrace();
}

上面就是获取已经安装的apk的基本信息,接下来看看如何获取未安装的apk文件的基本信息,这对于android动态加载时很有用的,我们来看:

得到未安装apk的图标:

public static Drawable getAppIcon(Context context, String apkFilepath) {
        PackageManager pm = context.getPackageManager();
        PackageInfo pkgInfo = getPackageInfo(context, apkFilepath);
        if (pkgInfo == null) {
            return null;
        }

        ApplicationInfo appInfo = pkgInfo.applicationInfo;
        if (Build.VERSION.SDK_INT >= 8) {
            appInfo.sourceDir = apkFilepath;
            appInfo.publicSourceDir = apkFilepath;
        }
        return pm.getApplicationIcon(appInfo);
    }
得到未安装apk的名称:

public static CharSequence getAppLabel(Context context, String apkFilepath) {
        PackageManager pm = context.getPackageManager();
        PackageInfo pkgInfo = getPackageInfo(context, apkFilepath);
        if (pkgInfo == null) {
            return null;
        }
        ApplicationInfo appInfo = pkgInfo.applicationInfo;
        if (Build.VERSION.SDK_INT >= 8) {
            appInfo.sourceDir = apkFilepath;
            appInfo.publicSourceDir = apkFilepath;
        }

        return pm.getApplicationLabel(appInfo);
 }
如果还觉得不够的话,我们可以得到:PackageInfo对象,通过该对象,我们就可以获得该未安装apk的activity,service,broadcastreceiver等,方法同"获得已安装apk中的activity,和service"   ,代码如下:

//得到PackageInfo对象,其中包含了该apk包含的activity和service
public static PackageInfo getPackageInfo(Context context, String apkFilepath) {
        PackageManager pm = context.getPackageManager();
        PackageInfo pkgInfo = null;
        try {
            pkgInfo = pm.getPackageArchiveInfo(apkFilepath, PackageManager.GET_ACTIVITIES | PackageManager.GET_SERVICES);
        } catch (Exception e) {
            // should be something wrong with parse
            e.printStackTrace();
        }
        return pkgInfo;
    }

        在android动态加载的时候,是通过DexClassLoader来实现的,通过这种方式来获取未安装程序的DexClassLoader对象:

String mNativeLibDir = mContext.getDir("pluginlib", Context.MODE_PRIVATE).getAbsolutePath();
 private DexClassLoader createDexClassLoader(String dexPath) {
        File dexOutputDir = mContext.getDir("dex", Context.MODE_PRIVATE);
        dexOutputPath = dexOutputDir.getAbsolutePath();
        DexClassLoader loader = new DexClassLoader(dexPath, dexOutputPath, mNativeLibDir, mContext.getClassLoader());
        return loader;
 }
   第一个参数dexPath就是我们apk的存放路径。

   第二个参数dexOutPath就是该apk文件对应的dex文件的存放路径,不可以为null。

   第三个参数是目标类中使用的C/C++库的列表,每个目录用File.pathSeparator间隔开; 可以为 null。

  第四个参数是该类装载器的父装载器,一般用当前执行类的装载器。

在获得了该DexClassLoader对象以后,我们就可以实现动态加载该apk中的方法了,详细的方法,请看我的另一篇文章:android动态加载

对于加载未安装的apk中的资源,由于使用的是不同的context,所以,这里先这样做:

 private AssetManager createAssetManager(String dexPath) {
        try {
            AssetManager assetManager = AssetManager.class.newInstance();
            Method addAssetPath = assetManager.getClass().getMethod("addAssetPath", String.class);
            addAssetPath.invoke(assetManager, dexPath);
            return assetManager;
        } catch (Exception e) {
            e.printStackTrace();
            return null;
        }
    }


private Resources createResources(AssetManager assetManager) {
        Resources superRes = mContext.getResources();
        Resources resources = new Resources(assetManager, superRes.getDisplayMetrics(), superRes.getConfiguration());
        return resources;
}
        此时的assetManager和resources就可以和平时一样来用了。







郑重声明:本站内容如果来自互联网及其他传播媒体,其版权均属原媒体及文章作者所有。转载目的在于传递更多信息及用于网络分享,并不代表本站赞同其观点和对其真实性负责,也不构成任何其他建议。