java dexclassloader_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) {

super.onCreate(savedInstanceState);

this.setContentView(R.layout.main);

}

@SuppressWarnings("unused")

private void sayHello(String msg){

Log.d("mmtag",msg);

}

}

接着建立另外一个工程,在这个工程中调用client工程中的view和调用sayHello()方法。在这个类中主要涉及到了DexClassLoader这个类的使用。

package com.example.dexclassloaderserver;

import java.lang.reflect.Method;

import java.util.Collections;

import java.util.List;

import dalvik.system.DexClassLoader;

import android.annotation.SuppressLint;

import android.app.Activity;

import android.content.Intent;

import android.content.pm.ActivityInfo;

import android.content.pm.PackageManager;

import android.content.pm.ResolveInfo;

import android.os.Bundle;

import android.util.Log;

public class MainActivity extends Activity {

@Override

protected void onCreate(Bundle savedInstanceState) {

super.onCreate(savedInstanceState);

setContentView(R.layout.activity_main);

useDexClassLoader();

}

@SuppressLint("NewApi")

public void useDexClassLoader() {

Intent mIntent = new Intent();

mIntent.setClassName("com.example.dexclassloaderclient",

"com.example.dexclassloaderclient.MainActivity");

PackageManager pm = this.getPackageManager();

List mList = pm.queryIntentActivities(mIntent,

PackageManager.MATCH_DEFAULT_ONLY);

ResolveInfo info = mList.get(0);

String apkPath = info.activityInfo.applicationInfo.sourceDir;

String optPath = this.getCodeCacheDir().getAbsolutePath();

String libPath = info.activityInfo.applicationInfo.nativeLibraryDir;

DexClassLoader clsLoader = new DexClassLoader(apkPath, optPath,

libPath, this.getClass().getClassLoader());

try {

Class cls = clsLoader

.loadClass("com.example.dexclassloaderclient.MainActivity");

Object obj = cls.newInstance();

Method invokeMethod = cls.getDeclaredMethod("sayHello",

new Class[] { String.class });

invokeMethod.setAccessible(true);

invokeMethod.invoke(obj, "hello world,DexClassLoader");

} catch (Exception e) {

e.printStackTrace();

}

}

}

生成的结果为:

I/dex2oat (20250): dex2oat took 1.547s (threads: 4)

D/mmtag (20229): hello world,DexClassLoader

D/OpenGLRenderer(20229): Render dirty regions requested: tru

这样就成功的调用了其他的apk中的方法。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值