IOS组件化与工程管理

谈及组件化其实网上也有不少文章了,但我个人认为不结合工程管理去单讲组件化恐怕很难让人理解概念,而去实践的时候也只是照猫画虎。

工程管理

组件化的实现很重要的一个组成部分应该是工程拆分,这里我的方案是采取git管理项目pod管理依赖很常见很普通的方法。
理想的状态每一个模块都是独立的,可以单独拿出来测试,发布,也就是每一个子模块其实都是一个git仓库,这里紧接着就是子仓库和主项目的关系问题,上边说到了git和pod,还有一个submodule我喜欢用这个来做子仓库的管理上边为啥没提它呢?
因为submodule本身就是git自带的就是git的一部分,常用命令有

#添加子仓库
git submodule add 仓库地址 路径
#初始化所有子仓库
git submodule init
#更新子仓库
git submodule update

#也可以初始化更新一起
git submodule update --init

pod只是帮我把依赖关系理清直接本地pod,因为坑爹的开发阶段难免有互相block的情况,那边东西弄完了,但还没有做发布还不稳定,但另一边已经急着要看一眼整体调用的效果了。。。
当然这时候也可以让对面先打个beta的tag,那样可想而知最后会有多少没用的tag,另一方面就是bug联调恢复节点排查的时候,另一方估计只有一个方案就是回滚上一个tag中间哪的问题一点点打tag联调。
上边说的打tag也只是个例子,当然你可以改Podfile对应不同branch,但那也是要每次调都要改一下的,但我这种模式由于是本地pod所以podfile不用动了每次都去指向对应的项目,剩下的就是对子仓库随意切换branch甚至commit节点都可以,调ok了直接commit一下submodule指向的更改即可。然后另一方更新一下再pod update把依赖关系重新建立一下(如果没有添加或删除,甚至这步都不需要,本地pod引用目录我们submodule的本地文件夹里面有什么变化这边自动会变,添加删除是因为依赖关系发生变化了所以跟着需要重新建立),当然理想情况回头有空单独整合一套submodule和pod的命令,submodule更新时判断有增删操作执行pod update其他情况不处理。

这里紧接着就是公共库的处理以及怎么去建立主仓库与子仓库的依赖,我这里把基本思路给出,具体情况还是自己再改动,这里首先就是建立podspec来提供依赖建立

Pod::Spec.new do |s|
  s.name                  = 'MainWorkSpaceDemo'
  s.version               = '1.0.0'
  s.summary               = 'A new container controller to slide  '
  s.homepage              = 'github.com'
  s.license               = { :type => 'MIT', :file => 'README.md' }
  s.author                = { 'heroims' => 'heroims@163.com' }
  s.source                = { :git => '', :tag => "#{s.version}" }
  s.platform              = :ios, '5.0'
  s.source_files          = 'ZYQRouter/*.{h,m}'
  s.requires_arc          = true
  #公共仓库
  s.subspec 'BaseTool' do |ss|
    ss.source_files = 'ZYQRouter/*.{h,m}'
  end
  #模块1
  s.subspec 'Module1' do |sss|
    sss.source_files = 'Module1/Module1Lib/*.{h,m}'
  end
  #模块2
  s.subspec 'Module2' do |ssss|
    ssss.source_files = 'Module2/Module2Lib/*.{h,m}'
  end

end

看见上边相比就明白了把,开发的时候最好要作为模块给人的东西放在一个目录下,当然不放也可以,这里就是为了方便
然后就是引用了,下面是module1工程的,只引用了一个公共库,真正开发的时候则会引用很多,然后build测试模块的app给测试,提供给主仓库的东西放在事先约定的目录下,其他的随便看心情,反正对别人没影响就是自己爽不爽

target 'Module1' do
  # Uncomment the next line if you're using Swift or would like to use dynamic frameworks
  # use_frameworks!

  # Pods for Module1
  pod 'MainWorkSpaceDemo/BaseTool', :path => '../MainWorkSpaceDemo.podspec'

  target 'Module1Tests' do
    inherit! :search_paths
    # Pods for testing
  end

  target 'Module1UITests' do
    inherit! :search_paths
    # Pods for testing
  end

