项目地址:STMAssembleView
1. 结构设计简洁
- 只公开最外层的STMAssembleView类,调用者只需引入一个类即可使用。
- 将业务与功能解耦,生成对应的类:“View”视图类(STMAssembleView、STMPartView)与“Maker”类(STMAssemvleMaker、STMPartMaker)。“Maker”类负责抽象出支持的功能属性,“View”类通过使用Maker,将子功能进行组合并实现最终业务功能(视图配置)。
2. 代码设计巧妙
- 对于“Maker”类,属性用于提供给调用者进行配置调用(即只使用getter),而属性的setter则另提供配套的设置方法,且设置方法返回的是对应“Maker”类型的Block对象,以实现链式调用。例如:
// STMAssembleMaker.m
- (STMAssembleMaker *(^)(STMAssembleAlignment))alignmentEqualTo {
return ^STMAssembleMaker *(STMAssembleAlignment alignment) {
// jiji - 设置对齐方式
self.alignment = alignment;
return self;
};
}
...
// 调用时
make.alignmentEqualTo(STMAssembleAlignmentCenter);
如代码所示:
由于方法alignmentEqualTo返回的是Block对象,调用方法时,通过添加()即直接隐含执行了Block。且由于Block执行后返回的仍然为make自身,可以在后面继续连续调用其他实例方法,实现了链式调用。
- 使用__weak指针,防止Block对象导致的内存引用循环。如:
// STMAssembleMaker.m
+ (void)fsAsync:(NSString *)string objects:(NSDictionary *)objs completion:(ParsingFormatStringCompleteBlock)completeBlock{
__weak __typeof(string) weakString = string; // 弱指针,防止Block执行时对该对象强引用
__weak __typeof(objs) weakObjs = objs;
dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_LOW, 0), ^{
__strong __typeof(weakString) strongString = weakString; // 声明临时的__strong指针并指向__weak指向的对象,可以保证在Block执行过程中该对象不被释放。
__strong __typeof(weakObjs) strongObjs = weakObjs;
if (strongString) {
[STMAssembleView createViewWithFormatString:strongString objects:strongObjs completion:completeBlock];
}
});
}
如代码所示:
使用__weak和__strong指针的情况在Block声明时具有统一性,可以很好地避免引用循环的产生,且能保证在Block执行时捕获对象的有效性。
注:在单独使用GCD时,由于dispatch_async的Block会在执行后进行release,理论上可以不必担心循环引用的问题。这里铭神这样做的目的是保证自身不会保留传入的对象,需要调用者自己负责参数的生命周期。
3. 知识点:
3.1 字符串的扫描处理
使用了NSScanner类对字符串进行扫描,按需将扫描结果进行归类,完成描述字符串的解析(后面根据解析结果调用对应的“View”类进行处理,生成页面)。
3.2 使用变参合成字符串
使用va_list指向变参列表的第一个参数地址,通过NSString合成出完整字符串。
//宏
NSString *ASS(NSString *format, ...) {
// jiji - 声明变参指针
va_list args;
if (format) {
// jiji - 初始化变参指针(指向第一个参数的地址,如这里的format的地址)
va_start(args, format);
// jiji - NSString直接通过变参指针读取并生成字符串
NSString *str = [[NSString alloc] initWithFormat:format arguments:args];
// jiji - 清除变参指针对象
va_end(args);
return str;
}
return @"";
}