这两天研究了android中动态装载功能,在项目中应用主要考虑到两大方面:
1,反破解,现在app的保护机制做的很不好,随便一个简单的破解工具,就可以对app进行反编译,进行二次打包(现在盗版app很猖獗,打包党很多进行植入广告,后门程序等手段,严重影响用户和app发行单位利益)
2,可以避免多次升级app,直接通过动态装载来源网络jar,dex即可完成。程序扩展做到了最好方式。
下面谈谈动态加载实现:
理论:
Dalvik虚拟机如同其他Java虚拟机一样,在运行程序时首先需要将对应的类加载到内存中。而在Java标准的虚拟机中,类加载可以从class文件中读取,
也可以是其他形式的二进制流,因此,我们常常利用这一点,在程序运行时手动加载Class,从而达到代码动态加载执行的目的
然而Dalvik虚拟机毕竟不算是标准的Java虚拟机,因此在类加载机制上,它们有相同的地方,也有不同之处。我们必须区别对待,Dalvik虚拟机识别的是dex文件,而不是class文件。因此,我们供类加载的文件也只能是dex文件,或者包含有dex文件的.apk或.jar文件。
在android中和类加载相关的两个类,DexClassLoader和PathClassLoader,
DexClassLoader
这个可以加载jar/apk/dex,也可以从SD卡中加载(一般测试使用该方式加载,一般使用内部存储目录 File dexOutputDir = context.getDir("dex", 0);)。
PathClassLoader (目前用途有限,基本很少使用)
只能加载已经安装到Android系统中的apk文件。
有一个细节,可能大家不容易注意到。PathClassLoader是通过构造函数new DexFile(path)来产生DexFile对象的;而DexClassLoader则是通过其静态方法loadDex(path, outpath, 0)
得到DexFile对象。这两者的区别在于DexClassLoader需要提供一个可写的outpath路径,用来释放.apk包或者.jar包中的dex文件。换个说法来说,
就是PathClassLoader不能主动从zip包中释放出dex,因此只支持直接操作dex格式文件,或者已经安装的apk(因为已经安装的apk在cache中存在缓存的dex文件)。
而DexClassLoader可以支持.apk、.jar和.dex文件,并且会在指定的outpath路径释放出dex文件。
加载好类后,通常我们可以通过Java反射机制来使用这个类但是这样效率相对不高,而且也比较复杂凌乱。更好的做法是定义一个interface,
并将这个interface写进容器端。待加载的类,继承自这个interface,并且有一个参数为空的构造函数,
以使我们能够通过Class的newInstance方法产生对象然后将对象强制转换为interface对象,就可以直接调用成员方法了。
实现:
创建android工程1,目录结构:
创建interface
- public
interface IDynamic { -
-
public void init(Context context); -
-
public void showTipe(); -
-
public void destory(); - }
创建实现类 DynamicImpl.java
- public
class DynamicImpl implements IDynamic{ -
-
Context mContext; -
@Override -
public void init(Context context) { -
this.mContext=context; -
} -
@Override -
public void destory() { -
mContext=null; -
} -
-
@Override -
public void showTipe() { -
Toast.makeText(mContext, "comeing dynamic tipe!", Toast.LENGTH_LONG).show(); -
} -
- }
生产jar包,注意这里打包的是IDynamic的实现类DynamicImpl.java,不打包接口类IDynamic.java
然后将打包好的jar文件拷贝到android的安装目录中的platform-tools目录下,使用dx命令:(我的jar文件是dynamicImp.jar)
dx --dex --output=dynamicImp_temp.jar dynamicImp.jar
这样就生成了dynamicImp_temp.jar,这个jar和dynamicImp.jar有什么区别呢?
其实这条命令主要首先将dynamicImp.jar编译成dynamicImp.dex文件(Android虚拟机认识的字节码文件),然后再将dynamicImp.dex文件压缩成dynamicImp_temp.jar,
当然你也可以压缩成.zip格式的,或者直接编译成.apk文件都可以的,最后把经过处理的dynamicImp文件放到sd卡根目录下
接下来打包interface类,IDynamic.java-->IDynamic.jar
创建android project2,将IDynamic.jar放入Libs目录
LoadActivity.java
- public
class LoadActivity extends Activity { -
-
//动态类加载接口 -
private IDynamic lib; -
-
@Override -
protected void onCreate(Bundle savedInstanceState) { -
super.onCreate(savedInstanceState); -
setContentView(R.layout.activity_main); -
loadDexByDexClassLoader(); -
Button showtip = (Button) findViewById(R.id.btshowTipe); -
-
showtip.setOnClickListener(new View.OnClickListener() { -
public void onClick(View view) { -
if(lib != null){ -
lib.showTipe(); -
}else{ -
Toast.makeText(getApplicationContext(), "类加载失败", 1500).show(); -
} -
} -
}); -
} -
-
-
void loadDexByDexClassLoader(){ -
-
//dex file path(file is apk or jar or zip格式) -
String dexPath = Environment.getExternalStorageDirect ory().toString() + File.separator + "dynamic_temp.jar"; -
//dex解压释放后的目录 -
File dexOutputDirs = getApplicationContext().getDir("dex", 0); -
//解压目录不能为外存储目录,这里google考虑到安全问题,外部存储会报异常 -
//String dexOutputDirs = Environment.getExternalStorageDirect ory().toString(); -
//1,dex压缩文件的路径 2,dex解压缩后存放的目录 3,C/C++依赖的本地库文件目录,可以为null,4,上一级的类加载器 -
DexClassLoader cl = new DexClassLoader(dexPath,dexOutputDirs.getAbsolutePath(),null,getClassLoader()); -
//类的装载实现 -
try { -
//使用DexClassLoader加载类 -
Class libProviderClazz = cl.loadClass("com.dymamic.impl.DynamicImpl"); -
lib = (IDynamic)libProviderClazz.newInstance(); -
if(lib != null){ -
lib.init(this); -
} -
} catch (ClassNotFoundException e) { -
e.printStackTrace(); -
} catch (InstantiationException e) { -
e.printStackTrace(); -
} catch (IllegalAccessException e) { -
e.printStackTrace(); -
} -
} -
- }
布局文件:
- "http://schemas.android.com/apk/res/android"
-
xmlns:tools="http://schemas.android.com/tools" -
android:layout_width="match_parent" -
android:layout_height="match_parent" -
android:paddingBottom="@dimen/activity_vertical_margin" -
android:paddingLeft="@dimen/activity_horizontal_margin" -
android:paddingRight="@dimen/activity_horizontal_margin" -
android:paddingTop="@dimen/activity_vertical_margin" -
tools:context=".MainActivity" > -
-
-
android:layout_width="match_parent" -
android:layout_height="match_parent" -
android:orientation="vertical" -
> -
-
android:id="@+id/btshowTipe" -
android:layout_width="match_parent" -
android:layout_height="wrap_content" -
android:text="showTipe" -
/> -
-
-
-
执行效果:
可能会出现异常问题:
导出jar时不能带接口文件,否则会报以下错:
java.lang.IllegalAccessError: Class ref in pre-verified class resolved to unexpected implementation
2014 year 6 month 7day 补充:
如果jar(dex转换过的Jar文件)进行网络下载更新,首先将源文件jav进行加密处理,下载完成后,拷贝到项目内部,进行解密处理(最后是.so进行)程序退出后删除该文件,下次启动重新解析,这样安全性能高一些。