• dalvik 动态加载jar,dex功能


    这两天研究了android中动态装载功能,在项目中应用主要考虑到两大方面:

    1,反破解,现在app的保护机制做的很不好,随便一个简单的破解工具,就可以对app进行反编译,进行二次打包(现在盗版app很猖獗,打包党很多进行植入广告,后门程序等手段,严重影响用户和app发行单位利益)

    2,可以避免多次升级app,直接通过动态装载来源网络jar,dex即可完成。程序扩展做到了最好方式。

    3,   解决Android 应用程序方法总数不能超过65K的问题,android程序一个dex文件只能包含65535个方法,当程序超过是会报异常如下
    Conversion to Dalvik format failed:Unable toexecute dex: method ID not in [0, 0xffff]: 65536

    这个时候可以采取动态加载机制解决此问题,腾讯游戏client端就是这么实现的。


    下面谈谈动态加载实现:

    理论:

    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

    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();
    	               }
    	            }
    	        });
        }
        
        /**使用DexClassLoader方式加载类*/
        void loadDexByDexClassLoader(){
        	
            //dex file path(file is apk or jar or zip格式)
            String dexPath = Environment.getExternalStorageDirectory().toString() + File.separator + "dynamic_temp.jar";
            //dex解压释放后的目录
            File dexOutputDirs = getApplicationContext().getDir("dex", 0);
            //解压目录不能为外存储目录,这里google考虑到安全问题,外部存储会报异常
            //String dexOutputDirs = Environment.getExternalStorageDirectory().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();
    		}
        }
        
    }

    布局文件:

    <RelativeLayout xmlns:android="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" >
    
        <LinearLayout
            android:layout_width="match_parent"
        	android:layout_height="match_parent"
        	android:orientation="vertical"
            >
    	<Button
    	    android:id="@+id/btshowTipe"
    	    android:layout_width="match_parent"
        	android:layout_height="wrap_content"
        	android:text="showTipe"
    	    />
    	
           </LinearLayout>
           
    </RelativeLayout>
    

    执行效果:



    可能会出现异常问题:

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


    2014 year 6 month 7day 补充:

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




  • 相关阅读:
    MAC OpenGL 环境搭建
    C++中调用OC代码
    XCode快捷键使用
    【iOS】史上最全的iOS持续集成教程 (下)
    【iOS】史上最全的iOS持续集成教程 (上)
    pod 指令无效
    iOS面试题总结(持续更新)
    数据结构与算法思维导图
    Swift编码规范总结
    同步异步执行问题
  • 原文地址:https://www.cnblogs.com/happyxiaoyu02/p/6818946.html
Copyright © 2020-2023  润新知