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

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

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,588评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,456评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,146评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,387评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,481评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,510评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,522评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,296评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,745评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,039评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,202评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,901评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,538评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,165评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,415评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,081评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,085评论 2 352

推荐阅读更多精彩内容