Android组件化:stitch框架

之前的文章【Android组件化开发框架】从整体上分析了要搭建一个组件化框架的技术原理。概括性的对组件化进行了简单的分析。

stitch 0.1 是我在项目实践过程中结合之前的理论进行完善后组织起来的框架。它完成了页面路由、组件生命周期、数据路由的基本功能,框架源码里也包含了搭建组件化可能需要的脚本示例。

框架包含3个部分:组件生命周期管理、页面交互、数据交互。我们依次对其进行解析。

组件生命周期管理

每个组件相当于1个Module,许多组件的业务是需要在App启动时进行初始化的,比如运营商支付sdk基本都需要在Application的onCreate方法中进行初始化。

stitch 0.1 框架采用手动配置的方式,组件将自己注入到stitch 0.1 框架中,主工程再通过stitch 0.1框架对组件生命周期进行统一管理。

具体使用方法:

1. 继承ComponentApplication
public abstract class ComponentApplication {
    //Application对象注入
    public void setApplication(Application application);

    //控制组件的初始化顺序,参看ComponentPriority
    public int level();

    //代理Application的OnCreate方法
    public void onCreate();

    //延迟初始化生命周期,用来注入页面以及数据交互接口
    public void onCreateDelay(ComponentRouterRegistry routerRegistry, ActivityRouterRegistry activityRouterRegistry);

    //代理Application的attachBaseContext方法
    public void attachBaseContext(Context baseContext) ;

    //代理Application的onTrimMemory方法
    public void onTrimMemory(int level) ;

    //代理Application的onConfigurationChanged方法
    public void onConfigurationChanged(Configuration newConfig);

    //代理Application的onLowMemory方法
    public void onLowMemory();
}

ComponentApplication是组件生命周期的代理类,代理了Application的常用关键方法。如果组件在App启动时进行某些初始化或需要监听生命周期,通过ComponentApplication即可实现。

2. 在Module的AndroidManifest.xml中进行配置
    <application>
        ...
        <meta-data
            android:name="bamboo.sample.account.component.AccountComponentApp"
            android:value="ComponentApplication" />
    </application>

特别要注意:meta-data的value是ComponentApplication,name才是我们module的代理Application类。不要搞反了。

3. 主工程里面的自定义Application修改

在主工程Application里面我们需要主动调用组件的代理Application的方法,stitch 0.1提供了两种方式:

1. 直接继承stitchApplication

public class MainApplication extends stitchApplication {
}

2. 通过StitcherHelper调用组件的生命周期。
参考stitchApplication。

public class stitchApplication{
    public void onCreate() {
        super.onCreate();
        StitcherHelper.onCreate();
    }

    public void onCreateDelay() {
        StitcherHelper.onCreateDelay();
    }

    public void attachBaseContext(Context baseContext) {
        super.attachBaseContext(baseContext);
        StitcherHelper.init(this);
        StitcherHelper.attachBaseContext(baseContext);
    }

    public void onTrimMemory(int level) {
        super.onTrimMemory(level);
    }

    public void onLowMemory() {
        super.onLowMemory();
        StitcherHelper.onLowMemory();
    }

    public void onConfigurationChanged(Configuration newConfig) {
        super.onConfigurationChanged(newConfig);
        StitcherHelper.onConfigurationChanged(newConfig);
    }
}

通过上面几步,就能对组件的生命周期进行管理。

在使用页面交互以及数据交互之前我们先思考一个问题,这个问题是:

组件之间不能互相依赖,那如果要获取数据,我们就必须要借助于第三方Module进行中转。我们要怎么设计这个Module更方便呢?

没错,我们确实需要一个额外的Module来做交互中转,也就是路由Module,如果以开发人员的角度来看,这个Module应该要具备以下几个因素:
1.方便Module管理自己的接口。
2.其他Module最好能实时看到我们的修改。

所以在介绍交互功能之前,我们需要先来实现这个路由Module的配置。

1. 创建一个Module:sampleouter
2. 修改sampleouter的build.gradle文件,在里面加入以下代码:

这一步的目的是为了把所有Module的projectDir的router文件夹都加入到sampleouter的源码文件夹中,这样一来,我们就能在自己的Module的router文件夹内管理自己对外公开的接口以及页面。

android {
    ...
    sourceSets {
        main {
            //将所有module里的router文件夹都作为路由module的源码文件夹,方便Module开发时的方便
            ArrayList<String> strings = new ArrayList<String>()
            File[] modules = rootDir.listFiles(new FileFilter() {
                boolean accept(File pathname) {
                    return (pathname.isDirectory()
                            && pathname.name != "gradle"
                            && pathname.name != "build"
                            && !pathname.name.startsWith("."))
                }
            })
            for (File f : modules) {
                strings.add(f.absolutePath + File.separator + "router")
            }
            //不要忘了把原始的源码目录添加进来
            strings.addAll(java.srcDirs)
            java.srcDirs = strings
        }
    }
}

配置完后,在Module里面创建router文件夹你会看到这样的效果。


Module中的router文件夹

