关闭

Android插件化开发之DexClassLoader动态加载dex、jar小Demo

标签: 动态加载DexClassLoaderDexjarnewInstance
4137人阅读 评论(4) 收藏 举报
分类:

一、温故动态加载ClassLoader机制

如果对Android的ClassLoader加载机制不熟悉,猛戳Android插件化开发动态加载基础之ClassLoader工作机制 http://blog.csdn.net/u011068702/article/details/53248960

二、介绍

         我们知道在Android中可以跟java一样实现动态加载jar,但是Android使用德海Dalvik VM,不能直接加载java打包jar的byte code,需要通过dx工具来优化Dalvik bytecode。
         Android在API中给出可动态加载的有:DexClassLoader 和 PathClassLoader(上面连接已经详细介绍)
         DexClassLoader:可加载jar、apk和dex,可以从SD卡中加载(这篇博客采用这种方式)

         PathClassLoader:只能加载已经安装搭配Android系统中的apk文件

三、曝Demo照片,不要怕,不多,很简单



四、编写接口文件

package com.example.testclassloader;

public interface ShowString {
	public String sayChenyu();
}


五、编写接口实现文件

package com.example.testclassloader;

import android.util.Log;

public class ShowStringClass implements ShowString{

	public static final String TAG = "ShowStringClass";
	
	@Override
	public String sayChenyu() {
		String chenyu = "chenyu";
		Log.i(TAG, chenyu);
		return chenyu;
	}

}

六、打包成jar文件编译成dex

我们把ShowStringClass.java文件打包生成showStringClass.jar文件,然后把文件放到sdk目录下的build-tools下的23.0.1目录下,我用的是ubuntu,所以会看到dex文件,如果是window会在这个目录下看到dex.bat文件,然后用下面命令把showStringClass.jar生成showStringClass_imle.jar的dex文件
dx --dex --output=showStringClass_impl.jar showStringClass.jar
然后再把showStringClass_impl.jar文件放到手机目录里面去用这个命令
adb push showStringClass_impl.jar  /sdcard/
具体操作图片如下


七、然后编写MainActivity.java文件

package com.example.testclassloader;

import java.io.File;

import android.content.Context;
import android.os.Bundle;
import android.os.Environment;
import android.support.v7.app.ActionBarActivity;
import android.util.Log;
import android.widget.TextView;
import dalvik.system.DexClassLoader;

public class MainActivity extends ActionBarActivity {

    public static final String TAG = "MainActivityClassLoader";
    public static final String SHOWSTRINGCLASS = "showStringClass_impl.jar";
    public static final String SHOWSTRINGCLASS_PATH= "com.example.testclassloader.ShowStringClass";
    public static final String DEX = "dex";
    public ShowStringClass mShowStringClass = null;
    public TextView mTv =  null;
    public int i = 0;
    
	@Override
	protected void onCreate(Bundle savedInstanceState) {
		super.onCreate(savedInstanceState);
		setContentView(R.layout.activity_main);
		mTv = (TextView)findViewById(R.id.hello);
		DexClassLoader(this);
	}
	
	/**
	 * 使用DexClassLoader方式加载类
	 */
	public void  DexClassLoader(Context context) {
		// dex压缩文件的路径(可以是apk,jar,zip格式)
        String dexPath = Environment.getExternalStorageDirectory().toString() + File.separator + SHOWSTRINGCLASS;

        // dex解压释放后的目录
        String dexOutputDirs = Environment.getExternalStorageDirectory().toString();

        //指定dexoutputpath为APP自己的缓存目录
        File dexOutputDir = context.getDir(DEX, 0);
       
        // 定义DexClassLoader
        // 第一个参数:是dex压缩文件的路径
        // 第二个参数:是dex解压缩后存放的目录
        // 第三个参数:是C/C++依赖的本地库文件目录,可以为null
        // 第四个参数:是上一级的类加载器
        //DexClassLoader dexClassLoader = new DexClassLoader(dexPath,dexOutputDirs,null,getClassLoader());
        DexClassLoader dexClassLoader = new DexClassLoader(dexPath,dexOutputDir.getAbsolutePath(),null,getClassLoader());

        Class libProvierClazz = null;
        // 使用DexClassLoader加载类
        try {
            libProvierClazz = dexClassLoader.loadClass(SHOWSTRINGCLASS_PATH);
            // 创建dynamic实例
            mShowStringClass = (ShowStringClass) libProvierClazz.newInstance();
            if (mShowStringClass != null) {
            	final String chenyu = mShowStringClass.sayChenyu();
            	if (chenyu != null) {
            		mTv.post(new Runnable() {
				@Override
				public void run() {
					mTv.setText(chenyu);
				}
            		});
            	}
            } else {
            	Log.d(TAG, "mShowStringClass is null");
            }
        } catch (Exception e) {
            e.printStackTrace();
        }
	}
	
