一、温故动态加载ClassLoader机制
如果对Android的ClassLoader加载机制不熟悉,猛戳Android插件化开发动态加载基础之ClassLoader工作机制 http://blog.csdn.net/u011068702/article/details/53248960
二、介绍
我们知道在Android中可以跟java一样实现动态加载jar,但是Android使用德海Dalvik VM,不能直接加载java打包jar的byte code,需要通过dx工具来优化Dalvik bytecode。
Android在API中给出可动态加载的有:DexClassLoader 和 PathClassLoader(上面连接已经详细介绍)
DexClassLoader:可加载jar、apk和dex,可以从SD卡中加载(这篇博客采用这种方式)
PathClassLoader:只能加载已经安装搭配Android系统中的apk文件
三、曝Demo照片,不要怕,不多,很简单
四、编写接口文件
package com.example.testclassloader; public interface ShowString { public String sayChenyu(); }
五、编写接口实现文件
package com.example.testclassloader; import android.util.Log; public class ShowStringClass implements ShowString{ public static final String TAG = "ShowStringClass"; @Override public String sayChenyu() { String chenyu = "chenyu"; Log.i(TAG, chenyu); return chenyu; } }
六、打包成jar文件编译成dex
我们把ShowStringClass.java文件打包生成showStringClass.jar文件,然后把文件放到sdk目录下的build-tools下的23.0.1目录下,我用的是ubuntu,所以会看到dex文件,如果是window会在这个目录下看到dex.bat文件,然后用下面命令把showStringClass.jar生成showStringClass_imle.jar的dex文件
dx --dex --output=showStringClass_impl.jar showStringClass.jar
然后再把showStringClass_impl.jar文件放到手机目录里面去用这个命令
adb push showStringClass_impl.jar /sdcard/
具体操作图片如下
七、然后编写MainActivity.java文件
package com.example.testclassloader; import java.io.File; import android.content.Context; import android.os.Bundle; import android.os.Environment; import android.support.v7.app.ActionBarActivity; import android.util.Log; import android.widget.TextView; import dalvik.system.DexClassLoader; public class MainActivity extends ActionBarActivity { public static final String TAG = "MainActivityClassLoader"; public static final String SHOWSTRINGCLASS = "showStringClass_impl.jar"; public static final String SHOWSTRINGCLASS_PATH= "com.example.testclassloader.ShowStringClass"; public static final String DEX = "dex"; public ShowStringClass mShowStringClass = null; public TextView mTv = null; public int i = 0; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); mTv = (TextView)findViewById(R.id.hello); DexClassLoader(this); } /** * 使用DexClassLoader方式加载类 */ public void DexClassLoader(Context context) { // dex压缩文件的路径(可以是apk,jar,zip格式) String dexPath = Environment.getExternalStorageDirectory().toString() + File.separator + SHOWSTRINGCLASS; // dex解压释放后的目录 String dexOutputDirs = Environment.getExternalStorageDirectory().toString(); //指定dexoutputpath为APP自己的缓存目录 File dexOutputDir = context.getDir(DEX, 0); // 定义DexClassLoader // 第一个参数:是dex压缩文件的路径 // 第二个参数:是dex解压缩后存放的目录 // 第三个参数:是C/C++依赖的本地库文件目录,可以为null // 第四个参数:是上一级的类加载器 //DexClassLoader dexClassLoader = new DexClassLoader(dexPath,dexOutputDirs,null,getClassLoader()); DexClassLoader dexClassLoader = new DexClassLoader(dexPath,dexOutputDir.getAbsolutePath(),null,getClassLoader()); Class libProvierClazz = null; // 使用DexClassLoader加载类 try { libProvierClazz = dexClassLoader.loadClass(SHOWSTRINGCLASS_PATH); // 创建dynamic实例 mShowStringClass = (ShowStringClass) libProvierClazz.newInstance(); if (mShowStringClass != null) { final String chenyu = mShowStringClass.sayChenyu(); if (chenyu != null) { mTv.post(new Runnable() { @Override public void run() { mTv.setText(chenyu); } }); } } else { Log.d(TAG, "mShowStringClass is null"); } } catch (Exception e) { e.printStackTrace(); } } /** * 打印系统的classLoader */ public void showClassLoader() { ClassLoader classLoader = getClassLoader(); if (classLoader != null){ Log.i(TAG, "[onCreate] classLoader " + i + " : " + classLoader.toString()); while (classLoader.getParent()!=null){ classLoader = classLoader.getParent(); Log.i(TAG,"[onCreate] classLoader " + i + " : " + classLoader.toString()); i++; } } } } @Override public void run() { mTv.setText(chenyu); } }); } } else { Log.d(TAG, "mShowStringClass is null"); } } catch (Exception e) { e.printStackTrace(); } } /** * 打印系统的classLoader */ public void showClassLoader() { ClassLoader classLoader = getClassLoader(); if (classLoader != null){ Log.i(TAG, "[onCreate] classLoader " + i + " : " + classLoader.toString()); while (classLoader.getParent()!=null){ classLoader = classLoader.getParent(); Log.i(TAG,"[onCreate] classLoader " + i + " : " + classLoader.toString()); i++; } } } }
@Override
public void run() {
mTv.setText(chenyu);
}
});
}
} else {
Log.d(TAG, "mShowStringClass is null");
}
} catch (Exception e) {
e.printStackTrace();
}
}
/**
* 打印系统的classLoader
*/
public void showClassLoader() {
ClassLoader classLoader = getClassLoader();
if (classLoader != null){
Log.i(TAG, "[onCreate] classLoader " + i + " : " + classLoader.toString());
while (classLoader.getParent()!=null){
classLoader = classLoader.getParent();
Log.i(TAG,"[onCreate] classLoader " + i + " : " + classLoader.toString());
i++;
}
}
}
}
八、运行Demo的结果爆照
在ubuntu终端打印结果如下
手机上面照片如下
说明加载外部的文件加载成功了
如果把上面那行代码改成这个
DexClassLoader dexClassLoader = new DexClassLoader(dexPath,dexOutputDirs,null,getClassLoader());
会报下面的错误
需要加上缓存Dex文件的目录
//指定dexoutputpath为APP自己的缓存目录 File dexOutputDir = context.getDir(DEX, 0);
九、总结
1、加深动态加载的理解
2、如何实现项目加载外部的Dex文件有了更好的理解
3、对DexClassLoader 、dexClassLoader.load(package.class)、 class.newInstance() 有了更好的理解