Android开发中类加载器DexClassLoader的简单使⽤讲解简介
“类装载器”(ClassLoader),顾名思义,就是⽤来动态装载class⽂件的。标准的Java SDK中有个ClassLoader类,借助此类可以装载需要的class⽂件,前提是ClassLoader类初始化必须制定class⽂件的路径。
import关键字引⽤的类⽂件和ClassLoader动态加载类的区别:
import引⽤类的两个特点:
1、必须存在于本地,当程序运⾏该类时,内部类装载器会⾃动装载该类。
2、编译时必须在现场,否则编译过程会因不到引⽤⽂件⽽不能正常编译。
classLoader的特点正好于import相反,⽽且更⾃由灵活。
每⼀个ClassLoader必须有⼀个⽗ClassLoader,在装载Class⽂件时,⼦ClassLoader会先请求其⽗ClassLoader加载该⽂件,只有当其⽗ClassLoader不到该⽂件时,⼦ClassLoader才会继承装载该类。这是⼀种安全机制。对于Android⽽⾔,最终的apk⽂件包含的是dex类型的⽂件,dex⽂件是将class⽂件重新打包,打包的规则⼜不是简单地压缩,⽽是完全对class⽂件内部的各种函数表,变量表进⾏优
化,产⽣⼀个新的⽂件,即dex⽂件。因此加载这种特殊的Class⽂件就需要特殊的类加载器DexClassLoader。
在Java中涉及到的类加载器就是ClassLoader这个类,通过ClassLoader.forName()的⽅法可以加载我们需要的类,从⽽实现在运⾏时动态加载类库的需求。但是在android中直接使⽤ClassLoader是⾏不通的,因为ClassLoader加载的java的字节码⽂件,⽽在android中使⽤的是dex格式的字节码,对此android专门提供了⼀个DexClassLoader类来完成动态加载apk的需求。
实例
下⾯⽤⼀个简单的例⼦来说明⼀下DexClassLoader这个类的使⽤,这个例⼦涉及到两个知识点:跨包取资源 & 反射调⽤⽅法。⾸先建⽴⼀个Client⼯程,这个⼯程很简单,只有⼀个简单的layout和⼀个sayHello()的⽅法。
public class Main extends Activity {
@Override
protected void onCreate(Bundle savedInstanceState) {
this.setContentView(R.layout.main);
}
@SuppressWarnings("unused")
private void sayHello(String msg){
Log.d("mmtag",msg);
}
}
接着建⽴另外⼀个⼯程,在这个⼯程中调⽤client⼯程中的view和调⽤sayHello()⽅法。在这个类中主要涉及到了DexClassLoader这个类的使⽤。
ample.dexclassloaderserver;
import flect.Method;
import java.util.Collections;
import java.util.List;
import dalvik.system.DexClassLoader;
import android.annotation.SuppressLint;
import android.app.Activity;
t.Intent;
t.pm.ActivityInfo;
t.pm.PackageManager;
t.pm.ResolveInfo;
import android.os.Bundle;
import android.util.Log;
public class MainActivity extends Activity {
@Override
protected void onCreate(Bundle savedInstanceState) {
setContentView(R.layout.activity_main);
useDexClassLoader();
}
@SuppressLint("NewApi")
public void useDexClassLoader() {
Intent mIntent = new Intent();
mIntent.setClassName("ample.dexclassloaderclient",
"ample.dexclassloaderclient.MainActivity");
PackageManager pm = PackageManager();
List<ResolveInfo> mList = pm.queryIntentActivities(mIntent,
PackageManager.MATCH_DEFAULT_ONLY);
ResolveInfo info = (0);
String apkPath = info.activityInfo.applicationInfo.sourceDir;
String optPath = CodeCacheDir().getAbsolutePath();
String libPath = info.activityInfo.applicationInfo.nativeLibraryDir;
DexClassLoader clsLoader = new DexClassLoader(apkPath, optPath,        libPath, Class().getClassLoader());
try {
Class cls = clsLoader
.loadClass("ample.dexclassloaderclient.MainActivity");
Object obj = wInstance();
Method invokeMethod = DeclaredMethod("sayHello",
new Class[] { String.class });
invokeMethod.setAccessible(true);
invokeMethod.invoke(obj, "hello world,DexClassLoader");
} catch (Exception e) {
e.printStackTrace();
}
}
简易安卓app开发}
⽣成的结果为:
I/dex2oat (20250): dex2oat took 1.547s (threads: 4)
D/mmtag  (20229): hello world,DexClassLoader
D/OpenGLRenderer(20229): Render dirty regions requested: tru
这样就成功的调⽤了其他的apk中的⽅法。