占位式插件化框架—Activity通信

1、我們想要在主APK中啟動(dòng)沒有安裝的插件p.apk的PluginActivity應(yīng)該怎么做呢?
2、插件p.apk中PluginActivity怎么啟動(dòng)同是插件包中的TestActivity?

\color{red}{項(xiàng)目主目錄}\


\color{red}{標(biāo)準(zhǔn)stander模塊}\
定義一個(gè)接口

import android.app.Activity;
import android.os.Bundle;
public interface ActivityInterface {
    /**
     * 把宿主(app)的環(huán)境  給  插件
     *
     * @param appActivity
     */
    void insertAppContext(Activity appActivity);
    void onCreate(Bundle savedInstanceState);
    void onStart();
    void onRestart();
    void onResume();
    void onDestroy();
    void onPause();
    void onStop();
}

\color{red}{插件plugin\_package}\

注意:

  • 1、因?yàn)椴寮嗀PK是沒有安裝到手機(jī)上谓媒,所以是無法擁有組件環(huán)境的。
    因?yàn)闆]有組件環(huán)境何乎,所以在插件中句惯,就不能使用this。
  • 2支救、所有關(guān)于操作抢野,組件環(huán)境的地方,都必須使用宿主的環(huán)境各墨。
  • 3指孤、在插件的ActivityAndroidManifest.xml里面不用配置 Activity。


public class BaseActivity extends Activity implements ActivityInterface {

    public Activity appActivity;

    @Override
    public void insertAppContext(Activity appActivity) {
        this.appActivity = appActivity;
    }
    ........
    public void setContentView(int resId) {
        appActivity.setContentView(resId);
    }
    public View findViewById(int layout) {
        return appActivity.findViewById(layout);
    }
    @Override
    public void startActivity(Intent intent) {
        Intent intentNew = new Intent();
        //className是包名加類名--com.migill.plugin_package.TestActivity
        intentNew.putExtra("className", intent.getComponent().getClassName());
        appActivity.startActivity(intentNew);
    }
    
}

BaseActivity中的setContentView()贬堵、 findViewById()恃轩、startActivity()方法的實(shí)現(xiàn)都是調(diào)用的宿主appActivity的方法。

public class PluginActivity extends BaseActivity  {
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_plugin);//執(zhí)行的是BaseActivity中的setContentView方法
        Log.e("migill","我是插件PluginActivity");
        findViewById(R.id.bt_start_activity).setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
                startActivity(new Intent(appActivity, TestActivity.class));
            }
        });
    }
}

PluginActivity中的setContentView()黎做、 findViewById()叉跛、startActivity()都是調(diào)用的
BaseActivity中的方法。

public class TestActivity extends BaseActivity {

    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_test);
        Log.e("migill","我是插件TestActivity");
        Toast.makeText(appActivity, "我是插件TestActivity", Toast.LENGTH_LONG).show();
    }
}

TestActivity中的setContentView()調(diào)用的是BaseActivity中的方法蒸殿。

把插件編譯出來的apk重新命名為p.apk筷厘。
放入 /storage/emulated/0/Android/data/com.migill.pluginproject/files/這個(gè)目錄下。


\color{red}{宿主}\