OK,sampleouter Module也配置好了,现在我们继续看页面交互要怎么实现。

页面交互

我们在讲述组件的生命周期管理时,可能你已经看到了组件生命周期代理类内有一个方法

    //延迟初始化生命周期,用来注入页面以及数据交互接口
    public void onCreateDelay(ComponentRouterRegistry routerRegistry, ActivityRouterRegistry activityRouterRegistry);

其中ActivityRouterRegistry就是我们进行页面交互的注册表,我们只需要将我们需要公开的页面注入到这里面就可以进行交互了。具体的实现步骤:

1. 在router文件夹里创建一个TaskInfoPage.class并继承ActivityPage
package bamboo.sample.tasksrouter;

//每个对外公开的页面都对应一个ActivityPage的子类
public class TaskInfoPage extends ActivityPage {
    public final String taskId;
    public TaskInfoPage(Context context, String taskId) {
        super(context);
        this.taskId = taskId;
    }
}
2. 创建一个TasksPageConsumer.class
public class TasksPageConsumer {
    //这个方法将会与TaskInfoPage进行连接。
    //所有的TaskInfoPage的页面交互请求都会最终调用到该方法中。
    public void consume(TaskInfoPage page) {
        Intent intent = new Intent(page.context, TaskInfoActivity.class);
        intent.putExtra("TaskInfoPage", page);
        page.context.startActivity(intent);
    }
}
3. 在Module的ComponentApplication实现类中注册
public class TasksComponentApp extends ComponentApplication {

    public void onCreateDelay(ComponentRouterRegistry routerRegistry, ActivityRouterRegistry activityRouterRegistry) {
        //将TasksPageConsumer注入到页面路由注册表中
        activityRouterRegistry.regiest(new TasksPageConsumer());
    }
4. 交互调用
//在需要调用TaskInfoPage的地方通过StitcherHelper使用
StitcherHelper.start(new TaskInfoPage(this, "taskId"));
StitcherHelper.start(new TaskInfoPage(this, "taskId"),1000/*requestCode*/);
5. 更简单的使用方式(PageConsumer注解)

实际上我们还有简单的方式进行页面注入,为什么我要先说常规模式呢? 因为如果万一简单的方式没办法满足你的需求的时候,你依然需要使用常规方式进行开发。

我们在继承实现ActivityPage时,还可以通过PageConsumer直接配置Activity或Action来进行Activity<->ActivityPage的连接。

@PageConsumer(clasz = "bamboo.sample.tasks.ui.TaskCountActivity")
public class TaskListPage extends ActivityPage {
    public TaskListPage(Context context) {
        super(context);
    }
}

这种方式是不需要在TaskPageConsumer中注册的,框架会自动搜索。

6. 特殊Intent参数配置

在页面交互时我们有时会需要对Intent设置Flag,或需要通过Action或Data方式进行交互,这个时候我们可以通过ActivityPage的targetIntent进行传递,并暂时关闭ActivityPage的连接,stitch 0.1会在这种情况下放弃PageConsumer注解中class的参数链接,直接尝试启动Activity。

    public void onActionTest(View view) {
        ActionTestPage page = new ActionTestPage(this);
        Intent targetIntent = new Intent();
        targetIntent.setAction("bamboo.sample.actiontest");
        targetIntent.addCategory(Intent.CATEGORY_DEFAULT);
        page.setTargetIntent(targetIntent);
        page.setAutoLink(false);
        StitcherHelper.start(page);
    }

优先级:
TaskPageConsumer > unAutolink > PageConsumer

数据交互

组件之间的数据交互与页面交互原理是相似的。在onCreateDelay方法中的ComponentRouterRegistry就是数据交互的路由注册表,我们通过它来进行注册。
具体使用参看以下步骤:

1.在router文件夹定义Module的ComponentOutput
package bamboo.sample.tasksrouter;
public interface ITaskComponent extends ComponentOutput{
    int getTaskCount();
}
2. 在Module 实现该接口
public class TasksComponentOutput implements ITaskComponent {
    public int getTaskCount() {
        return 1000;
    }
3. 在onCreateDelay方法中进行注册
public class TasksComponentApp extends ComponentApplication {

    public void onCreateDelay(ComponentRouterRegistry routerRegistry, ActivityRouterRegistry activityRouterRegistry) {
        //将TasksComponentOutput注入到数据路由注册表中
        routerRegistry.regiest(registerComponentOutput,new TasksPageConsumer());
    }
}
4.使用
public class ComponentInput {
    public int getTaskCount() {
        ITaskComponent taskComponent = StitcherHelper.searchComponentOutput(ITaskComponent.class);
        return taskComponent == null ? -1 : taskComponent.getTaskCount();
    }
}

具体的数据交互流程入下图:

数据交互流程

到这里,stitch的使用方式就介绍完了。

还没来得及发到maven仓库,所以想要使用的请先暂时移步github进行查看

框架已升级为自动注册,详情请看Android组件化:stitch 自动注册

stitch 源码及示例

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

推荐阅读更多精彩内容