筆記: iOS 鏈式編程

鏈式編程的介紹與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é)點。

@End

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容