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

1、我们想要在主APK中启动没有安装的插件p.apk的PluginActivity应该怎么做呢?
2、插件p.apk中PluginActivity怎么启动同是插件包中的TestActivity?

\color{red}{项目主目录}\


\color{red}{标准stander模块}\
定义一个接口

import android.app.Activity;
import android.os.Bundle;
public interface ActivityInterface {
    /**
     * 把宿主(app)的环境  给  插件
     *
     * @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、因为插件APK是没有安装到手机上,所以是无法拥有组件环境的。
    因为没有组件环境,所以在插件中,就不能使用this。
  • 2、所有关于操作,组件环境的地方,都必须使用宿主的环境。
  • 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()方法的实现都是调用的宿主appActivity的方法。

public class PluginActivity extends BaseActivity  {
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_plugin);//执行的是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()都是调用的
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()调用的是BaseActivity中的方法。

把插件编译出来的apk重新命名为p.apk。
放入 /storage/emulated/0/Android/data/com.migill.pluginproject/files/这个目录下。


\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、现加载插件里面的 class
            //dexClassLoader需要一个缓存目录 /data/data/当前应用的包名/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();
            // 我们要执行此方法,为了把插件包的路径添加进去 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);
            //实例化 插件包里面的 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", "我是宿主传递过来的信息");
            //执行插件里面的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()主要就是根据类全名实例化对象,在强转成ActivityInterface类型对象activityInterface,activityInterface在这个对象中注入宿主,activityInterface在执行的onCreate()。
ProxyActivity中的startActivity()方法是重新创建一个ProxyActivity页面,接着就会执行onCreate()方法。
插件activity之间实现跳转的时候最终是通过调用插件中的BaseActivity中的startActivity()方法,这个方法又调用了宿主的appActivity.startActivity(intentNew),最终调用ProxyActivity中的startActivity()方法。
看到这里,我们开头问的两个问题就都解决了,那么我们在思考一个问题,为什么要有代理的Activity?
那是因为插件中的Activity,并不是一个能够运行的组件,所以需要代理的Activity去代替插件中的Activity,例如,activity的任务进栈。


©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,921评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,635评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,393评论 0 338
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,836评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,833评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,685评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,043评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,694评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 42,671评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,670评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,779评论 1 332
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,424评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,027评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,984评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,214评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,108评论 2 351
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,517评论 2 343

推荐阅读更多精彩内容