DexClassLoader使用-(1)

可以先参考java的java类加载器

一、DexClassLoader

  一般情况下,我们使用import就可以了,为什么还要使用类装载器呢?

import中所引用的类文件有两个特点:

1.一定在存在于本地,当程序运行时需要这个类时,内部类装载器就会自动装载,程序员感知不到这个过程。

2.编译时一定要在现场,否则会因找不到引用文件而不能正常编译。

  但是在有的情况下,要用的类不一定满足这些条件,如从运程下载一个类并在本地运行,例子如applet执行的java,另一种是要引用的class文件不方便在编译时直接参与 ,而只能在运行时动态调用。如在android中修改FrameWork中已经有的类,为了保持与原生的FrameWork最小化的修改,可以使用类装载器动态装载自己定义 的jar包。

二、DexClassLoader的使用方法

  有两个apk。

  • 插件部分

定义了类PluginClass。如下

public class PluginClass {
    public PluginClass(){
        Log.d("PluginCLass", "PluginClass initiazed");
    }

    public int function1(int a,int b){
        return a +b;
    }
}

还定义了

一个空的Activity,这个作用只是为了让宿主apk中可以找到package而已,所以只要有就可以了。

<activity
            android:name=".BlankActivity"
            android:label="@string/title_activity_blank" >
            <intent-filter>
                <action android:name="com.example.lsj.testapp.plugin.client"/>
            </intent-filter>
        </activity>

我们将可以通过intent-filter得到相关的信息。

  • 宿主部分

通过下面的ClassLoader可以装载插件中的类,再构造出Method对象 ,并构造出Method对象所使用的参数对象 ,然后才能调用 。

ublic void useDexClassLoader(){
        Intent intent = new Intent("com.example.lsj.testapp.plugin.client", null);
        PackageManager pm = getPackageManager();
        final List<ResolveInfo> plugins = pm.queryIntentActivities(intent, 0);
        ResolveInfo rinfo = plugins.get(0);
        ActivityInfo ainfo = rinfo.activityInfo;

        String div = System.getProperty("path.separator");
        String packageName = ainfo.packageName;
        String dexPath = ainfo.applicationInfo.sourceDir;
        String dexOutputDir = getApplicationInfo().dataDir;
        String libPath = ainfo.applicationInfo.nativeLibraryDir;

        DexClassLoader dexClassLoader = new DexClassLoader(dexPath,
                dexOutputDir,libPath,this.getClass().getClassLoader());
        try{
            Class <?> clazz = dexClassLoader.loadClass(packageName+".PluginClass");
            Object obj = clazz.newInstance() ;
            Class [] params = new Class[2];
            params[0]= Integer.TYPE;
            params[1] = Integer.TYPE ;
            Method action = clazz.getMethod("function1", params) ;
            Integer ret = (Integer)action.invoke(obj, 1,2);
            Log.e(TAG, "return value is :"+ ret);

        }catch ( ClassNotFoundException e){

        }catch (InstantiationException e2 ){

        }catch (IllegalAccessException e3){

        }catch (NoSuchMethodException e4){

        }catch (InvocationTargetException e5){

        }
    }

DexClassLoader方法参数:

dexPath:目标所在的apk或者jar文件的路径,装载器将从路径中寻找指定的目标类。

dexOutputDir:由于dex 文件在APK或者 jar文件中,所以在装载前面前先要从里面解压出dex文件,这个路径就是dex文件存放的路径,在 android系统中,一个应用程序对应一个linux用户id ,应用程序只对自己的数据目录有写的权限,所以我们存放在这个路径中。

libPath :目标类中使用的C/C++库。

最后一个参数是该装载器的父装载器,一般为当前执行类的装载器。

 这里我们得到的Class对象是CLassLoader可以识别的类,而PluginClass是程序执行后可以识别的类,此时仅仅装载了PluginClass的程序代码,还没有创建出其对象 ,接下来可以使用Class.newInsance()方法来调用PluginClass类的构造方法,并返回一个真正的PluginClass对象。

  但是我们不能直接去调用 PluginClass的任何方法,只能通过反射去调用其方法。

  反射机制时主要使用了Method类,Class对象的getMethod()方法可以返回这个类中任何方法,getMethod()有两个参数 ,一个是要访问的函数名,另一个是函数的参数类型。

  得到目标类的Method对象后,就可以调用Method对象的invoke方法,这个方法的第一个参数是执行目标函数的对象。

 