	/**
	 * 打印系统的classLoader
	 */
	public void showClassLoader() {
		ClassLoader classLoader = getClassLoader();
        if (classLoader != null){
            Log.i(TAG, "[onCreate] classLoader " + i + " : " + classLoader.toString());
            while (classLoader.getParent()!=null){
                classLoader = classLoader.getParent();
                Log.i(TAG,"[onCreate] classLoader " + i + " : " + classLoader.toString());
                i++;
            }
        }
	}
}

八、运行Demo的结果爆照

ubuntu终端打印结果如下

手机上面照片如下

说明加载外部的文件加载成功了

如果把上面那行代码改成这个
DexClassLoader dexClassLoader = new DexClassLoader(dexPath,dexOutputDirs,null,getClassLoader());
会报下面的错误

需要加上缓存Dex文件的目录
      //指定dexoutputpath为APP自己的缓存目录
        File dexOutputDir = context.getDir(DEX, 0);

九、总结

1、加深动态加载的理解
2、如何实现项目加载外部的Dex文件有了更好的理解
3、对DexClassLoader 、dexClassLoader.load(package.class)、 class.newInstance() 有了更好的理解

2
0
查看评论

DexClassLoader和PathClassLoader加载Dex流程

0x00    在上一篇文章apk安装和优化原理,在最后我们分析了DexClassLoader和PathClassLoader的构造函数的不同。    PathClassLoader最后调用的是new DexFile(pathFile),而DexCl...
  • jltxgcy
  • jltxgcy
  • 2016-01-21 12:52
  • 5234

Android中的类装载器DexClassLoader

@SuppressLint("NewApi") private void useDexClassLoader(){ //创建一个意图,用来找到指定的apk Intent intent = new Intent("com.suchangli.andro...
  • com360
  • com360
  • 2013-11-04 15:37
  • 44623

Android插件化探索(一)类加载器DexClassLoader

在Java环境中,有个概念叫做“类加载器”(ClassLoader),其作用是动态装载Class文件。标准的Java SDK中有一个ClassLoader类,借助它可以装载想要的Class文件,每个ClassLoader对象在初始化时必须指定Class文件的路径。
  • maplejaw_
  • maplejaw_
  • 2016-05-24 21:43
  • 6592

DexClassLoader和PathClassLoader的区别

在使用Java虚拟机时,我们经常自定义继承自ClassLoader的类加载器。然后通过defineClass方法来从一个二进制流中加载Class。而在Android中我们无法这么使用,Android中ClassLoader的defineClass方法具体是调用VMClassLoader的define...
  • mynameishuangshuai
  • mynameishuangshuai
  • 2016-10-05 10:53
  • 6536

DexClassLoader动态加载分析

转载自:http://www.blogfshare.com/dexclassloader.html 看到原来有把原始的dex文件加密保存,然后解密后使用DexClassLoader加载文件的方法,就来分析下DexClassLoader的加载流程: 源码地址:http://androidx...
  • QQ1084283172
  • QQ1084283172
  • 2017-04-21 11:46
  • 586

用DexClassLoader实现加壳

dexclassloader 解决组件类生命周期问题
  • beyond296089727
  • beyond296089727
  • 2015-05-01 13:00
  • 4745

插件化知识详细分解及原理 之ClassLoader及dex加载过程

接着上篇我们这一篇说ClassLoader及dex加载过程。为了解决65535这个问题,Google提出了multidex方案,即一个apk文件可以包含多个dex文件。 不过值得注意的是,除了第一个dex文件以外,其他的dex文件都是以资源的形式被加载的, 换句话说就是在Application初...
  • yulong0809
  • yulong0809
  • 2017-02-27 11:38
  • 2629

Android之ClassLoader类加载器(MultiDex、动态加载dex)

android classloader multidex
  • cheng545
  • cheng545
  • 2017-07-15 12:52
  • 487

DexClassLoader自定义加载Assets目录下的dex、jar文件

  • 2015-02-20 21:33
  • 2KB
  • 下载

DexClassLoader和PathClassLoader类加载机制

0x00    在DexClassLoader和PathClassLoader加载Dex流程一文中,我们分析了dex文件如何形成了DexFile结构体。本文中讲解类加载机制,实际上就是生成ClassObject对象。    我们以DexClassLoader为...
  • jltxgcy
  • jltxgcy
  • 2016-01-21 20:29
  • 2557
    个人资料
    • 访问:1144972次
    • 积分:18605
    • 等级:
    • 排名:第594名
    • 原创:739篇
    • 转载:316篇
    • 译文:11篇
    • 评论:450条
    联系方式

    我的github



    就职:深信服科技

    QQ:2657607916

    QQ讨论群:346618607

    微信:chenyu_920310

    邮箱:2657607916@qq.com

    最新评论