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文件的。