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

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

一、温故动态加载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照片,不要怕,不多,很简单

1.png

四、编写接口文件

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/

具体操作图片如下

1.png

七、然后编写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++;
            }
        }
  }
}
  @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++;
            }
        }
  }
}

@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终端打印结果如下

1.png

手机上面照片如下

1.png

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

 

如果把上面那行代码改成这个

DexClassLoader dexClassLoader = new DexClassLoader(dexPath,dexOutputDirs,null,getClassLoader());

会报下面的错误

1.png

需要加上缓存Dex文件的目录

      //指定dexoutputpath为APP自己的缓存目录
        File dexOutputDir = context.getDir(DEX, 0);

九、总结

1、加深动态加载的理解

2、如何实现项目加载外部的Dex文件有了更好的理解

3、对DexClassLoader 、dexClassLoader.load(package.class)、 class.newInstance() 有了更好的理解

相关文章
|
5月前
|
Java Android开发
|
2月前
|
XML API Android开发
码农之重学安卓:利用androidx.preference 快速创建一、二级设置菜单(demo)
本文介绍了如何使用androidx.preference库快速创建具有一级和二级菜单的Android设置界面的步骤和示例代码。
56 1
码农之重学安卓:利用androidx.preference 快速创建一、二级设置菜单(demo)
|
14天前
|
Java Maven 开发工具
第一个安卓项目 | 中国象棋demo学习
本文是作者关于其第一个安卓项目——中国象棋demo的学习记录,展示了demo的运行结果、爬坑记录以及参考资料,包括解决Android Studio和maven相关问题的方法。
第一个安卓项目 | 中国象棋demo学习
|
5月前
|
Android开发
Android 插件化
Android 插件化
52 0
|
5月前
|
Java Android开发
Android编译的jar里面是dex
Android编译的jar里面是dex
58 0
|
5月前
|
存储 Android开发
Android插件化-Broadcast篇,2024年最新安卓面试自我介绍
Android插件化-Broadcast篇,2024年最新安卓面试自我介绍
|
5月前
|
Android开发 Kotlin API
Android插件化探索与发现,kotlin协程切换线程
Android插件化探索与发现,kotlin协程切换线程
|
5月前
|
缓存 Android开发
Android插件化——高手必备的Hook技术,零基础开发android
Android插件化——高手必备的Hook技术,零基础开发android
|
3月前
|
Java
[JarEditor]可直接修改jar包的IDEA插件
### 修改JAR包变得更简单:JarEditor插件简介 **背景:** 开发中常需修改JAR包中的class文件,传统方法耗时费力。JarEditor插件让你一键编辑JAR包内文件,无需解压。 **插件使用:** 1. **安装:** 在IDEA插件市场搜索JarEditor并安装。 2. **修改class:** 打开JAR文件中的class,直接编辑,保存后一键构建更新JAR。 3. **文件管理:** 右键菜单支持在JAR内新增/删除/重命名文件等操作。 4. **搜索:** 使用内置搜索功能快速定位JAR包内的字符串。
313 2
[JarEditor]可直接修改jar包的IDEA插件
|
3月前
|
弹性计算 Java Serverless
Serverless 应用引擎操作报错合集之上传自定义JAR包,启动时报错,是什么导致的
Serverless 应用引擎(SAE)是阿里云提供的Serverless PaaS平台,支持Spring Cloud、Dubbo、HSF等主流微服务框架,简化应用的部署、运维和弹性伸缩。在使用SAE过程中,可能会遇到各种操作报错。以下是一些常见的报错情况及其可能的原因和解决方法。