end

就此基本的项目依赖思路构建就算讲完了,剩下的就是调用了,其实上边的东西你掌握好了,组件化就已经可以单用这种工程管理模式解决了
下面的就是大家能经常搜到的一些组件化的东西,在我看来剩下的只是锦上添花,工程管理好组件化才有真正意义。从开发的角度上不论开发什么用什么语言,组件化或者说模块化通用思路就是分离多个仓库然后自动化建立依赖关系,项目工程里互相调用适当的用反射的方法实现调用,基本上每一个小仓库就都能独立的运行了。
顺便放一下我的ZYQRouter:https://github.com/heroims/ZYQRouter
里面的demo虽然没用submodule但基本可以阐述完整这套东西。
正式开始讲代码里的架构,ZYQRouter主要是方便各个模块之间的互相调用。之所以重复造轮子,其实只是自己项目需要,另外就是想完善这套组件化的实现。
这里ZYQRouter分为页面路由和方法路由,页面路由负责根据URL做各页面跳转甚至远程调度方法路由,方法路由则是提供target-action实例方法调用和invokeSelectorObjects反射调用静态方法,目的就是让各个模块开发过程中不引用对方的情况下也可独立按约定调用对方模块运行调试自己的相关内容,大家都开发完各个单元测试ok,集成到主项目里就可以基本跑通,当然现实是联调还是会通常出些小问题,但没什么大碍。

页面路由

关于页面路由如下,用过蘑菇街Router的看这个会很亲切,我只是在它的基础上添加了重定向这个功能,这重定向的由来一个是动态更新页面跳转逻辑方便,另一个就是我们自己的需求客服系统里。。。。你会发现一个订单链接地址由客服发来,网页上用这链接用户打开的就是网页自己订单,客服打开就是客服系统该用户的订单,app上用户打开就是用户订单页面,于是救星就是重定向,把xxx.xxx.xxx/crm-order/orderidxxx.xxx.xxx/order/orderid都重定向到applink://order/orderid,还有就是订单有大改动的时候
则是xxx.xxx.xxx/crm-order/orderidapplink://order/orderid重定向到xxx.xxx.xxx/order/orderid直接开网页用户订单,还有很多奇葩需求全靠重定向这救命稻草,所以这个重定向真的很实用。

顺便再说下注册的事,因为我的Router里提供了target-action的调用所以上面说的远程调度target-action可以用一个url如applink://target-action/:target/:action?xxx=xxx完成,只用注册applink://target-action/:target/:action内部调用target-action方法。
而让所有部门全依照这一个逻辑规则产出链接简直天方夜谭,前端放在网页上的链接按这样估计一堆人吐槽,但仅仅ios部门之间按照这一规则跑还是可以的。
但当然有比较折中的方法,毕竟注册太多url也占地啊,这时候神奇的重定向就又可以上线救援了,如xxx.xxx.xxx/order?xxx=xxx这类直接重定向xxx.xxx.xxx/orderapplink://target-action/ordertarget/orderaction这就好了,你注册的就可以少点但前提是你的target-action里处理的情况多。
另外写页面路由最好根据模块单独创建相应的类,比如Module1里可以单独的建个Module1PageFactory,有个方法-(void)openModule1VC1WithO1:(id)o1 o2:(id)o2 o3:(id)o3类似方法然后+(void)load里注册Router调用open的方法,这样开发阶段用方法路由,而在需要从外部进入时采用页面路由方式也就是URL方式

/**
 重定向 URLPattern 到对应的 newURLPattern 
 @param URLPattern 原scheme
 @param newURLPattern 新scheme
 */
+ (void)redirectURLPattern:(NSString *)URLPattern toURLPattern:(NSString*)newURLPattern;

