Android 之动态加载代码

1,803 阅读4分钟

 

一、前言

在项目研发中会遇到部分功能经常变更,经常升级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();
        }

    }
}

整个动态加载类的流程就是这样的