Android插件化開發(fā)之DexClassLoader動態(tài)加載dex、jar小Demo

一、溫故動態(tài)加載ClassLoader機制

如果對Android的ClassLoader加載機制不熟悉,猛戳Android插件化開發(fā)動態(tài)加載基礎之ClassLoader工作機制 http://blog.csdn.net/u011068702/article/details/53248960
二、介紹

         我們知道在Android中可以跟java一樣實現(xiàn)動態(tài)加載jar,但是Android使用德海Dalvik VM,不能直接加載java打包jar的byte code,需要通過dx工具來優(yōu)化Dalvik bytecode。
         Android在API中給出可動態(tài)加載的有:DexClassLoader 和 PathClassLoader(上面連接已經(jīng)詳細介紹)
         DexClassLoader:可加載jar、apk和dex,可以從SD卡中加載(這篇博客采用這種方式)

 

         PathClassLoader:只能加載已經(jīng)安裝搭配Android系統(tǒng)中的apk文件

 
三、曝Demo照片,不要怕,不多,很簡單
 


四、編寫接口文件

    package com.example.testclassloader;
     
    public interface ShowString {
        public String sayChenyu();
    }


五、編寫接口實現(xiàn)文件

    package com.example.testclassloader;
     
    import android.util.Log;
     
    public class ShowStringClass implements ShowString{
     
        public static final String TAG = "ShowStringClass";
        
        @Override
        public String sayChenyu() {
            String chenyu = "chenyu";
            Log.i(TAG, chenyu);
            return chenyu;
        }
     
    }

 
六、打包成jar文件編譯成dex
我們把ShowStringClass.java文件打包生成showStringClass.jar文件,然后把文件放到sdk目錄下的build-tools下的23.0.1目錄下,我用的是ubuntu,所以會看到dex文件,如果是window會在這個目錄下看到dex.bat文件,然后用下面命令把showStringClass.jar生成showStringClass_imle.jar的dex文件

dx --dex --output=showStringClass_impl.jar showStringClass.jar

然后再把showStringClass_impl.jar文件放到手機目錄里面去用這個命令

adb push showStringClass_impl.jar  /sdcard/

具體操作圖片如下

 


七、然后編寫MainActivity.java文件

    package com.example.testclassloader;
     
    import java.io.File;
     
    import android.content.Context;
    import android.os.Bundle;
    import android.os.Environment;
    import android.support.v7.app.ActionBarActivity;
    import android.util.Log;
    import android.widget.TextView;
    import dalvik.system.DexClassLoader;
     
    public class MainActivity extends ActionBarActivity {
     
        public static final String TAG = "MainActivityClassLoader";
        public static final String SHOWSTRINGCLASS = "showStringClass_impl.jar";
        public static final String SHOWSTRINGCLASS_PATH= "com.example.testclassloader.ShowStringClass";
        public static final String DEX = "dex";
        public ShowStringClass mShowStringClass = null;
        public TextView mTv =  null;
        public int i = 0;
        
        @Override
        protected void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            setContentView(R.layout.activity_main);
            mTv = (TextView)findViewById(R.id.hello);
            DexClassLoader(this);
        }
        
        /**
         * 使用DexClassLoader方式加載類
         */
        public void  DexClassLoader(Context context) {
            // dex壓縮文件的路徑(可以是apk,jar,zip格式)
            String dexPath = Environment.getExternalStorageDirectory().toString() + File.separator + SHOWSTRINGCLASS;
     
            // dex解壓釋放后的目錄
            String dexOutputDirs = Environment.getExternalStorageDirectory().toString();
     
            //指定dexoutputpath為APP自己的緩存目錄
            File dexOutputDir = context.getDir(DEX, 0);
           
            // 定義DexClassLoader
            // 第一個參數(shù):是dex壓縮文件的路徑
            // 第二個參數(shù):是dex解壓縮后存放的目錄
            // 第三個參數(shù):是C/C++依賴的本地庫文件目錄,可以為null
            // 第四個參數(shù):是上一級的類加載器
            //DexClassLoader dexClassLoader = new DexClassLoader(dexPath,dexOutputDirs,null,getClassLoader());
            DexClassLoader dexClassLoader = new DexClassLoader(dexPath,dexOutputDir.getAbsolutePath(),null,getClassLoader());
     
            Class libProvierClazz = null;
            // 使用DexClassLoader加載類
            try {
                libProvierClazz = dexClassLoader.loadClass(SHOWSTRINGCLASS_PATH);
                // 創(chuàng)建dynamic實例
                mShowStringClass = (ShowStringClass) libProvierClazz.newInstance();
                if (mShowStringClass != null) {
                    final String chenyu = mShowStringClass.sayChenyu();
                    if (chenyu != null) {
                        mTv.post(new Runnable() {
                    @Override
                    public void run() {
                        mTv.setText(chenyu);
                    }
                        });
                    }
                } else {
                    Log.d(TAG, "mShowStringClass is null");
                }
            } catch (Exception e) {
                e.printStackTrace();
            }
        }
        
        /**
         * 打印系統(tǒng)的classLoader
         */
        public void showClassLoader() {
            ClassLoader classLoader = getClassLoader();
            if (classLoader != null){
                Log.i(TAG, "[onCreate] classLoader " + i + " : " + classLoader.toString());
                while (classLoader.getParent()!=null){
                    classLoader = classLoader.getParent();
                    Log.i(TAG,"[onCreate] classLoader " + i + " : " + classLoader.toString());
                    i++;
                }
            }
        }
    }

    @Override
                public void run() {
                    mTv.setText(chenyu);
                }
                    });
                }
            } else {
                Log.d(TAG, "mShowStringClass is null");
            }
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
    
    /**
     * 打印系統(tǒng)的classLoader
     */
    public void showClassLoader() {
        ClassLoader classLoader = getClassLoader();
        if (classLoader != null){
            Log.i(TAG, "[onCreate] classLoader " + i + " : " + classLoader.toString());
            while (classLoader.getParent()!=null){
                classLoader = classLoader.getParent();
                Log.i(TAG,"[onCreate] classLoader " + i + " : " + classLoader.toString());
                i++;
            }
        }
    }
}

 
八、運行Demo的結(jié)果爆照
在ubuntu終端打印結(jié)果如下


手機上面照片如下


說明加載外部的文件加載成功了
 
如果把上面那行代碼改成這個

DexClassLoader dexClassLoader = new DexClassLoader(dexPath,dexOutputDirs,null,getClassLoader());

會報下面的錯誤


需要加上緩存Dex文件的目錄

          //指定dexoutputpath為APP自己的緩存目錄
            File dexOutputDir = context.getDir(DEX, 0);

 
九、總結(jié)
1、加深動態(tài)加載的理解
2、如何實現(xiàn)項目加載外部的Dex文件有了更好的理解
3、對DexClassLoader 、dexClassLoader.load(package.class)、 class.newInstance() 有了更好的理解
 


 

 
作者:chen.yu
深信服三年半工作經(jīng)驗,目前就職游戲廠商,希望能和大家交流和學習,
微信公眾號:編程入門到禿頭 或掃描下面二維碼
零基礎入門進階人工智能(鏈接)