/**
 *  注册 URLPattern 对应的 Handler,在 handler 中可以初始化 VC,然后对 VC 做各种操作
 *
 *  @param URLPattern 带上 scheme,如 applink://beauty/:id
 *  @param handler    该 block 会传一个字典,包含了注册的 URL 中对应的变量。
 *                    假如注册的 URL 为 applink://beauty/:id 那么,就会传一个 @{@"id": 4} 这样的字典过来
 */
+ (void)registerURLPattern:(NSString *)URLPattern toHandler:(ZYQRouterHandler)handler;

/**
 *  注册 URLPattern 对应的 ObjectHandler,需要返回一个 object 给调用方
 *
 *  @param URLPattern 带上 scheme,如 applink://beauty/:id
 *  @param handler    该 block 会传一个字典,包含了注册的 URL 中对应的变量。
 *                    假如注册的 URL 为 applink://beauty/:id 那么,就会传一个 @{@"id": 4} 这样的字典过来
 *                    自带的 key 为 @"url" 和 @"completion" (如果有的话)
 */
+ (void)registerURLPattern:(NSString *)URLPattern toObjectHandler:(ZYQRouterObjectHandler)handler;

/**
 *  取消注册某个 URL Pattern
 *
 *  @param URLPattern
 */
+ (void)deregisterURLPattern:(NSString *)URLPattern;

/**
 *  打开此 URL
 *  会在已注册的 URL -> Handler 中寻找,如果找到,则执行 Handler
 *
 *  @param URL 带 Scheme,如 applink://beauty/3
 */
+ (void)openURL:(NSString *)URL;

/**
 *  打开此 URL,同时当操作完成时,执行额外的代码
 *
 *  @param URL        带 Scheme 的 URL,如 applink://beauty/4
 *  @param completion URL 处理完成后的 callback,完成的判定跟具体的业务相关
 */
+ (void)openURL:(NSString *)URL completion:(void (^)(id result))completion;

/**
 *  打开此 URL,带上附加信息,同时当操作完成时,执行额外的代码
 *
 *  @param URL        带 Scheme 的 URL,如 applink://beauty/4
 *  @param parameters 附加参数
 *  @param completion URL 处理完成后的 callback,完成的判定跟具体的业务相关
 */
+ (void)openURL:(NSString *)URL withUserInfo:(NSDictionary *)userInfo completion:(void (^)(id result))completion;

/**
 * 查找谁对某个 URL 感兴趣,如果有的话,返回一个 object
 *
 *  @param URL
 */
+ (id)objectForURL:(NSString *)URL;

/**
 * 查找谁对某个 URL 感兴趣,如果有的话,返回一个 object
 *
 *  @param URL
 *  @param userInfo
 */
+ (id)objectForURL:(NSString *)URL withUserInfo:(NSDictionary *)userInfo;

/**
 *  是否可以打开URL
 *
 *  @param URL
 *
 *  @return
 */
+ (BOOL)canOpenURL:(NSString *)URL;

/**
 *  调用此方法来拼接 urlpattern 和 parameters
 *
 *  #define ROUTE_BEAUTY @"beauty/:id"
 *  [ZYQRouter generateURLWithPattern:ROUTE_BEAUTY, @[@13]];
 *
 *
 *  @param pattern    url pattern 比如 @"beauty/:id"
 *  @param parameters 一个数组,数量要跟 pattern 里的变量一致
 *
 *  @return
 */
+ (NSString *)generateURLWithPattern:(NSString *)pattern parameters:(NSArray *)parameters;

方法路由

关于方法路由如下,target-action模式就是自动根据class来alloc init初始化完target对象,然后@selector把那action方法调用了返回,而静态方法则是runtime搞定,日常需求基本满足,但还有点缺陷注释里已说明,由于invokeSelectorObjects根据className和selectorName调用静态方法所以封装成了C方法,另外就是这个不常用算是尝试。

