一、思路
1、这段时间因为要做一个自定义的3Dwidget,这个widget是一个时钟,想实现的功能是可以给该widget更换皮肤(View)和交互,皮肤都是又图片和字符串组成的,但是又要考虑可能用户会选择不同的皮肤(皮肤中的某些元素不需要,又或者多添加一些元素),开始想吧,可以将所有的对象都先实例化,然后再在SD卡的配置文件中读取哪些是需要的元素,但是这样太有局限性了,一个是内存会耗费,另一个是根本就不灵活。偶然在Q群看到胡哥给的一个用.class文件创建对象的文章,那么我想可不可以在android里用动态加载的方式来切换皮肤呢?我的每个类所需要做的并不多,而且每个不同皮肤都是在另一个类(类似一个ViewGroup)中被调用,那个类几乎就不需要更改。那如果我将这些类的所必须的方法抽出来做一个接口,那么ViewGroup中只要调用这个接口的方法即可,具体的实现由其子类实现。
2、整理了一下思路,决定用上一个代理模式。
public interface ITest { public void log(String content); }
public class Test implements ITest { ITest test; public Test(Context context) { getTestInstanceDynamicFromSDCard(); } private void getTestInstanceDynamicFromSDCard() { // get test instance by dynamic } @Override public void log(String content) { test.log(content); } }
如此一来ViewGroup中只要实例化Test类,调用相应的方法即可,而其具体做什么将有其内部getTestInstanceDynamicFromSDCard所得到ITest对象实现。那么具体的问题就是如何动态加载一个类并创建其对象!
二、ClassLoader类来加载
1、看了胡哥推荐的用.class文件创建对象,于是也跟着用类似的方法来实现:
(http://wlh0706-163-com.iteye.com/blog/2098619)
2、但是总会失败,并且报错“can't load this type of class file”查了资料后才知道是虚拟机的问题!
在使用标准Java虚拟机时,我们经常自定义继承自ClassLoader的类加载器。然后通过defineClass方法来从一个二进制流中加载Class。(为什么要继承ClassLoader?如果不继承他就无法使用defineClass方法),但是在android的Dalvik虚拟机里走不通,但是android有提供两个类以供使用PathClassLoader、DexClassLoader
三、DexClassLoader加载(从SD卡加载)
1、PathClassLoader因为路径设计到system/app,故不作考虑。
2、DexClassLoader可以动态加载.apk、.jar和.dex文件而且路径不受限制,而且根据我的需求,显然是将自己分离的类打包成jar包会很灵活,因为这样我可以指定主程序对哪个路径的jar包进行加载(便于切换资源)
3、将所有程序都编译好后,包括需要动态加载的类也放在同一主程序运行以保证程序正常,将需要动态加载的隔离开,如实现一个DemoTest,放到com.example.out包下
package com.example.out; import android.util.Log; import com.example.biz.ITest; public class DemoTest implements ITest { @Override public void log(String content) { Log.d("example", content); } }
4、导出成jar包(eclipse直接右击该类export->java->jar file->next->存储目录+jar报名,不如就叫Dynamic.jar->finish),此时可以将原项目中的插件部分删除咯!
5、将Dynamic。jar拷贝到android-sdk-windows\platform-tools下将其转换成DexClassLoader所能识别的二进制jar包:cd 到platform-tools目录下,执行命令dx --dex --output=DynamicTest.jar Dynamic.jar 其中DynamicTest.jar为新的jar包,是我们所需要的二进制jar包,Dynamic.jar 为eclipse直接导出的jar包
6、将DynamicTest.jar拷贝到SD卡,例如:../sbclock/DynamicTest.jar
7、程序实现动态加载:
package com.example.dynamic; import android.app.Activity; import android.os.Bundle; import com.example.biz.ITest; import com.example.biz.Test; public class MainActivity extends Activity { private ITest mTest; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); mTest = new Test(this); } @Override protected void onDestroy() { mTest.log("it is just a test"); super.onDestroy(); } }
Test类动态加载:
package com.example.biz;
import java.io.File;
import java.lang.reflect.Constructor;
import dalvik.system.DexClassLoader;
import android.annotation.SuppressLint;
import android.content.Context;
public class Test implements ITest {
private ITest mTest;
private Context mContext;
private String mJarPath;
private String mClassName;
public Test(Context context) {
mContext = context;
getTestInstanceDynamicFromSDCard();
}
@SuppressLint("NewApi")
private void getTestInstanceDynamicFromSDCard() {
// get test instance by dynamic
mJarPath = android.os.Environment.getExternalStorageDirectory()
.getAbsolutePath() + "/sbclock/DynamicTest.jar";// 前半部分为获得SD卡的目录
mClassName = "com.example.out.DemoTest";//和导出之前的包名和类名保持一致
File dexOutputDir = mContext.getDir("dex", 0);//
File file = new File(mJarPath);
DexClassLoader classLoader = new DexClassLoader(file.getAbsolutePath(),
dexOutputDir.getAbsolutePath(), null, mContext.getClassLoader());
try {
Class<?> clazz = classLoader.loadClass(mClassName);
Constructor<?> constructor = clazz.getConstructor(new Class[] {});//得到构造器
mTest = (ITest) constructor.newInstance();//实例化
} catch (Exception e) {
e.printStackTrace();
}
}
@Override
public void log(String content) {
mTest.log(content);
}
}
8、结果,当退出该程序时确实有打印:
D/example (28710): it is just a test
9、更换其他的实现方式,还是只需要使用这个框架,在Test类中注册一个广播,外来程序可以发送广播通过Intent将其路径名和包名传过来即可创建新的子类!
10.1、在切换成其他子类时需要注意的是子类的名称不要相同,在实践中本来想使用统一的包名和类名,结果因为其对象在程序内存中已经有了,所以不会再次创建!
10.2、导出的jar包其实就是主程序中的一部分,所以不论其类中需要导入其他的类(而该jar包不含有)也没关系,因为动态加载之后这个对象就是主程序中的一部分。
10.3.导出的文件资源的加载可以从主程序中加载,也可以从SD卡中加载(最好是SD卡,这样就不同的jar包可以对应不同的资源了)
10.4、这应该算是一个粗略的插件框架了,如果想实现功能多的插件,那还是得将接口考虑周到!
相关推荐
DexClassLoader学习源码,使用时,请删除Lib.Java文件,否则会出现loadClass错误。
一个动态加载android 的class框架, 过程是先解压一个完全的apk,然后创建自己的dexclassloader,然后动态加载指定的activity的过程
一、Android插件化开发,常见有三种实现方式: (1)Android 动态加载dex文件 (2)Android动态加载APK文件(代理模式) (3)Hook技术动态加载APK文件中的Activity 二、Android 动态加载dex文件 1.反射方式加载(较...
DexClassLoader自定义加载Assets目录下的dex、jar文件,对Assets目录下jar、dex进行加载
【Android 逆向】类加载器 ClassLoader ( 使用 DexClassLoader 动态加载字节码文件 | 准备 DEX 字节码文件 ) https://hanshuliang.blog.csdn.net/article/details/121776627 博客源码
首先第一个是 jar 文件的制作,Java 里面直接把 .class 文件打包到 .jar 文件里面就可以了,但是 Android 的 Dalvik VM 是不认 Java ...要动态加载其它类,可以用的 Class Loader 有: DexClassLoader,PathClassLoader
android-custom-class-loading-sample,源码。需要使用ant进行编译。编译之后可以动态的使用DexClassLoader加载代码并执行。
主要介绍了Android开发中类加载器DexClassLoader的简单使用讲解,DexClassLoader可以看作是一个特殊的Java中的ClassLoader,需要的朋友可以参考下
实现安卓的多 dex 加载和 dex 加解密,学习了解APP的加固原理。 理解APP动态加载原理。
通过对ClassLoader中两个子类加载dex,来熟悉安卓中的apk加载流程
这个一个demo,用来实现加载class文件,如果在实际项目中可以实现,动态修改代码的业务逻辑 首先在安卓中如果我们想实现的动态加载,比如知道安卓的底层运行原理, 首先安卓底层下载的时候使用的是 Classloader,同时...
用来开发Android插件APK,并通过动态加载的方式在宿主程序中运行。 若插件APK是完全独立的APK,那么插件apk也可独立安装运行。 若插件APK不是完全独立的apk,比如和插件宿主程序共用一些依赖库,那么插件apk...
普通的Android程序Dalvik虚拟机都是从一个默认的地方载入程序需要的类文件(dex文件),而Dalvik虚拟机还提供了从其他地方载入类的能力(比如从设备的内部存储空间以及互联网). 这种自定义类加载机制可以使用于一些场景...
android_dex Android DexClassLoader
加载一个Activity肯定不会像加载一般的类那样,因为activity作为系统的组件有自己的生命周期,有系统的很多回调控制,所以自定义一个DexClassLoader类加载器来加载插件中的Activity肯定是不可以的。 首先不得不了解...
Android dexclassLoader 热更demo,详细讲解请看 https://blog.csdn.net/zhou3299136/article/details/105796479
Android DexClassLoader调用其他apk中的函数 实现动态加载
简介实现Android Activity的代理机制,可以实现:在AndroidManifest.xml中仅声明一个代理Activity,即可打开多个被代理Activity被代理Activity可以动态更新,即使用DexClassLoader加载的Activity使用方法原派生于...