首先引入一个概念,动态载入技术是什么?为何要引入动态载入?它有什么优势呢?首先要明确这几个问题。咱们先从java
应用程序入手,你们都知道在Android App中。一个应用程序dex文件的方法数最大不能超过65536个。不然,你的applinux
将出异常了,那么假设越大的项目那确定超过了,像美团、支付宝等都是使用动态载入技术。支付宝在去年的一个技微信
术分享类会议上就推崇让应用程序插件化,而美团也发布了他们的解决方式:Dex本身主动拆包和动态载入技术。app
因此使ide
用动态载入技术解决此类问题。学习
而它的长处可以让应用程序实现插件化、插拔式结构,对后期维护做用那不用说了。ui
动态载入技术就是使用类载入器载入对应的apk、dex、jar(必须含有dex文件)。再经过反射得到该apk、dex、jar内部的资源(class、图片、color等等)进而供宿主app使用。this
因此咱们在宿主app和插件app的manifest上都定义一个一样的sharedUserId。spa
/** * 查找手机内所有的插件 * @return 返回一个插件List */ private List<PluginBean> findAllPlugin() { List<PluginBean> plugins = new ArrayList<>(); PackageManager pm = getPackageManager(); //经过包管理器查找所有已安装的apk文件 List<PackageInfo> packageInfos = pm.getInstalledPackages(PackageManager.GET_UNINSTALLED_PACKAGES); for (PackageInfo info : packageInfos) { //获得当前apk的包名 String pkgName = info.packageName; //获得当前apk的sharedUserId String shareUesrId = info.sharedUserId; //推断这个apk是不是咱们应用程序的插件 if (shareUesrId != null && shareUesrId.equals("com.sunzxyong.myapp") && !pkgName.equals(this.getPackageName())) { String label = pm.getApplicationLabel(info.applicationInfo).toString();//获得插件apk的名称 PluginBean bean = new PluginBean(label,pkgName); plugins.add(bean); } } return plugins; }经过这段代码,咱们就可以轻松的获取手机内存在的所有插件。当中PluginBean是定义的一个实体类而已,就不贴它的代码了。
List<HashMap<String, String>> datas = new ArrayList<>(); List<PluginBean> plugins = findAllPlugin(); if (plugins != null && !plugins.isEmpty()) { for (PluginBean bean : plugins) { HashMap<String, String> map = new HashMap<>(); map.put("label", bean.getLabel()); datas.add(map); } } else { Toast.makeText(this, "没有找到插件,请先下载。", Toast.LENGTH_SHORT).show(); } showEnableAllPluginPopup(datas);四、假设找到后,那么咱们选择相应的插件时,在宿主app中就载入插件内相应的资源,这个才是PathClassLoader的重点。咱们首先看看怎么实现的吧:
/** * 载入已安装的apk * @param packageName 应用的包名 * @param pluginContext 插件app的上下文 * @return 相应资源的id */ private int dynamicLoadApk(String packageName, Context pluginContext) throws Exception { //第一个參数为包括dex的apk或者jar的路径,第二个參数为父载入器 PathClassLoader pathClassLoader = new PathClassLoader(pluginContext.getPackageResourcePath(),ClassLoader.getSystemClassLoader()); // Class<?> clazz = pathClassLoader.loadClass(packageName + ".R$mipmap");//经过使用自身的载入器反射出mipmap类进而使用该类的功能 //參数:一、类的全名,二、是否初始化类,三、载入时使用的类载入器 Class<?> clazz = Class.forName(packageName + ".R$mipmap", true, pathClassLoader); //使用上述两种方式都可以,这里咱们获得R类中的内部类mipmap,经过它获得相应的图片id,进而给咱们使用 Field field = clazz.getDeclaredField("one"); int resourceId = field.getInt(R.mipmap.class); return resourceId; }
public PathClassLoader(String dexPath, ClassLoader parent)中当中第一个參数是经过插件的上下文来获取插件apk的路径,事实上获取到的就是/data/app/apkthemeplugin.apk。那么插件的上下文怎么获取呢?在宿主app中咱们仅仅有本app的上下文啊,答案就是为插件app建立一个上下文:
//获取相应插件中的上下文,经过它可获得插件的Resource Context plugnContext = this.createPackageContext(packageName, CONTEXT_IGNORE_SECURITY | CONTEXT_INCLUDE_CODE);经过插件的包名来建立上下文,只是这样的方法仅仅适合获取已安装的app上下文。或者不需要经过反射直接经过插件上下文getResource().getxxx(R.*.*);也行,而这里用的是反射方法。
plugnContext.getResources().getDrawable(resouceId)就可以获取相应id的Drawable获得该图片资源进而宿主app的可用它设置背景等。
。.net
public PackageInfo getPackageArchiveInfo(String archiveFilePath, int flags)它的參数恰好是传入一个FilePath。而后返回apk文件的PackageInfo信息:
/** * 获取未安装apk的信息 * @param context * @param archiveFilePath apk文件的path * @return */ private String[] getUninstallApkInfo(Context context, String archiveFilePath) { String[] info = new String[2]; PackageManager pm = context.getPackageManager(); PackageInfo pkgInfo = pm.getPackageArchiveInfo(archiveFilePath, PackageManager.GET_ACTIVITIES); if (pkgInfo != null) { ApplicationInfo appInfo = pkgInfo.applicationInfo; String versionName = pkgInfo.versionName;//版本 Drawable icon = pm.getApplicationIcon(appInfo);//图标 String appName = pm.getApplicationLabel(appInfo).toString();//app名称 String pkgName = appInfo.packageName;//包名 info[0] = appName; info[1] = pkgName; } return info; }
/** * @param apkName * @return 获得相应插件的Resource对象 */ private Resources getPluginResources(String apkName) { try { AssetManager assetManager = AssetManager.class.newInstance(); Method addAssetPath = assetManager.getClass().getMethod("addAssetPath", String.class);//反射调用方法addAssetPath(String path) //第二个參数是apk的路径:Environment.getExternalStorageDirectory().getPath()+File.separator+"plugin"+File.separator+"apkplugin.apk" addAssetPath.invoke(assetManager, apkDir+File.separator+apkName);//将未安装的Apk文件的加入进AssetManager中,第二个參数为apk文件的路径带apk名 Resources superRes = this.getResources(); Resources mResources = new Resources(assetManager, superRes.getDisplayMetrics(), superRes.getConfiguration()); return mResources; } catch (Exception e) { e.printStackTrace(); } return null; }
/** * 载入apk得到内部资源 * @param apkDir apk文件夹 * @param apkName apk名字,带.apk * @throws Exception */ private void dynamicLoadApk(String apkDir, String apkName, String apkPackageName) throws Exception { File optimizedDirectoryFile = getDir("dex", Context.MODE_PRIVATE);//在应用安装文件夹下建立一个名为app_dex文件夹文件夹,假设已经存在则不建立 Log.v("zxy", optimizedDirectoryFile.getPath().toString());// /data/data/com.example.dynamicloadapk/app_dex //參数:一、包括dex的apk文件或jar文件的路径,二、apk、jar解压缩生成dex存储的文件夹。三、本地library库文件夹,通常为null,四、父ClassLoader DexClassLoader dexClassLoader = new DexClassLoader(apkDir+File.separator+apkName, optimizedDirectoryFile.getPath(), null, ClassLoader.getSystemClassLoader()); Class<?当中经过new DexClassLoader()来建立未安装apk的类载入器,咱们来看看它的參数:> clazz = dexClassLoader.loadClass(apkPackageName + ".R$mipmap");//经过使用apk本身的类载入器,反射出R类中相应的内部类进而获取咱们需要的资源id Field field = clazz.getDeclaredField("one");//获得名为one的这张图片字段 int resId = field.getInt(R.id.class);//获得图片id Resources mResources = getPluginResources(apkName);//获得插件apk中的Resource if (mResources != null) { //经过插件apk中的Resource获得resId相应的资源 findViewById(R.id.background).setBackgroundDrawable(mResources.getDrawable(resId)); } }
public DexClassLoader(String dexPath, String optimizedDirectory, String libraryPath, ClassLoader parent)
A class loader that loads classes from .jar and .apk files containing a classes.dex entry. This can be used to execute code not installed as part of an application. This class loader requires an application-private, writable directory to cache optimized classes. Use Context.getDir(String, int) to create such a directory: File dexOutputDir = context.getDir("dex", 0); Do not cache optimized classes on external storage. External storage does not provide access controls necessary to protect your application from code injection atta,因此咱们用getDir()方法在应用内部建立一个dexOutputDir。
copyApkFile("apkthemeplugin-1.apk");
copyApkFile("apkthemeplugin-1.apk"); copyApkFile("apkthemeplugin-2.apk"); copyApkFile("apkthemeplugin-3.apk");