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

2017-03-03  本文已影响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文件的。

上一篇下一篇

猜你喜欢

热点阅读