public class PluginManager {
    private static PluginManager pluginManager;
    private Context context;
    public static PluginManager getInstance(Context context) {
        if (pluginManager == null) {
            synchronized (PluginManager.class) {
                if (pluginManager == null) {
                    pluginManager = new PluginManager(context);
                }
            }
        }
        return pluginManager;
    }
    public PluginManager(Context context) {
        this.context = context;
    }
    private DexClassLoader dexClassLoader;
    private Resources resources;
    public void loadPlugin() {
        try {
            File file = new File(context.getExternalFilesDir(null).getAbsolutePath() + File.separator + "p.apk");
            if (!file.exists()) {
                Log.e("migill", "插件包宏所,不存在......");
                return;
            }
            String pluginPaht = file.getAbsolutePath();

            //1酥艳、現(xiàn)加載插件里面的 class
            //dexClassLoader需要一個(gè)緩存目錄 /data/data/當(dāng)前應(yīng)用的包名/pDir
            File fileDir = context.getDir("pDir", Context.MODE_PRIVATE);
            Log.e("migill", "pluginPaht : " + pluginPaht);
            Log.e("migill", "fileDir : " + fileDir.getAbsolutePath());
            //Activity class
            dexClassLoader = new DexClassLoader(pluginPaht, fileDir.getAbsolutePath(), null, context.getClassLoader());

            //2、加載插件里面的layout
            //加載資源
            AssetManager assetManager = AssetManager.class.newInstance();
            // 我們要執(zhí)行此方法楣铁,為了把插件包的路徑添加進(jìn)去 public final int addAssetPath(String path) path是路徑
            Method addAssetPathMethod = assetManager.getClass().getMethod("addAssetPath", String.class);
            addAssetPathMethod.invoke(assetManager, pluginPaht);//插件包的路徑 pluginPaht
            Resources r = context.getResources(); //宿主的資源配置信息
            //特殊的 Resource , 加載插件里面的資源的玖雁,Resources
            resources = new Resources(assetManager, r.getDisplayMetrics(), r.getConfiguration());
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
    public ClassLoader getClassLoader() {
        return dexClassLoader;
    }
    public Resources getResources() {
        return resources;
    }
}

public class MainActivity extends AppCompatActivity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
    }
    public void loadPlugin(View view) {
        PluginManager.getInstance(this).loadPlugin();
    }

    public void startPluginActivity(View view) {
        File file = new File(this.getExternalFilesDir(null).getAbsolutePath() + File.separator + "p.apk");
        String path = file.getAbsolutePath();
        Log.e("migill", "MainActivity path:" + path);
        //獲取插件包里面的Activity
        PackageManager packageManager = this.getPackageManager();
        PackageInfo packageInfo = packageManager.getPackageArchiveInfo(path, PackageManager.GET_ACTIVITIES);
        ActivityInfo activityInfo = packageInfo.activities[0];
        Log.e("migill", "MainActivity activityInfoName:" + activityInfo.name);
        //占位代理Activity
        Intent intent = new Intent(this, ProxyActivity.class);
        intent.putExtra("className", activityInfo.name);
        startActivity(intent);
    }
}
public class ProxyActivity extends Activity {
    @Override
    public Resources getResources() {
        return PluginManager.getInstance(this).getResources();
    }
    @Override
    public ClassLoader getClassLoader() {
        return PluginManager.getInstance(this).getClassLoader();
    }
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        String className = getIntent().getStringExtra("className");
        Log.e("migill", "ProxyActivity onCreate() " + className);
        try {
            Class mPlaginActivityClass = getClassLoader().loadClass(className);
            //實(shí)例化 插件包里面的 Activity
            Constructor constructor = mPlaginActivityClass.getConstructor(new Class[]{});
            Object mPluginActivity = constructor.newInstance(new Object[]{});
            ActivityInterface activityInterface = (ActivityInterface) mPluginActivity;
            //注入宿主
            activityInterface.insertAppContext(this);
            Bundle bundle = new Bundle();
            bundle.putString("appName", "我是宿主傳遞過來的信息");
            //執(zhí)行插件里面的onCreate(bundle)
            activityInterface.onCreate(bundle);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
    @Override
    public void startActivity(Intent intent) {
        String className = intent.getStringExtra("className");
        Log.e("migill", "ProxyActivity startActivity " + className);
        Intent proxyIntent = new Intent(this, ProxyActivity.class);
        proxyIntent.putExtra("className", className);
        super.startActivity(proxyIntent);
    }
}

ProxyActivity中的onCreate()主要就是根據(jù)類全名實(shí)例化對象更扁,在強(qiáng)轉(zhuǎn)成ActivityInterface類型對象activityInterface盖腕,activityInterface在這個(gè)對象中注入宿主赫冬,activityInterface在執(zhí)行的onCreate()。
ProxyActivity中的startActivity()方法是重新創(chuàng)建一個(gè)ProxyActivity頁面溃列,接著就會(huì)執(zhí)行onCreate()方法劲厌。
插件activity之間實(shí)現(xiàn)跳轉(zhuǎn)的時(shí)候最終是通過調(diào)用插件中的BaseActivity中的startActivity()方法,這個(gè)方法又調(diào)用了宿主的appActivity.startActivity(intentNew)听隐,最終調(diào)用ProxyActivity中的startActivity()方法补鼻。
看到這里,我們開頭問的兩個(gè)問題就都解決了雅任,那么我們在思考一個(gè)問題风范,為什么要有代理的Activity?
那是因?yàn)椴寮械腁ctivity,并不是一個(gè)能夠運(yùn)行的組件沪么,所以需要代理的Activity去代替插件中的Activity,例如硼婿,activity的任務(wù)進(jìn)棧。


