插樁式實現(xiàn)插件化(一)

插件化(一)

一、前言

  • 定義:將整個app拆分成多個模塊瞻凤,這些模塊包括一個宿主多個插件捞挥,每個模塊都是一個apk文件(組件化的每個模塊為lib文件),最終打包時煌抒,宿主apk插件apk分開打包芍躏。(組件化最后只有一個apk文件)
  • 插件化優(yōu)勢:
    1. 宿主和插件分開編譯,互不影響睛驳;
    2. 并發(fā)開發(fā)薄货,宿主和插件分開同時開發(fā)遥倦;
    3. 動態(tài)更新插件瞎抛;
    4. 按需求下載模塊插件
    5. 解決65535問題;
  • 插件化的難度
    1. 插件的四大組件生命周期不好管理伤提,因為插件并沒有安裝,所以沒有上下文,所有與生命周期有關(guān)的方法都不能直接調(diào)用;
    2. android 9.0@hide注解舞痰,導(dǎo)致這個方法對開發(fā)者不可見。
    3. 兼容性問題诀姚;
    4. 宿主和插件之間的通信問題响牛,因此必須制定一套標(biāo)準(zhǔn),讓宿主和插件都遵循這套標(biāo)準(zhǔn),從而解決通信問題和插件的生命周期的管理問題娃善。

二论衍、 實現(xiàn)思路分析

新建一個library專門用來定義標(biāo)準(zhǔn),然后讓宿主apk插件apk都依賴這個library聚磺,而從宿主apk插件apk都必須實現(xiàn)這套標(biāo)準(zhǔn)坯台,插件apk的生命周期就得到了管理。

    public interface PayInterfaceActivity {
    /**
     *通過這個方法將宿主activity的context傳遞給插件
     *該方法被宿主調(diào)用
     */        
    void attach(Activity proxyActivity);

    /**
     * 生命周期
     */
    void onCreate(Bundle savedInstanceState);

    void onStart();

    void onResume();

    void onPause();

    void onStop();
        
    void onRestart();    

    void onDestroy();

    void onSaveInstanceState(Bundle outState);

    boolean onTouchEvent(MotionEvent event);

    void onBackPressed();
}

宿主apk代碼實現(xiàn)

public class MainActivity extends AppCompatActivity {

    private static final String TAG = "MainActivity";
    private Button mBtn_load;
    private Button mBtn_jump;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        PluginManager.getInstance().setContext(this);
        mBtn_load = findViewById(R.id.btn_load);
        mBtn_jump = findViewById(R.id.btn_jump);
        setListener();
    }

    private void setListener() {
        mBtn_load.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                //加載插件瘫寝,先將插件復(fù)制到制定目錄
                loadPlugin();
            }
        });
        mBtn_jump.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                //從宿主activity跳轉(zhuǎn)到插件的activity
                //實際上就是從宿主的其他activity跳轉(zhuǎn)到自己的ProxyActivity蜒蕾,proxyActivity再去加載插件的activity
                Intent intent = new Intent(MainActivity.this, ProxyActivity.class);
                //activities[0]代表launcher的activity
                intent.putExtra("className", PluginManager.getInstance().getPackageInfo().activities[0].name);
                startActivity(intent);
            }
        });
    }


    private void loadPlugin() {
        //這個目錄下文件只能被自己的app訪問,其他的沒有區(qū)別
        File pluginDir = getDir("plugin", Context.MODE_PRIVATE);
        String name = "pluginb.apk";
        String filePath = new File(pluginDir, name).getAbsolutePath();
        File file = new File(filePath);
        if (file.exists()) {
            file.delete();
        }
        InputStream is = null;
        OutputStream os = null;
        try {
            File file1 = new File(Environment.getExternalStorageDirectory(), name);
            Log.i(TAG, "load: file1  = " + file1.getAbsolutePath());
            is = new FileInputStream(file1);
            os = new FileOutputStream(file);
            int len = 0;
            byte[] buffer = new byte[1024];
            while ((len = is.read(buffer)) != -1) {
                os.write(buffer, 0, len);
            }
            if (new File(filePath).exists()) {
                Toast.makeText(this, "加載成功", Toast.LENGTH_SHORT).show();
            }

            //文件復(fù)制成功焕阿,加載插件
            PluginManager.getInstance().loadPath(this);
        } catch (Exception e) {
            e.printStackTrace();
        } finally {
            try {
                if (is != null) {
                    is.close();
                }
                if (os != null) {
                    os.close();
                }
            } catch (IOException e) {
                e.printStackTrace();
            }
        }
    }
}

插件管理類

/**
 * description:插件管理類咪啡,專門用來管理插件的加載過程,
 */
public class PluginManager {

    private Context mContext;

    private PluginManager() {
    }

    private static final PluginManager instance = new PluginManager();

    public static PluginManager getInstance() {
        return instance;
    }


    private PackageInfo mPackageInfo;
    private DexClassLoader mDexClassLoader;
    private Resources mResources;


    //加載插件
    public void loadPath(Context context) {
        //獲取所有的activity
        //首先找到插件的位置
        String pluginbPath = getPluginPath(context);

        //獲取插件apk里面所有的activity
        PackageManager manager = context.getPackageManager();
        mPackageInfo = manager.getPackageArchiveInfo(pluginbPath, PackageManager.GET_ACTIVITIES);

        File dexOutFile = context.getDir("dex", Context.MODE_PRIVATE);
        /**
         * 實例化DexClassLoader和mResources對象
         * dexPath:一個apk文件的路徑
         * optimizedDirectory:緩存路徑暮屡,這是虛擬機(jī)做優(yōu)化用的
         * librarySearchPath:依賴的包撤摸,可以為空
         * ClassLoader
         */
        mDexClassLoader = new DexClassLoader(pluginbPath, dexOutFile.getAbsolutePath(), null, context.getClassLoader());
        try {
            //因為構(gòu)造方法被@hide注釋了,所以只能使用反射
            AssetManager assetManager = AssetManager.class.newInstance();
            Method addAssetPath = AssetManager.class.getMethod("addAssetPath", String.class);
            addAssetPath.invoke(assetManager, pluginbPath);
            mResources = new Resources(assetManager, context.getResources().getDisplayMetrics(), context.getResources().getConfiguration());

        } catch (Exception e) {
            e.printStackTrace();
        }

    }

    private String getPluginPath(Context context) {
        File pluginDir = context.getDir("plugin", Context.MODE_PRIVATE);
        String name = "pluginb.apk";
        return new File(pluginDir, name).getAbsolutePath();
    }

    public DexClassLoader getDexClassLoader() {
        return mDexClassLoader;
    }

    public Resources getResources() {
        return mResources;
    }

    public PackageInfo getPackageInfo() {
        return mPackageInfo;
    }

    public void setContext(Context context) {
        mContext = context;
    }
}

宿主中專門加載插件中的activity

/**
 * description:ProxyActivity專門用來加載淘票票的內(nèi)容褒纲,和一般的activity不一樣准夷,
 * 加載一個activity,需要知道兩個東西莺掠,xxxx.class,資源assert衫嵌,不僅僅是resource
 * 這里僅是為了理解插件化的原理,因此通過intent來傳遞class的類名
 */
public class ProxyActivity extends Activity {
    //需要加載的淘票票的activity的全類名
    private String className;
    private static final String TAG = "ProxyActivity";
    private PayInterfaceActivity mPayInterfaceActivity;

    @Override
    protected void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        className = getIntent().getStringExtra("className");
        Log.i(TAG, "onCreate: className = " + className);
        //拿到對應(yīng)的class文件(why?)彻秆,這里不能通過反射來獲取Class文件楔绞,因為插件app沒有安裝(為什么沒有安裝就不能獲取唇兑?)酒朵,
        //通過classLoader來獲取class文件
        try {
            //className--->Class--反射-->className對應(yīng)的activity對象--->調(diào)用activity的onCreate()方法實現(xiàn)activity的加載
            Class<?> activityClass = getClassLoader().loadClass(className);
            //實例化activity對象,就是插件apk的類扎附,具體到這里就是插件的mainActivity
            Constructor<?> constructor = activityClass.getConstructor();
            //這里的instance實際上就是activity蔫耽,要想讓這個activity現(xiàn)實出來,必須調(diào)用onCreate()方法
            //當(dāng)然可以通過反射調(diào)用onCreate()方法帕棉,但是反射有性能上的消耗
            Object instance = constructor.newInstance();
            //因為instance就是插件apk的MainActivity针肥,而插件apk的Activity實現(xiàn)了PayInterfaceActivity接口饼记,因此可以強轉(zhuǎn)
            // TODO: 2018/4/27 java.lang.ClassCastException: com.wvbx.alipayplugin.MainActivity cannot be cast to com.wvbx.paystandard.PayInterfaceActivity
            mPayInterfaceActivity = (PayInterfaceActivity) instance;
            mPayInterfaceActivity.attach(this);
            //如果宿主activity想給插件activity傳值的話香伴,就可以通過這個bundle實現(xiàn)
            Bundle bundle = new Bundle();
            mPayInterfaceActivity.onCreate(bundle);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }


    @Override
    public void startActivity(Intent intent) {
        //從插件的第一個activity跳轉(zhuǎn)到插件的第二個activity要做的事情
        String className = intent.getStringExtra("className");
        Intent intent1 = new Intent(this, ProxyActivity.class);
        intent1.putExtra("className", className);
        super.startActivity(intent1);
    }

    //因為ProxyActivity里面放的是插件的activity,
    //所以通過控制ProxyActivity的生命周期來控制插件的activity的生命周期
    //插件實現(xiàn)了PayInterfaceActivity接口具则,所以可以打到這個目的
    @Override
    protected void onStart() {
        super.onStart();
        mPayInterfaceActivity.onStart();
    }

    @Override
    protected void onPause() {
        super.onPause();
        mPayInterfaceActivity.onPause();
    }

    @Override
    protected void onDestroy() {
        super.onDestroy();
        mPayInterfaceActivity.onDestroy();
    }

    //專門用來加載插件的classLoader,通過插件
    @Override
    public ClassLoader getClassLoader() {
        return PluginManager.getInstance().getDexClassLoader();
    }

    @Override
    public Resources getResources() {
        return PluginManager.getInstance().getResources();
    }
}

插件activity的基類

/**
 * description:
 *         由于插件apk沒有安裝即纲,所以滅有上下文,所以所有需要上下文的方法都不能按照以前的方法使用博肋,
 *         需要將宿主的activity傳遞給插件低斋,通過宿主的上下文來完成插件里面與上下文有關(guān)的方法的調(diào)用
 */
public class BaseActivity extends Activity implements PayInterfaceActivity {

    //宿主傳遞過來的activity
    protected Activity that;

    @Override
    public void attach(Activity proxyActivity) {
        this.that = proxyActivity;
    }


    @Override
    public <T extends View> T findViewById(int id) {
        if (that != null) {
            return that.findViewById(id);
        } else {
            return super.findViewById(id);
        }
    }

    @Override
    public void setContentView(int layoutResID) {
        if (that != null) {
            that.setContentView(layoutResID);
        } else {
            super.setContentView(layoutResID);
        }
    }

    @Override
    public Intent getIntent() {
        if (that != null) {
            return that.getIntent();
        } else {
            return super.getIntent();
        }
    }

    @Override
    public ClassLoader getClassLoader() {
        if (that != null) {
            return that.getClassLoader();
        } else {
            return super.getClassLoader();
        }
    }

    @NonNull
    @Override
    public LayoutInflater getLayoutInflater() {
        if (that != null) {
            return that.getLayoutInflater();
        } else {
            return super.getLayoutInflater();
        }
    }

    @Override
    public ApplicationInfo getApplicationInfo() {
        if (that != null) {
            return that.getApplicationInfo();
        } else {
            return super.getApplicationInfo();
        }
    }

    @Override
    public Window getWindow() {
        if (that != null) {
            return that.getWindow();
        } else {
            return super.getWindow();
        }
    }


    @Override
    public void startActivity(Intent intent) {
        //插件里面的activity跳轉(zhuǎn)蜂厅,本質(zhì)上是宿主activity的跳轉(zhuǎn),即從proxyActivity1 跳轉(zhuǎn)到proxyActivity2
        //proxyActivity-->className
        Intent i = new Intent();
        //這里activity只能是standard模式
        i.putExtra("className",intent.getComponent().getClassName());
        //這里的that即proxyActivity
        that.startActivity(i);
    }

