最近想研究一波插件化,在简书上看到这个感觉还不错,就按照这个顺序看下去了。不过得小记一下,省的以后我还得去上面提到的文里去找链接。我这里记得就是自己操作的一遍流程。。其实我也是根据他的文来操作的,属于HelloWorld级别吧。
操作
本来还想介绍一下概念,想了想,算了,不误人子弟了。感兴趣自己搜。
1.新建工程
没什么好说的,打开Android Studio新建一个工程都会吧。我这里新建了一个名为fordex的项目,包名为com.xiasuhuei321.firstpro。
2.新建一个接口和一个类
接口和类非常简单,类是接口的实现类,主要功能就是弹一个Toast。接口:
public interface IShowToast {
public int showToast(Context context);
}
实现类:
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中输入下面的代码:
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执行。
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目录中。首先是文件拷贝操作:
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:
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)
}
}
最后的执行结果:
整个流程比较简单,不需要赘述了。