动态加载DEX

这两天研究了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  IDynamic.java

 

[java]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. public interface IDynamic  
  2.   
  3.     public void init(Context context);  
  4.       
  5.     public void showTipe();  
  6.       
  7.     public void destory();  
  8.  

创建实现类 DynamicImpl.java

 

 

[java]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. public class DynamicImpl implements IDynamic{  
  2.   
  3.     Context mContext;  
  4.     @Override  
  5.     public void init(Context context)  
  6.         this.mContext=context;  
  7.      
  8.      @Override  
  9.     public void destory()  
  10.          mContext=null 
  11.      
  12.   
  13.     @Override  
  14.     public void showTipe()  
  15.         Toast.makeText(mContext, "comeing dynamic tipe!"Toast.LENGTH_LONG).show();  
  16.      
  17.       
  18.  

生产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

 

[java]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. public class LoadActivity extends Activity  
  2.   
  3.     //动态类加载接口  
  4.     private IDynamic lib;  
  5.       
  6.     @Override  
  7.     protected void onCreate(Bundle savedInstanceState)  
  8.         super.onCreate(savedInstanceState);  
  9.         setContentView(R.layout.activity_main);  
  10.         loadDexByDexClassLoader();  
  11.         Button showtip (Button) findViewById(R.id.btshowTipe);  
  12.              
  13.          showtip.setOnClickListener(new View.OnClickListener()  
  14.                 public void onClick(View view)  
  15.                    if(lib != null){  
  16.                        lib.showTipe();  
  17.                    }else 
  18.                        Toast.makeText(getApplicationContext(), "类加载失败"1500).show();  
  19.                     
  20.                  
  21.             });  
  22.      
  23.       
  24.       
  25.     void loadDexByDexClassLoader(){  
  26.           
  27.         //dex file path(file is apk or jar or zip格式)  
  28.         String dexPath Environment.getExternalStorageDirectory().toString() File.separator "dynamic_temp.jar" 
  29.         //dex解压释放后的目录  
  30.         File dexOutputDirs getApplicationContext().getDir("dex"0);  
  31.         //解压目录不能为外存储目录,这里google考虑到安全问题,外部存储会报异常  
  32.         //String dexOutputDirs Environment.getExternalStorageDirectory().toString();  
  33.         //1,dex压缩文件的路径  2,dex解压缩后存放的目录  3,C/C++依赖的本地库文件目录,可以为null,4,上一级的类加载器  
  34.         DexClassLoader cl new DexClassLoader(dexPath,dexOutputDirs.getAbsolutePath(),null,getClassLoader());  
  35.          //类的装载实现  
  36.         try  
  37.             //使用DexClassLoader加载类  
  38.             Class libProviderClazz cl.loadClass("com.dymamic.impl.DynamicImpl");  
  39.             lib (IDynamic)libProviderClazz.newInstance();  
  40.             if(lib != null){  
  41.                 lib.init(this);  
  42.              
  43.         catch (ClassNotFoundException e)  
  44.             e.printStackTrace();  
  45.         catch (InstantiationException e)  
  46.             e.printStackTrace();  
  47.         catch (IllegalAccessException e)  
  48.             e.printStackTrace();  
  49.          
  50.      
  51.       
  52.  

布局文件:

 

 

[java]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. "http://schemas.android.com/apk/res/android"  
  2.     xmlns:tools="http://schemas.android.com/tools"  
  3.     android:layout_width="match_parent"  
  4.     android:layout_height="match_parent"  
  5.     android:paddingBottom="@dimen/activity_vertical_margin"  
  6.     android:paddingLeft="@dimen/activity_horizontal_margin"  
  7.     android:paddingRight="@dimen/activity_horizontal_margin"  
  8.     android:paddingTop="@dimen/activity_vertical_margin"  
  9.     tools:context=".MainActivity"  
  10.   
  11.     
  12.         android:layout_width="match_parent"  
  13.         android:layout_height="match_parent"  
  14.         android:orientation="vertical"  
  15.          
  16.     
  17.         android:id="@+id/btshowTipe"  
  18.         android:layout_width="match_parent"  
  19.         android:layout_height="wrap_content"  
  20.         android:text="showTipe"  
  21.         />  
  22.       
  23.          
  24.          
  25.   

执行效果:

 



可能会出现异常问题:

导出jar时不能带接口文件,否则会报以下错:
java.lang.IllegalAccessError: Class ref in pre-verified class resolved to unexpected implementation


2014 year 6 month 7day 补充:

如果jar(dex转换过的Jar文件)进行网络下载更新,首先将源文件jav进行加密处理,下载完成后,拷贝到项目内部,进行解密处理(最后是.so进行)程序退出后删除该文件,下次启动重新解析,这样安全性能高一些。

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值