Android 之动态加载代码
博客专区 > MrXI 的博客 > 博客详情
Android 之动态加载代码
MrXI 发表于1年前
Android 之动态加载代码
  • 发表于 1年前
  • 阅读 389
  • 收藏 13
  • 点赞 2
  • 评论 0

腾讯云 学生专属云服务套餐 10元起购>>>   

一、前言

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

    }
}

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

 

四、开源代码库

最后再分享一个自己积攒很久的代码库,只有你想不到,没有用不到的,欢迎star

https://github.com/xijiufu

由于github服务器在美国,有时访问很慢,还提供了开源中国地址库,2个仓库代码均同步更新:

http://git.oschina.net/xijiufu

 

共有 人打赏支持
粉丝 26
博文 13
码字总数 33406
×
MrXI
如果觉得我的文章对您有用,请随意打赏。您的支持将鼓励我继续创作!
* 金额(元)
¥1 ¥5 ¥10 ¥20 其他金额
打赏人
留言
* 支付类型
微信扫码支付
打赏金额:
已支付成功
打赏金额: