鏈式編程的介紹與Masonry
首先看一下OC下使用優(yōu)秀第三方框架Masonry在實現(xiàn)一個自動布局時候的實現(xiàn)代碼:
[self.demoTextField mas_makeConstraints:^(MASConstraintMaker *make) {
make.top.equalTo(self.view).and.offset(100);
make.width.equalTo(self.view);
make.height.equalTo(@44);
}];
其中在對y位置添加約束的時候,使用了連續(xù)的.語法。這就是鏈式編程。
BabyBluetooth
Masonry中的鏈式可能相對比較零散,并不能體現(xiàn)出鏈式的任務(wù)邏輯連貫性。
下面介紹另外一個優(yōu)秀的第三方框架BabyBluetooth,BabyBluetooh是簡單易用的藍牙庫,基于CoreBluetooth的封裝,并兼容ios和mac osx。
CoreBluetooth所有方法都是通過委托完成,代碼冗余且順序凌亂。BabyBluetooth使用block方法,可以重新按照功能和順序組織代碼,并使用鏈式編程將一組任務(wù)用一條鏈完成。
baby.having(self.currPeripheral).and.channel(channelOnPeropheralView).then.connectToPeripherals().discoverServices().discoverCharacteristics().readValueForCharacteristic().discoverDescriptorsForCharacteristic().readValueForDescriptors().begin();
用一條鏈就完成了一整套的連接peripheral,發(fā)現(xiàn)服務(wù),發(fā)現(xiàn)特性,讀取特性值,發(fā)現(xiàn)描述,讀取描述值的任務(wù)鏈,而不需要多次的分散調(diào)用,流程邏輯非常清晰。
開工
說了辣么多,那么到底怎么實現(xiàn)呢?
首先
我們都知道self.aString其中的aString是個屬性,類型看樣子應(yīng)該是個正經(jīng)的NSString類型.
那么self.aString()是個什么寫法呢,我們把aString換個名字self.functionPointer()。
好了,現(xiàn)在從名字可以看的出來這個functionPointer貌似是個函數(shù)指針,那么self.functionPointer+()也就代表調(diào)用函數(shù),如果這個函數(shù)有參數(shù)那么完整的調(diào)用應(yīng)該是這樣的self.functionPointer(@"testString");
。但是這樣只能一次調(diào)用,并沒有實現(xiàn)鏈式的效果。
別急,假如我們的函數(shù)指針指向的函數(shù)是有返回類型的呢?并且,返回的類型就是self的類型。那么,self.functionPointer(@"testString")執(zhí)行完成后的結(jié)果就是self
,再用這個結(jié)果通過函數(shù)指針去調(diào)函數(shù),好了,鏈出來了。
self.functionPointer(@"firstString").functionPointer(@"secondString").functionPointer(@"thirdString");
大概的寫法是這樣的:
JCChainStyleManager.h文件
#import <Foundation/Foundation.h>
@class JCChainStyleManager;
typedef JCChainStyleManager* (*CLikeFunction) (NSString *);
@interface JCChainStyleManager : NSObject
#pragma mark - 用函數(shù)指針實現(xiàn)的鏈式
- (CLikeFunction)dotMessageWithCFunction;
@end
JCChainStyleManager.m 文件
@implementation JCChainStyleManager
- (CLikeFunction)dotMessageWithCFunction{
return testFunction;
}
JCChainStyleManager *testFunction(NSString *aString){
NSLog(@"this is a C like function and log: %@",aString);
return [JCChainStyleManager new];};
@end
調(diào)用處
JCChainStyleManager *chainManager = [JCChainStyleManager new];
chainManager.dotMessageWithCFunction(@"6666").dotMessageWithCFunction(@"999");
感覺用函數(shù)指針的方式來解釋更加的易懂,但是在實際使用的時候c函數(shù)并不能捕獲到到同一個self對象,每次函數(shù)執(zhí)行完成要完成鏈式都要重新創(chuàng)建一個self對象,除非將self做成單例的模式。
Block
介于使用函數(shù)指針的方式實在太逆天,而且需要要到c函數(shù),所以下面切換到常規(guī)的block實現(xiàn)的方式。
JCChainStyleManager.h文件
@class JCChainStyleManager;
typedef JCChainStyleManager *(^JCChainVoidBlock)();
typedef JCChainStyleManager *(^JCChainStringBlock)(NSString *);
@interface JCChainStyleManager : NSObject
#pragma mark - 用block實現(xiàn)的鏈式
- (JCChainVoidBlock)begin;
- (JCChainVoidBlock)firstBlock;
- (JCChainStringBlock)secondBlock;
@end
JCChainStyleManager.m 文件
@implementation JCChainStyleManager
- (JCChainVoidBlock)begin{
return ^JCChainStyleManager *() {
NSLog(@"begin");
return self;
};
}
- (JCChainVoidBlock)firstBlock{
return ^JCChainStyleManager *() {
NSLog(@"聽局座唱rap??,看諸葛琴魔");
return self;
};
}
- (JCChainStringBlock)secondBlock{
return ^JCChainStyleManager *(NSString * aString) {
NSLog(@"%@",aString);
return self;
};
}
@end
調(diào)用處
JCChainStyleManager *chainManager = [JCChainStyleManager new];
chainManager.firstBlock().secondBlock(@"asd").begin();
當(dāng)執(zhí)行chainManager.firstBlock的時候,實際是使用了getter
方法,并獲取到了類型為JCChainVoidBlock的塊,然后使用chainManager.firstBlock+()的方式執(zhí)行了代碼塊,并且返回了self,以調(diào)用下個鏈節(jié)點。