时间: 2024-10-08 22:44:16

DexClassLoader使用-(1)的相关文章

DexClassLoader

DexClassLoader加载自己写的第三方jar包,例如金山毒霸需要加载ksremote.jar. 现在将DexClassLoader加载jar包成果分享一下.   1.新建Android工程,封装功能java类.   2.选中需要导出的文件夹,右键选中“Export”->"Java(Jar file)"导出jar文件.   3.使用dx工具将jar包转换为android 字节码.      命令:dx  --dex --Output=xx.jar    hello.jar 

Andorid DexClassLoader的创建过程解析(基于5.0)

做Android插件框架时,经常会用到dex的动态加载,就要直接或间接的使用DexClassLoader,在new DexClassLoader的时候Android系统做了很多工作,下面我们详细分析一下: public class DexClassLoader extends BaseDexClassLoader { public DexClassLoader(String dexPath, String optimizedDirectory, String libraryPath, Class

通过DexClassLoader动态加载代码

动态加载代码,会有多种需求,有的是APK过大,想缩小点:有的是部分代码需要灵活变动,例如视频站点的解析规则. 奉上一个Demo,在这个demo中验证了从dexclassloader加载 1.db 2.,sharedpreference 3.webview 4. so库 5.context 6.传入listener回调 7.加载不同package下的类 欢迎大家下载: http://download.csdn.net/download/ameryzhu/8970167 版权声明:本文为博主原创文章

Android中的类装载器DexClassLoader

http://blog.csdn.net/com360/article/details/14125683 java中,有个概念叫做“类加载器”(ClassLoader),它的作用就是动态的装载Class文件.标准的java sdk中有一个 ClassLoader类,借助这个类可以装载想要的Class文件,每个ClassLoader对象在初始化时必须制定Class文件的路径. 可能有人会问,在写程序的时候不是有import关键字可以引用制定的类吗?为何还要使用这个类加载器呢? 原因其实是这样的,使

Android插件化(二):使用DexClassLoader动态加载assets中的apk

Android插件化(二):使用DexClassLoader动态加载assets中的apk 简介 上一篇博客讲到,我们可以使用MultiDex.java加载离线的apk文件.需要注意的是,apk中的类是加载到当前的PathClassLoader当中的,如果apk文件过多,可能会出现ANR的情况.那么,我们能不能使用DexClassLoader加载apk呢?当然是可以的!首先看一下Doc文档. A class loader that loads classes from .jar and .apk

android DexClassLoader动态加载技术详解

介绍 做项目到一定庞大的时候就会发现方法数太多,安装包根本就装不上去了,这个也不足为奇,我们都知道当方法数目超过65536这个数目限制的时候,挡在2.x的系统上面就会出现无法安装的情况,这个时候动态加载技术就显得非的重要了,我们的项目中为了兼容2.x的手机也是用到了android的动态加载技术,这里我会详细的讲解一下怎么去用,怎么实战,我感觉,空谈理论不如动手来得实在. 实例 下面就通过一个例子反复的说明怎么来实现动态加载,通过不同的方法来调用. 准备工作 1:新建一个java工程(我比较懒我就

DexClassLoader和PathClassLoader类载入机制

0x00 在DexClassLoader和PathClassLoader载入Dex流程一文中,我们分析了dex文件怎样形成了DexFile结构体.本文中解说类载入机制,实际上就是生成ClassObject对象. 我们以DexClassLoader为例.解说类载入机制,PathClassLoader是一样的. 我们在载入类时一般会调用loadClass,那么我们就从loadClass来開始分析. 0x01 DexClassLoader类没有loadClass方法.所以调用的是父类ClassLoad

Android插件化开发之DexClassLoader动态加载dex、jar小Demo

一.温故动态加载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

被用坏的DexClassLoader,关于DexClassLoader内存泄露的问题

原因: 目前有两种猜测,第一种就是类加载容易,但是类卸载就不那么容易了,第二种猜测就是类缓存机制,从classloader的运行过程可以分析到有个过程是缓存常用类的,感觉这个过程没法释放,因为他会认为你以后会一直用. 测试内存泄露代码: 可以用DDMS测试下,当你调用GC的时候byte是可以回收的,但是classloader就没法回收了,也许你认为是因为调用了activity的classloader造成的,但是回头想想这个classloader可是整个应用的,除非你关闭应用,否则无法去掉clas