LJ的Blog

学海无涯苦做舟

0%

Android动态加载dex初试

最近想研究一波插件化,在简书上看到这个感觉还不错,就按照这个顺序看下去了。不过得小记一下,省的以后我还得去上面提到的文里去找链接。我这里记得就是自己操作的一遍流程。。其实我也是根据他的文来操作的,属于HelloWorld级别吧。

操作

本来还想介绍一下概念,想了想,算了,不误人子弟了。感兴趣自己搜。

1.新建工程

没什么好说的,打开Android Studio新建一个工程都会吧。我这里新建了一个名为fordex的项目,包名为com.xiasuhuei321.firstpro。

2.新建一个接口和一个类

接口和类非常简单,类是接口的实现类,主要功能就是弹一个Toast。接口:

1
2
3
public interface IShowToast {
public int showToast(Context context);
}

实现类:

1
2
3
4
5
6
7
public class ShowToastImpl implements IShowToast {  

@Override
public String getToast() {
return "我来自另一个dex文件";
}
}

3.利用Android Studio的build.gradle生成jar包

eclipse生成jar包是非常简单的,在Android Studio里面可以利用build.gradle生成jar包,记得在app下的build.gradle中输入下面的代码:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
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/xiasuhuei321/firstpro/')
//打包到jar后的目录结构
// into('src/main/java/com/xiasuhuei/firstpro/')
into('com/xiasuhuei321/firstpro/')
//去掉不需要打包的目录和文件
exclude('test/', 'IShowToast.class', 'BuildConfig.class', 'R.class')
//去掉R$开头的文件
exclude{ it.name.startsWith('R$');}
}
makeJar.dependsOn(clearJar, build)

生成的jar包在app/build/libs中

4.利用dx工具生成dex

dx工具在sdk下的build-tools中,我这里将dx的目录配置到.bash.profile中了,可以直接用dx命令。windows可以像我一样配置系统变量即可,也可以切换到build-tools目录中执行dx命令。mac或者linux也可以切换到该目录,然后./dx执行。

1
dx --dex --output=/Users/lj/Desktop/duijie/dynamic_dex.jar /Users/lj/Desktop/duijie/dynamic.jar 

前一个是dex输出位置,后一个是当前jar包位置(我这里把jar拷贝出来了)。生成新的jar之后解压看一下里面有啥:

有啥
原来dex就在这个jar里面。

5.通过生成dex加载实现类

因为这个3.0的预览版点了个支持kotlin,就用kotlin写了一小段,不过个人觉得并不影响阅读,毕竟没什么难的代码。首先将生成的jar拷贝到项目的assets目录中。首先是文件拷贝操作:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
fun Context.copyFiles(fileName: String, desFile: File) {
val input = assets.open(fileName)
val output = FileOutputStream(desFile.absoluteFile)
val data: ByteArray = kotlin.ByteArray(1024)
var i: Int
while (true) {
i = input.read(data)
if (i == -1) break
output.write(data, 0, i)
}

input.close()
output.close()

}

给Context简单的扩展一个复制文件的方法,kotlin的特性,挺好使。

当然原项目中的接口实现类可以删了,目前目录底下是这些东西:

目录

MainActivity:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
class MainActivity : AppCompatActivity() {

override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContentView(R.layout.activity_main)
loadDexClass()
}

fun loadDexClass(){
var cacheFile = externalCacheDir
var internalPath = cacheFile.absolutePath+ File.separator+"dynamic_dex.jar"
var desFile = File(internalPath)
if(!desFile.exists()){
desFile.createNewFile()
copyFiles("dynamic_dex.jar",desFile)
}

// 开始加载dex class
val dexClassLoader = DexClassLoader(internalPath,cacheFile.absolutePath,null,classLoader)
val libClazz = dexClassLoader.loadClass("com.xiasuhuei321.firstpro.ShowToastImpl")
val impl = libClazz.newInstance() as IShowToast
impl.showToast(this)
}
}

最后的执行结果:

结果

整个流程比较简单,不需要赘述了。