Android开发中类加载器DexClassLoader的简单使用讲解
简介
“类装载器”(ClassLoader),顾名思义,就是用来动态装载class文件的。标准的JavaSDK中有个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()的方法。
publicclassMainextendsActivity{
@Override
protectedvoidonCreate(BundlesavedInstanceState){
super.onCreate(savedInstanceState);
this.setContentView(R.layout.main);
}
@SuppressWarnings("unused")
privatevoidsayHello(Stringmsg){
Log.d("mmtag",msg);
}
}
接着建立另外一个工程,在这个工程中调用client工程中的view和调用sayHello()方法。在这个类中主要涉及到了DexClassLoader这个类的使用。
packagecom.example.dexclassloaderserver;
importjava.lang.reflect.Method;
importjava.util.Collections;
importjava.util.List;
importdalvik.system.DexClassLoader;
importandroid.annotation.SuppressLint;
importandroid.app.Activity;
importandroid.content.Intent;
importandroid.content.pm.ActivityInfo;
importandroid.content.pm.PackageManager;
importandroid.content.pm.ResolveInfo;
importandroid.os.Bundle;
importandroid.util.Log;
publicclassMainActivityextendsActivity{
@Override
protectedvoidonCreate(BundlesavedInstanceState){
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
useDexClassLoader();
}
@SuppressLint("NewApi")
publicvoiduseDexClassLoader(){
IntentmIntent=newIntent();
mIntent.setClassName("com.example.dexclassloaderclient",
"com.example.dexclassloaderclient.MainActivity");
PackageManagerpm=this.getPackageManager();
List<ResolveInfo>mList=pm.queryIntentActivities(mIntent,
PackageManager.MATCH_DEFAULT_ONLY);
ResolveInfoinfo=mList.get(0);
StringapkPath=info.activityInfo.applicationInfo.sourceDir;
StringoptPath=this.getCodeCacheDir().getAbsolutePath();
StringlibPath=info.activityInfo.applicationInfo.nativeLibraryDir;
DexClassLoaderclsLoader=newDexClassLoader(apkPath,optPath,
libPath,this.getClass().getClassLoader());
try{
Classcls=clsLoader
.loadClass("com.example.dexclassloaderclient.MainActivity");
Objectobj=cls.newInstance();
MethodinvokeMethod=cls.getDeclaredMethod("sayHello",
newClass[]{String.class});
invokeMethod.setAccessible(true);
invokeMethod.invoke(obj,"helloworld,DexClassLoader");
}catch(Exceptione){
e.printStackTrace();
}
}
}
生成的结果为:
I/dex2oat(20250):dex2oattook1.547s(threads:4) D/mmtag(20229):helloworld,DexClassLoader D/OpenGLRenderer(20229):Renderdirtyregionsrequested:tru
这样就成功的调用了其他的apk中的方法。