最近想研究一波插件化,在簡書上看到這個感覺還不錯,就按照這個順序看下去了。不過得小記一下,省的以后我還得去上面提到的文里去找鏈接。我這里記得就是自己操作的一遍流程。。其實我也是根據(jù)他的文來操作的,屬于HelloWorld級別吧。
操作
本來還想介紹一下概念,想了想,算了,不誤人子弟了。感興趣自己搜。
1.新建工程
沒什么好說的,打開Android Studio新建一個工程都會吧。我這里新建了一個名為fordex的項目,包名為com.xiasuhuei321.firstpro。
2.新建一個接口和一個類
接口和類非常簡單,類是接口的實現(xiàn)類,主要功能就是彈一個Toast。接口:
public interface IShowToast {
public int showToast(Context context);
}
實現(xiàn)類:
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后的目錄結(jié)構(gòu)
// 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可以像我一樣配置系統(tǒng)變量即可,也可以切換到build-tools目錄中執(zhí)行dx命令。mac或者linux也可以切換到該目錄,然后./dx執(zhí)行。
dx --dex --output=/Users/lj/Desktop/duijie/dynamic_dex.jar /Users/lj/Desktop/duijie/dynamic.jar
前一個是dex輸出位置,后一個是當前jar包位置(我這里把jar拷貝出來了)。生成新的jar之后解壓看一下里面有啥:

原來dex就在這個jar里面。
5.通過生成dex加載實現(xiàn)類
因為這個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的特性,挺好使。
當然原項目中的接口實現(xiàn)類可以刪了,目前目錄底下是這些東西:

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)
}
}
最后的執(zhí)行結(jié)果:

整個流程比較簡單,不需要贅述了。