    @Override
    public void setContentView(View view) {
        if (that != null) {
            that.setContentView(view);
        } else {
            super.setContentView(view);
        }
    }

    @Override
    public void onCreate(Bundle savedInstanceState) {

    }

    @Override
    public void onStart() {

    }

    @Override
    public void onResume() {

    }

    @Override
    public void onPause() {

    }

    @Override
    public void onRestart() {

    }

    @Override
    public void onStop() {

    }

    @Override
    public void onDestroy() {

    }

    @Override
    public void onSaveInstanceState(Bundle outState) {

    }
}

繼承了BaseActivity的子activity

public class MainActivity extends BaseActivity {

    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        //這里不能在xml中寫onclick事件膊畴,因為沒有上下文掘猿,
        findViewById(R.id.main).setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                //這里不能用MainActivity.this,因為沒有上下文
                Toast.makeText(that, "加載成功", Toast.LENGTH_SHORT).show();
                that.startActivity(new Intent(that, SecondActivity.class));
            }
        });
    }
    
}
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市唇跨,隨后出現(xiàn)的幾起案子稠通,更是在濱河造成了極大的恐慌,老刑警劉巖买猖,帶你破解...
    沈念sama閱讀 218,036評論 6 506
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件改橘,死亡現(xiàn)場離奇詭異,居然都是意外死亡玉控,警方通過查閱死者的電腦和手機(jī)飞主,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,046評論 3 395
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來高诺,“玉大人碌识,你說我怎么就攤上這事±僚眩” “怎么了丸冕?”我有些...
    開封第一講書人閱讀 164,411評論 0 354
  • 文/不壞的土叔 我叫張陵,是天一觀的道長薛窥。 經(jīng)常有香客問我胖烛,道長,這世上最難降的妖魔是什么诅迷? 我笑而不...
    開封第一講書人閱讀 58,622評論 1 293
  • 正文 為了忘掉前任佩番,我火速辦了婚禮,結(jié)果婚禮上罢杉,老公的妹妹穿的比我還像新娘趟畏。我一直安慰自己,他們只是感情好滩租,可當(dāng)我...
    茶點故事閱讀 67,661評論 6 392
  • 文/花漫 我一把揭開白布赋秀。 她就那樣靜靜地躺著,像睡著了一般律想。 火紅的嫁衣襯著肌膚如雪猎莲。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,521評論 1 304
  • 那天技即,我揣著相機(jī)與錄音著洼,去河邊找鬼。 笑死,一個胖子當(dāng)著我的面吹牛身笤,可吹牛的內(nèi)容都是我干的豹悬。 我是一名探鬼主播,決...
    沈念sama閱讀 40,288評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼液荸,長吁一口氣:“原來是場噩夢啊……” “哼瞻佛!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起娇钱,我...
    開封第一講書人閱讀 39,200評論 0 276
  • 序言:老撾萬榮一對情侶失蹤涤久,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后忍弛,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體响迂,經(jīng)...
    沈念sama閱讀 45,644評論 1 314
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,837評論 3 336
  • 正文 我和宋清朗相戀三年细疚,在試婚紗的時候發(fā)現(xiàn)自己被綠了蔗彤。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 39,953評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡疯兼,死狀恐怖然遏,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情吧彪,我是刑警寧澤待侵,帶...
    沈念sama閱讀 35,673評論 5 346
  • 正文 年R本政府宣布,位于F島的核電站姨裸,受9級特大地震影響秧倾,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜傀缩,卻給世界環(huán)境...
    茶點故事閱讀 41,281評論 3 329
  • 文/蒙蒙 一那先、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧赡艰,春花似錦售淡、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,889評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至料身,卻和暖如春汤纸,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背惯驼。 一陣腳步聲響...
    開封第一講書人閱讀 33,011評論 1 269
  • 我被黑心中介騙來泰國打工蹲嚣, 沒想到剛下飛機(jī)就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人祟牲。 一個月前我還...
    沈念sama閱讀 48,119評論 3 370
  • 正文 我出身青樓隙畜,卻偏偏與公主長得像,于是被迫代替她去往敵國和親说贝。 傳聞我的和親對象是個殘疾皇子议惰,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 44,901評論 2 355

推薦閱讀更多精彩內(nèi)容