组件化细节

中间件作用:

1、负责转发信息。
2、用runtime反射调用,让中间件解除对各个组件的依赖,同时又能调到各个组件暴露出来的方法。
总结:组件通过中间件通信,中间件通过 runtime 接口解耦。

实现:

1、利用JLRoutes,保存一个全局的map,key是url,value是对应存放的block数组,url和block都会常驻在内存中,当打开一个URL时,JLRoutes就可以遍历 , 这个全局的map,通过url来执行对应的block。
2、我们首先写一个AppDelegate+JLRouter的category拓展。在这里实现路由的注册。

- (void)addRouter { 
    [JLRoutes setVerboseLoggingEnabled:YES];
    //跳转指定界面
    [[JLRoutes globalRoutes] addRoute:Router_PushViewController handler:^BOOL(NSDictionary * _Nonnull parameters) {
        NSString *controller = parameters[@"controller"];
        UIViewController *currentVc = [self currentViewController];
        NSString *title = parameters[@"title"];
        UIViewController *vc = [[NSClassFromString(controller) alloc] init];
        [self paramToVc:vc param:parameters];
        vc.hidesBottomBarWhenPushed = YES;
        if (title) {
            vc.title = parameters[@"title"];
        }
        [currentVc.navigationController pushViewController:vc animated:YES];
        return YES;
    }];  
}

3、当注册完路由,就可以不用import,或者依赖,完成跳转。
4、然后运用runtime完成消息的转发。

/*! runtime 赋值 */
- (void)paramToVc:(UIViewController *) v param:(NSDictionary *)parameters {
    //runtime将参数传递至需要跳转的控制器
    unsigned int outCount = 0;
    objc_property_t * properties = class_copyPropertyList(v.class , &outCount);
    for (int i = 0; i < outCount; i++) {
        objc_property_t property = properties[i];
        NSString *key = [NSString stringWithUTF8String:property_getName(property)];
        NSString *param = parameters[key];
        if (param != nil) {
            [v setValue:param forKey:key];
        }
    }
}

5、以上可看到,我们不在需要去import需要跳转的模块。也不需要知道跳转模块接收什么类型的数据。开发者只需要知道跳转模块叫什么。就可以完成跳转。 至此我们完成了中间件的实现。

模块设计原则

越底层的模块,应该越稳定,越抽象,越具有高复用度。

不要让稳定的模块依赖不稳定的模块, 减少依赖

提升模块的复用度,自完备性有时候要优于代码复用

每个模块只做好一件事情,不要让Common出现

按照你架构的层数从上到下依赖,不要出现下层模块依赖上层模块的现象,
业务模块之间也尽量不要耦合

你可能感兴趣的:(组件化细节)