组件化之路之接口隔离

在描述我们解决组件化关于接口隔离方法之前,先抛出两个问题:

  1. 大家如何拆分非业务模块的模块,比如用户信息模块,vip信息模块等非业务模块。
  2.大家对于接口和模型的理解,如何访问接口的属性,以及如何实现KVO监听接口属性变化(比如我们经常使用的RAC操作)。
  3.如何管理那些protocol,怎么在外部调用那些实现接口的类的方法以及属性。

一、关于模块化的拆分

  目前在我们APP里面,实际上是集合了连个APP,一个是学生版,一个是教师版。学生端的功能 是纯Native的代码,教师端的是RN的代码,但是被集成我们的APP中。教师功能有一些模块是和学生端是共用的,比如用户信息,埋点系统,设置页面等等。
  由于在之后的版本迭代里面,教师端的功能,会从我们的主APP中迁移出去,但是有些功能在教师端还能用,我们我们必须要实现组件化。下面是我们组件化的简单结构图,今天主要是针对关于user,vip这两个模块的抽取进行讲解。大家主要看框图上黄色的位置。


image.png

下图是我们根据上面的拆分方式,加上我们的对业务的拆分之后,整理出来的结构。这些代码目前在pod里面,很大家做组件化一样,我们使用的也是podspec,具体我就不写了。


image.png
二、关于接口的设计

通常情况下,我们在APP中会有一个userManager这个单例去管理我们整个APP中用户信息变化。然后在userManager实现对应的增删改查,属性访问等操作。在做组件化之后,我们有一个userManagerProtocol这个接口包含所有原来userManger相关属性和方法,为了能够让大家在调用用户信息的时候感觉没有太大的区别。因为是公司代码,不方便透露,所有我就写了一个简单的demo,来解释我们的实现。这是改造之前userManger头文件。

@class YCUser;
@interface YCUserManager : NSObject

@property (nonatomic, assign) BOOL isStudent;

- (NSDictionary *)retunUserInfo;

- (NSString *)returnToken;

// 更新
- (void)updateUserInfo:(NSDictionary *)userInfo;

- (void)updateUserToken:(NSString *)token;

- (YCUser *)returnUser;

@end

改造之后,userManger头文件,是的什么都没有了,

@interface YCUserManager : NSObject

@end

但是我们多了一个userManagerProtocol,为了能够快速访问到用户属性,我们还定义了一个userProtocol,userProtocol这个接口里面的属性和user里面的属性保持一致。目的是为了实现在外部和访问类的姿势保持一致。 头文件如下:

@protocol YCUserManagerProtocal <NSObject, BHServiceProtocol>

@property (nonatomic, assign) BOOL isStudent;

- (NSDictionary *)retunUserInfo;

- (NSString *)returnToken;

// 更新
- (void)updateUserInfo:(NSDictionary *)userInfo;

- (void)updateUserToken:(NSString *)token;

- (id<YCUserProtocol>)returnUser;

@end

对于userManger实现类如下:

#import "YCUserManager.h"
#import "YCUserManagerProtocal.h"
#import "YCUser.h"
#import <MJExtension.h>
#import <BeeHive.h>

@BeeHiveService(YCUserManagerProtocal, YCUserManager)

@interface YCUserManager() <YCUserManagerProtocal>

@property (nonatomic, strong) YCUser *studentUser;

@property (nonatomic, strong) NSDictionary *userDict;

@property (nonatomic, copy) NSString *token;

@end

@implementation YCUserManager

@synthesize  isStudent = _isStudent;

+ (BOOL)singleton {
    return YES;
}

static YCUserManager *_manager = nil;
+ (instancetype)shareInstance {
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        _manager = [[YCUserManager alloc] init];
    });
    return _manager;
}

- (void)updateUserInfo:(NSDictionary *)userInfo {
    
    self.userDict = userInfo;
    
    YCUser *user = [YCUser mj_objectWithKeyValues:userInfo];
    self.studentUser = user;
    
    self.isStudent = YES;
}

- (void)updateUserToken:(NSString *)token {
    self.token = token;
}

- (NSString *)returnToken {
    return self.token;
}

- (NSDictionary *)retunUserInfo {
    return self.userDict;
}

- (id<YCUserProtocol>)returnUser {
    return (id)self.studentUser;
}

@end

