美文网首页
Android studio 生成Jar,和动态加载Dex

Android studio 生成Jar,和动态加载Dex

作者: 脏仙人 | 来源:发表于2017-03-03 17:42 被阅读5051次

    Dalvik 虚拟机和标准的java虚拟机加载机制的区别:

    Dalvik虚拟机如同其他Java虚拟机一样,在运行程序时首先需要将对应的类加载到内存中。而在Java标准的虚拟机中,类加载可以从class文件中读取,也可以是其他形式的二进制流,因此,我们常常利用这一点,在程序运行时手动加载Class,从而达到代码动态加载执行的目的,但是Dalvik虚拟机毕竟不算是标准的Java虚拟机。如下图。

    DexClassLoader:可以加载jar/apk/dex,可以从SDK中加载未加载的apk、

    PathClassLOader:要传入系统中Apk的存放Path,所以只能加载已经安装的apk文件。

    与jvm不同的是Dalvik不能直接加载.dex文件,而是要通过从ClassLoader派生出的两个类 DexClassLoader和PathClassLoader来加载.dex文件。

    接下来我们实现两个步骤的准备工作。

    一、打包出一个需要用到的jar文件

    二、将jar文件转换成dex文件。

    Android studio打包jar文件

    新建一个项目创建完成后项目目录如下:

    在目录com.loaderdome包下面创建一个dynamic包用来放一个接口 Dynamic 在接口里面写一个sayHello()方法,返回Srtring

    新建一个impl包,并实现Dynamic接口(这个文件我们会打包成Dex文件来通过接口调用)

    点击Build -->Rebuild project,点击完成后需要打包成jar 的class文件会在下面所示的目录中显示,箭头指出的就是我们需要打包的class文件

    配置app moudle的 build.gradle 文件添加以下配置代码(这里注意添加在 app moudle 下面的 gradle里面而project下面的gradle)

    //删除dynamic.jar包任务

    task clearJar(type: Delete) {

    delete('libs/dynamic.jar')

    }

    //打包任务

    task makeJar(type: org.gradle.api.tasks.bundling.Jar) {

    //指定生成的jar名称

    baseName 'dynamic'

    //从哪里打包class文件

    from('build/intermediates/classes/debug/com/loaderdexdome/dynamic/')

    //打包到jar后的目录结构

    into('com/loaderdexdome/dynamic/')

    //去掉不需要打包的目录和文件

    exclude('text/', 'Dynamic.class', 'R.class', 'BuildConfig.class')

    //去掉R$开头的文件

    exclude { it.name.startsWith('R$'); }

    }

    makeJar.dependsOn(clearJar, build)

    在terminal 里面输入命令 gradle makeJar  出现下面的提示后我们的Jar就生成完成了。(这里如果弹出不是内部命令的提示,需要去配置以下gradle的环境)

    生成后的jar文件会放在app-->libs目录下面,箭头所示就是我们生成的jar文件

    到这里第一步就完成了。接着我们进行第二步将将生成的jar文件转换成可以被Dalvik虚拟机识别的dex文件。

    进入sdk目录的build-tools选一个版本进去将我们生成的jar拷贝到此目录下。这里我选择的是25.0.2版本。下图箭头指出的就是我考进去的jar

    我们在terminal进入到此目录下面 执行dx --dex --output=dynamic_dex.jar dynamic.jar 命令如下图:

    执行成功后dynamic_dex.jar文件会出现在我们刚才cd进去的目录下面                                         这里需要注意如果不把dynamic.jar拷贝到此目录下会报错找不到dynamic.jar文件

    接下来我们将此目录下的dynamic.jar和dynamic_dex.jar拿出来到这里dynamic.jar就没有用了。dynamic_dex.jar就是我们要动态加载的dex。                                                                            为了避免出错我们将项目里面 lids下面的dynamic.jar和impl包和包里的DynamicImpl类都删除掉,删除后的目录如下

    现在我们就可以开始来干正事了

    首先在app目录下创建assets目录将我们的dynamic_dex.jar文件放到此目录下面。

    创建一个FileUtils工具类用来将assets目录下的dynamic_dex.jar copy到app/data/cache目录下 源码如下

    public class FileUtils {

    public static void copyFiles(Context context, String fileName, File desFile) {

    InputStream in = null;

    OutputStream out = null;

    try {

    in = context.getApplicationContext().getAssets().open(fileName);

    out = new FileOutputStream(desFile.getAbsolutePath());

    byte[] bytes = new byte[1024];

    int i;

    while ((i = in.read(bytes)) != -1)

    out.write(bytes, 0, i);

    } catch (IOException e) {

    e.printStackTrace();

    } finally {

    try {

    if (in != null)

    in.close();

    if (out != null)

    out.close();

    } catch (IOException e) {

    e.printStackTrace();

    }

    }                                                                                                                                           }

    public static boolean hasExternalStorage() {

    return Environment.getExternalStorageState().equals(Environment.MEDIA_MOUNTED);

    }

    /**

    * 获取缓存路径

    *

    * @param context

    * @return 返回缓存文件路径

    */

    public static File getCacheDir(Context context) {

    File cache;

    if (hasExternalStorage()) {

    cache = context.getExternalCacheDir();

    } else {

    cache = context.getCacheDir();

    }                                                                                                                                          

    if (!cache.exists())

    cache.mkdirs();

    return cache;

    }

    }

    修改activity代码。

    public class MainActivity extends AppCompatActivity {

    private Dynamic dynamic;

    @Override

    protected void onCreate(Bundle savedInstanceState) {

    super.onCreate(savedInstanceState);

    setContentView(R.layout.activity_main);

    findViewById(R.id.tv_click).setOnClickListener(new View.OnClickListener() {

    @Override

    public void onClick(View view) {

    loadDexClass();

        }                                                                                                                                          });                                                                                                                                        }

    /**

    * 加载dex文件中的class,并调用其中的sayHello方法

    */

    private void loadDexClass() {

    File cacheFile = FileUtils.getCacheDir(getApplicationContext());

    String internalPath = cacheFile.getAbsolutePath() + File.separator + "dynamic_dex.jar";

    File desFile = new File(internalPath);

    try {

    if (!desFile.exists()) {

    desFile.createNewFile();

    FileUtils.copyFiles(this, "dynamic_dex.jar", desFile);

    }

    } catch (IOException e) {

    e.printStackTrace();

    }

    //下面开始加载dex class

    DexClassLoader dexClassLoader = new DexClassLoader(internalPath, cacheFile.getAbsolutePath(), null, getClassLoader());

    try {

    //加载的类名为jar文件里面完整类名,写错会找不到此类hh

    Class libClazz = dexClassLoader.loadClass("com.loaderdexdome.dynamic.impl.DynamicImpl");

    dynamic = (Dynamic) libClazz.newInstance();

    if (dynamic != null)

    Toast.makeText(this, dynamic.sayHello(), Toast.LENGTH_LONG).show();

    } catch (Exception e) {

    e.printStackTrace();

    }

    }

    }

    运行后效果如下:

    运行后将生成好的App拿出来解压也可以看到里面是有两个dex文件的。

    相关文章

      网友评论

          本文标题:Android studio 生成Jar,和动态加载Dex

          本文链接:https://www.haomeiwen.com/subject/fkdrgttx.html