一、前言
在项目研发中会遇到部分功能经常变更,经常升级app会对用户产生反感,造成体验很差。
项目中有这样一个功能:下载到本地的视频需要在播放时加载字幕,但是某些视频我们的服务器中不存在字幕,经过调研发现字幕库网站能通过影片名查询到相应的字幕,并下载下来。但是问题是字幕库没有公开的字幕查询接口,只能通过一些逆向分析后,对页面进行解析捕获到了字幕的下载路径,字幕下载路径是嵌套在html代码中,这样只有通过JSoup技术(不懂得可以查百度,这儿就不细说)对html页面进行动态解析,拿到字幕的下载地址,之后再下载到我们的服务器上面,开发的都知道使用第三方的总是不稳定,都说吃人嘴软,拿人手短,还是不如自己的,第三方的网页布局变化了,那么使用jsoup解析的地址就全部出错了,但是不能因为这个就去对app进行重新打包发布新版本,这样对用户的体验不好,那么我们就要使用动态加载技术去改变这样的频繁打包工作。
解决思路:把经常变化的放在服务器上面,每次启动app的时候就从服务器上面下载下来逻辑,再动态的加载到app的包里面,动态打包我们的app的,实例化对象,如果字幕库发生变化,我们就只需要更新服务器上面的解析代码,重新下载相应的逻辑加载到app。
二、使用方式
如何实现动态加载的流程?
第一:制作dex文件。
第二:把制作的dex文件发布到服务器上面,从服务器上面下载dex文件之后动态打包到app中
制作的工具类:
public class JsoupUtils {
public static String html2Url(String html) {
return "url" + html;
}
}复制代码
①、制作dex文件
然后编译之后再androidstudio的build/intermediates/classes/debug/ 下面会看到你的包名生成的字节码,之后使用Java打包命令:jar -cvf把指定的字节码打包成jar文件,如下:
出现这个表示打包成功,
然后再把jar文件打包成dex文件,现在就要使用dex命令,dx.bat文件,在build目录下,或者配置环境变量:
出现以下表示打包成功:
至此dex文件打包成功
②、把制作的dex文件发布到服务器上面,从服务器上面下载dex文件之后动态打包到app中
在此演示则不去服务器下载,省略下载的步骤,直接放在assets目录下面:
先把assets目录下的utils_dex.jar拷贝到sd卡上面
三、加载器
以下就是类加载器:
使用反射与类加载器
android中的类加载器主要有三个:
(1)、URLClassLoader
只能用于加载jar文件,但是由于dalvik不能直接识别jar文件,所以android中无法使用这个类加载器
(2)、PathClassLoader
它只能加载已经安装的apk,因为PathClassLoader只会读取/data/dalvik-cache/目录下的dex文件,
例如安装一个apk的时候,就会在这个目录下面的x86目录下生成每个apk对应的dex文件:
使用PathClassLoader加载apk时,它就会在这个目录下面去查找对应的DEX文件,如果apk没有安装,则会报错,ClassNotFoundException
(3)、DexClassLoader
是最理想的加载器,它的构造函数包括四个参数
1、dexPath:目标类所在的APK或jar文件的路径,类加载器将从该路径中寻找指定的目标类,该类必须是apk或者jar的全路径,如果包含多个路径,路径之间必须使用特定的分隔符分隔,特定的分隔符可以使用System.getProperty("path.separtor")获得;
2、dexOutputDir:由于dex文件被包含在apk或者jar文件中,因此在装载目标类之前需要先从apk或jar文件中解压出dex文件,该参数就是定制解压出的dex文件存放的路径,在Android系统中,一个应用程序一般对应一个Linux用户,应用程序仅对属于自己的数据目录路径有写的权限,因此,该参数可以使用该程序的数据路径
3、libPath:指目标类中使用的C/C++库存放的路径
4、classload是指该装载器的父装载器,一般为当前执行类的装载器
直接上代码:
package com.parse.dex;
import android.content.Context;
import android.os.Bundle;
import android.os.Environment;
import android.support.v7.app.AppCompatActivity;
import android.widget.Toast;
import java.io.File;
import java.lang.reflect.Method;
import dalvik.system.DexClassLoader;
public class MainActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
initLoadDex();
}
private void initLoadDex() {
FileUtils.copyAssetsJarToFile(this, "utils_dex.jar", "utils_dex.jar");
File file = new File(Environment.getExternalStorageDirectory().toString() + File.separator + "utils_dex.jar");
File optimizedDexOutputPath = getDir("dex", Context.MODE_PRIVATE);
DexClassLoader dexClassLoader = new DexClassLoader(file.getAbsolutePath(), optimizedDexOutputPath.getAbsolutePath(), null, getClassLoader());
try {
Class loadClass = dexClassLoader.loadClass("com.parse.dex.JsoupUtils");
Method html2Url = loadClass.getMethod("html2Url", String.class);
String s = (String) html2Url.invoke(loadClass, "解析html文件");
Toast.makeText(this, s, Toast.LENGTH_SHORT).show();
} catch (Exception e) {
e.printStackTrace();
}
}
}复制代码
整个动态加载类的流程就是这样的