/**
 *
 *  调度工程内的组件方法
 *  [ZYQRouter performTarget:@"xxxClass" action:@"xxxxActionWithObj1:obj2:obj3" objects:obj1,obj2,obj3,nil]
 *  内部自动 alloc init 初始化对象
 *
 *  @param targetName    执行方法的类
 *  @param actionName    方法名
 *  @param object1,... 不定参数 不支持C基本类型
 *
 *  @return 方法回参
 */
+ (id)performTarget:(NSString*)targetName action:(NSString*)actionName objects:(id)object1,...;

/**
 *
 *  调度工程内的组件方法
 *  [ZYQRouter performTarget:@"xxxClass" action:@"xxxxActionWithObj1:obj2:obj3" shouldCacheTaget:YES objects:obj1,obj2,obj3,nil]
 *  内部自动 alloc init 初始化对象
 *
 *  @param targetName    执行方法的类
 *  @param actionName    方法名
 *  @param shouldCacheTaget   设置target缓存
 *  @param object1,... 不定参数 不支持C基本类型
 *
 *  @return 方法回参
 */
+ (id)performTarget:(NSString*)targetName action:(NSString*)actionName shouldCacheTaget:(BOOL)shouldCacheTaget objects:(id)object1,...;

/**
 *
 *  调度工程内的组件方法
 *  [ZYQRouter performTarget:@"xxxClass" action:@"xxxxActionWithObj1:obj2:obj3" shouldCacheTaget:YES objects:obj1,obj2,obj3,nil]
 *  内部自动 alloc init 初始化对象
 *
 *  @param targetName    执行方法的类
 *  @param actionName    方法名
 *  @param shouldCacheTaget   设置target缓存
 *  @param objectsArr   参数数组 不支持C基本类型
 *
 *  @return 方法回参
 */
+ (id)performTarget:(NSString*)targetName action:(NSString*)actionName shouldCacheTaget:(BOOL)shouldCacheTaget objectsArr:(NSArray*)objectsArr;

/**
 *
 *  添加未找到Target 或 Action 逻辑
 *
 *  @param notFoundHandler    未找到方法回调
 *  @param targetName    类名
 *
 *  @return
 */
+ (void)addNotFoundHandler:(ZYQNotFoundTargetActionHandler)notFoundHandler targetName:(NSString*)targetName;

/**
 *  删除Target缓存
 *
 *  @return
 */
+ (void)removeTargetsCacheWithTargetName:(NSString*)targetName;
+ (void)removeTargetsCacheWithTargetNames:(NSArray*)targetNames;
+ (void)removeAllTargetsCache;

/**
 不定参静态方法调用 (最多支持7个,原因不定参方法传给不定参方法实在没啥好办法。。。。暂时如此)
 id result=(__bridge id)zyq_invokeSelectorObjects(@"Class", @"actionWithObj1:obj2:obj3",obj1,obj2,obj3,nil);
 
 c类型转换配合__bridge_transfer __bridge
 利用IMP返回值只是指针,不支持C基本类型
 
 @param className 类名
 @param selectorName,... 方法名,不定参数
 @return 返回值
 */
void * zyq_invokeSelectorObjects(NSString *className,NSString* selectorName,...);

最后就是页面路由和方法路由遇到找不到的处理方案了,主要思路就是不crash、好判断,页面路由就判断一下是网页的就跳转url不是就报个提示算了,方法路由return nil吧。。这里仁者见仁智者见智,反正可以自己定制,差不多就讲到这吧。

事件链路由

/**
 响应链传递路由
 
 用于解决多级嵌套UI对象的上级事件响应,省去delegate protocol逐级传递,跨级传递
 
 @param eventName 事件名
 @param userInfo 扩展信息
 */
-(void)zyq_routerEventWithName:(NSString *)eventName userInfo:(id)userInfo;

这个主要解决多层级UI对象嵌套的时候,事件传递繁琐,通过Event完成对事件定义,一级级传递到响应者的过程在开发中就可以省略了。
只需要如下使用的时候在发起和接受地方写好处理即可!

//调用
[self.nextResponder zyq_routerEventWithName:eventName userInfo:userInfo];

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

推荐阅读更多精彩内容