?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末禽车,一起剝皮案震驚了整個(gè)濱河市寇漫,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌殉摔,老刑警劉巖州胳,帶你破解...
    沈念sama閱讀 206,839評論 6 482
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異逸月,居然都是意外死亡栓撞,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,543評論 2 382
  • 文/潘曉璐 我一進(jìn)店門彻采,熙熙樓的掌柜王于貴愁眉苦臉地迎上來腐缤,“玉大人,你說我怎么就攤上這事肛响×朐粒” “怎么了?”我有些...
    開封第一講書人閱讀 153,116評論 0 344
  • 文/不壞的土叔 我叫張陵特笋,是天一觀的道長剃浇。 經(jīng)常有香客問我,道長猎物,這世上最難降的妖魔是什么虎囚? 我笑而不...
    開封第一講書人閱讀 55,371評論 1 279
  • 正文 為了忘掉前任,我火速辦了婚禮蔫磨,結(jié)果婚禮上淘讥,老公的妹妹穿的比我還像新娘。我一直安慰自己堤如,他們只是感情好蒲列,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,384評論 5 374
  • 文/花漫 我一把揭開白布窒朋。 她就那樣靜靜地躺著,像睡著了一般蝗岖。 火紅的嫁衣襯著肌膚如雪侥猩。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,111評論 1 285
  • 那天抵赢,我揣著相機(jī)與錄音欺劳,去河邊找鬼。 笑死铅鲤,一個(gè)胖子當(dāng)著我的面吹牛划提,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播邢享,決...
    沈念sama閱讀 38,416評論 3 400
  • 文/蒼蘭香墨 我猛地睜開眼腔剂,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了驼仪?” 一聲冷哼從身側(cè)響起掸犬,我...
    開封第一講書人閱讀 37,053評論 0 259
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎绪爸,沒想到半個(gè)月后湾碎,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 43,558評論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡奠货,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,007評論 2 325
  • 正文 我和宋清朗相戀三年介褥,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片递惋。...
    茶點(diǎn)故事閱讀 38,117評論 1 334
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡柔滔,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出萍虽,到底是詐尸還是另有隱情睛廊,我是刑警寧澤,帶...
    沈念sama閱讀 33,756評論 4 324
  • 正文 年R本政府宣布杉编,位于F島的核電站超全,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏邓馒。R本人自食惡果不足惜嘶朱,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,324評論 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望光酣。 院中可真熱鬧疏遏,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,315評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至宝当,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間胆萧,已是汗流浹背庆揩。 一陣腳步聲響...
    開封第一講書人閱讀 31,539評論 1 262
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留跌穗,地道東北人订晌。 一個(gè)月前我還...
    沈念sama閱讀 45,578評論 2 355
  • 正文 我出身青樓,卻偏偏與公主長得像蚌吸,于是被迫代替她去往敵國和親锈拨。 傳聞我的和親對象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,877評論 2 345

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