大家在上面的文件中会看到#import <BeeHive.h> ,这是阿里出的一个框架,BeeHive是基于Spring的Service理念,虽然可以使模块间的具体实现与接口解耦,但无法避免模块对接口类的依赖关系。在我们的模块化中主要用这个类实现模块与接口解耦的功能。在使用这个类的时候,为了能够找个这个类,需要注册一下,注册service姿势有三种,具体关于BeeHive的更过功能以及使用姿势可以看 [https://github.com/alibaba/BeeHive]

1. 通过BeeHiveService宏进行Annotation标记。
    @BeeHiveService(YCUserManagerProtocal, YCUserManager)   

2. 读取本地Pilst文件
   [BHContext shareInstance].serviceConfigName = @"BeeHive.bundle/BHService";

3.Load方法注册 
   [[BeeHive shareInstance] registerService:@protocol(YCUserManagerProtocal) service:[YCUserManager class]];

三、关于接口管理以及使用

@interface ViewController ()

@end

@implementation ViewController

- (void)viewDidLoad {
    [super viewDidLoad];
    
    NSDictionary *dict = @{
                           @"name":@"周文超",
                           @"age":@(12),
                           @"student" : @{
                                   @"studentName" : @"李向红",
                                   @"studentAge" : @(11)
                                   }
                           };
    
    id<YCUserManagerProtocal> userManager = [[BeeHive shareInstance] createService:@protocol(YCUserManagerProtocal)];
    
    // 更新数据
    [userManager updateUserInfo:dict];
    
    [RACObserve(userManager, isStudent) subscribeNext:^(id x) {
        
    }];
    
    // 获取用户信息
    id<YCUserProtocol> user = [userManager returnUser];
    
    [userManager updateUserToken:@"我是token"];
    
    NSLog(@"%@----%@-----%@", user.name, [userManager returnToken], [user.student mj_JSONObject]);
    
    [RACObserve(user, name) subscribeNext:^(id  _Nullable x) {

    }];
    
}

先看代码,对着代码进行讲解。

  id<YCUserManagerProtocal> userManager = [[BeeHive shareInstance] createService:@protocol(YCUserManagerProtocal)];

在BeeHive中是通过BHServiceManager来管理各个Protocol的。BHServiceManager中只会管理已经被注册过的Protocol。上面我们对YCUserManagerProtocal进行了注册。因此我们可以通过这种方式获取到我们注册的service。
大家可能疑惑为什么,对于一个接口的属性可以实现KVO操作,这也是OC的特性。protocol 的本质类似一个抽象类,这个声明了一些纯虚方法或者属性。在java中,这个叫接口类在编码中,通过继承协议,实现了协议中描述的这一套方法或者方法赋值操作。我们在接口里面定义了一个isStudent属性,在实现类里面,我们做了

@synthesize  isStudent = _isStudent;

所以我们可以实现对它的监听,同样,我们通过

   // 获取user
   id<YCUserProtocol> user = [userManager returnUser];

   [RACObserve(userManager, isStudent) subscribeNext:^(id x) {

    }];
  
  // 监听user属性变化
   [RACObserve(user, name) subscribeNext:^(id  _Nullable x) {

    }];
注意:大家需要注意的几点
  1. 在使用BeeHive的时候一定要注册
  2. 如果要想某个服务是单利,需要重写下面这个方法
+ (BOOL)singleton {
    return YES;
}

3.对于实现接口的类,大家在做模型转json的时候,会附带几个protocol的四个属性,hash, superclass, debugDescription, description

博客表述的不是很清楚,我吧Demo放在github上,大家有什么不懂的可以直接看代码。

[https://github.com/zwcshy/ModulaInterface.git]

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

推荐阅读更多精彩内容

  • 1、通过CocoaPods安装项目名称项目信息 AFNetworking网络请求组件 FMDB本地数据库组件 SD...
    阳明先生_x阅读 15,967评论 3 119
  • 该文章属于<简书 — 刘小壮>原创,转载请注明: <简书 — 刘小壮> http://www.jianshu.co...
    Yiart阅读 4,579评论 3 49
  • 今天点出来了马克思主义历史唯物主义和辨证唯物主义的东西,这些和中国现有的现实里面都有依据,可以和中国古代道家思想联...
    智囊团阅读 209评论 0 0
  • 2018年5月千日,日拱一卒,积跬步,精进未来的自己。 “老敢”,东北话,不懂行,没见过世面。 今天陪着公公婆婆外...
    凡尘花仙子阅读 1,158评论 1 0