首先引入一個概念,動態載入技術是什麼?爲何要引入動態載入?它有什麼優勢呢?首先要明確這幾個問題。咱們先從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");