主要介紹鏈?zhǔn)骄幊淘砝恚约叭绾蝿?chuàng)建鏈?zhǔn)骄幊?/p>
編程范式
在介紹鏈?zhǔn)骄幊讨埃紫葋砹私庀率裁词蔷幊谭妒健?br> 編程范式是編程語言的一種分類潮秘,是指從事軟件工程的一類典型的編程風(fēng)格
常見的編程范式
常見的編程范式主要有以下幾種
面向過程編程(Process Oriented Programming,POP):屬于典型的程序流程思想易结,即按照一定的順序枕荞,按部就班的工作,特別適合解決線性的問題衬衬,其中過程化編程語言主要包含機(jī)器語言、C等支持過程化的語言
面向?qū)ο缶幊蹋∣bject Oriented Programming改橘,OOP):包含3個(gè)基本概念:封裝滋尉、繼承、多態(tài)飞主。通過類狮惜、方法、對(duì)象和消息傳遞碌识,其相關(guān)的語言包含Java碾篡、Objective-C等
面向切面編程(AOP):是函數(shù)式編程的一種衍生范型。利用AOP可以對(duì)業(yè)務(wù)邏輯的各個(gè)部分進(jìn)行分離筏餐,降低業(yè)務(wù)間的耦合度开泽,提升程序的可重用性。例如OC中的Method Swizzling魁瞪、消息轉(zhuǎn)發(fā)就是采用AOP的典型
函數(shù)式編程(FP):是一種結(jié)構(gòu)化編程穆律,即如何編寫程序的方法論。其核心思想就是將運(yùn)算過程分解成一系列可復(fù)用函數(shù)的調(diào)用导俘,其中函數(shù)是重中之重峦耘。也是比較火熱且推崇的一種編程范式。
響應(yīng)式編程:簡單理解為就是一點(diǎn)觸發(fā)旅薄,多點(diǎn)響應(yīng)辅髓,例如OC中的KVO、通知等少梁,觸發(fā)者只負(fù)責(zé)觸發(fā)洛口,不理會(huì)結(jié)果
鏈?zhǔn)骄幊?/strong>:運(yùn)用點(diǎn)語言將很多函數(shù)串聯(lián)起來,例如OC中的Masory和Swift中的Snapkit
POP凯沪、OOP绍弟、AOP優(yōu)劣對(duì)比
POP
- 優(yōu)點(diǎn)
流程化的編程,任務(wù)明確著洼,即在開發(fā)前就已經(jīng)明確了最終實(shí)現(xiàn)和最終效果
開發(fā)效率高樟遣,代碼短小精悍而叼,適合結(jié)合數(shù)據(jù)結(jié)構(gòu)來開發(fā)高效率的程序,例如算法等
流程明確豹悬,具體步驟清晰葵陵,便于節(jié)點(diǎn)分析
- 缺點(diǎn)
- 需要深入思考,耗費(fèi)精力
- 代碼重用性地瞻佛,基本是用于解決一種固定的問題脱篙,且不易擴(kuò)展,維護(hù)難度大
- 對(duì)于復(fù)雜業(yè)務(wù)伤柄,面向過程的模塊化難度高绊困,耦合度高
OOP
- 優(yōu)點(diǎn)
結(jié)構(gòu)清晰,不同類承擔(dān)不同的職責(zé)
封裝性适刀,將事務(wù)進(jìn)行抽象秤朗,便于流程中的行為分析、操作
易擴(kuò)展笔喉,代碼復(fù)用性高取视,可繼承、覆蓋
實(shí)現(xiàn)簡單常挚,維護(hù)相對(duì)簡單
- 缺點(diǎn)
在面向過程的基礎(chǔ)上高度抽象作谭,和底層代碼交互少,不適合底層開發(fā)和游戲開發(fā)
對(duì)于事務(wù)而言奄毡,本身是面向過程的折欠,過度的封裝會(huì)導(dǎo)致事務(wù)本身的復(fù)雜性提高
AOP
- 優(yōu)點(diǎn)
簡單、易用吼过、易擴(kuò)展
降低模塊間的耦合度
設(shè)計(jì)決定的遲邦定(即運(yùn)行時(shí)綁定)
提升代碼的復(fù)用性
- 缺點(diǎn)
增加額外的重復(fù)代碼怨酝,且緊耦合
每個(gè)業(yè)務(wù)邏輯都需要一個(gè)裝飾器實(shí)現(xiàn)或代理
使用麻煩,必須增加容器
綜上所述那先,三者是一個(gè)相互補(bǔ)充和完善的邏輯
POP是以功能為中心來思考和組織程序的农猬,注重功能的實(shí)現(xiàn)
OOP是以對(duì)象為中心,強(qiáng)調(diào)整體性售淡,注重封裝斤葱,代碼整潔且規(guī)范
AOP是以業(yè)務(wù)解耦為中心,解決OOP中業(yè)務(wù)間高度耦合的問題
函數(shù)式編程
函數(shù)式編程是一種結(jié)構(gòu)化編程揖闸,即如何編寫程序的方法論揍堕。其核心思想就是將運(yùn)算過程分解成一系列可復(fù)用函數(shù)的調(diào)用,其中函數(shù)是重中之重汤纸。也是比較火熱且推崇的一種編程范式衩茸。
簡單理解為就是函數(shù)和數(shù)據(jù)類型是一致的,也是可以作為函數(shù)的參數(shù)贮泞、返回值楞慈。例如OC幔烛、Swift中的map、filter囊蓝、reduce函數(shù)等饿悬,每個(gè)函數(shù)的處理結(jié)果給到下一個(gè)函數(shù),最后的結(jié)果由自身函數(shù)調(diào)出聚霜。
如下所示
計(jì)算: (1+2)*3/4
f1(a, b) = a + b
f2(c) = c * 3
f3(d) = d / 4
所以整個(gè)計(jì)算等價(jià)于
f(x) = f3( f2( f1(1, 2) ) )
對(duì)應(yīng)到OC中狡恬,其核心點(diǎn)就是Block,如下所示
@interface Test: NSObject
- (Test *(^)(NSString *str))handle;
@end
@implementation Test
- (Test *(^)(NSString *str))handle{
return ^(NSString *str){
return self;
};
}
@end
<!--調(diào)用-->
Test *t = [[Test alloc] init];
t.handle(@"1111").handle(@"22222");
鏈?zhǔn)骄幊?/h2>
鏈?zhǔn)骄幊淌呛瘮?shù)式編程的一種體現(xiàn)蝎宇。
鏈?zhǔn)骄幊痰闹行乃枷耄?strong>方法的返回值必須是方法的調(diào)用者
鏈?zhǔn)骄幊痰暮诵恼Z法:點(diǎn)語法 + Block
鏈?zhǔn)骄幊痰奶攸c(diǎn):使用點(diǎn)語法將對(duì)象的多個(gè)函數(shù)連起來調(diào)用
鏈?zhǔn)骄幊淌呛瘮?shù)式編程的一種體現(xiàn)蝎宇。
鏈?zhǔn)骄幊痰闹行乃枷耄?strong>方法的返回值必須是方法的調(diào)用者
鏈?zhǔn)骄幊痰暮诵恼Z法:點(diǎn)語法 + Block
鏈?zhǔn)骄幊痰奶攸c(diǎn):使用點(diǎn)語法將對(duì)象的多個(gè)函數(shù)連起來調(diào)用
首先說點(diǎn)語法弟劲,在OC中,我們常應(yīng)用于getter姥芥、setter方法兔乞,是一種特殊的語法糖,OC中是通過 [receiver message] 來調(diào)用方法的撇眯,所以getter报嵌、setter的點(diǎn)語法最終會(huì)調(diào)用對(duì)應(yīng)屬性的getter虱咧、setter方法
其次來說Block熊榛,在OC中,Block既是匿名函數(shù)腕巡,也是對(duì)象玄坦,具體的可參考iOS-底層原理 30:Block底層原理文章,里面有詳細(xì)的講解
最后回到我們的焦點(diǎn):鏈?zhǔn)骄幊袒娉粒覀円绾螌?shí)現(xiàn)呢煎楣?其實(shí)很簡單,只需要在返回值作相應(yīng)改動(dòng)即可车伞,如下所示
@interface Test: NSObject
- (Test *)a;
- (Test *)b;
- (Test *)c;
@end
<!--調(diào)用-->
Test *t = [[Test alloc] init];
t.a.b.c;
可是通過上面的例子發(fā)現(xiàn)择懂,點(diǎn)語法是有了,確實(shí)連起來另玖,但是并不能傳參呀困曙,此時(shí)就需要借助Block了,在OC中谦去,常用的傳值方式主要由代理慷丽、通知、Block等鳄哭,其中滿足點(diǎn)語法的就當(dāng)屬Block了要糊。其次回想函數(shù)式編程,當(dāng)返回值是帶參block的getter方法時(shí)就實(shí)現(xiàn)了參數(shù)的傳遞妆丘。如下所示
<!--.h文件-->
@interface Test : NSObject
@property (nonatomic, strong) Test *(^block1)(NSString *name);
@property (nonatomic, strong) Test *(^block2)(NSInteger age);
@property (nonatomic, strong) Test *(^block3)(void);
- (Test *(^)(void))handleData;
@end
<!--.m文件-->
@interface Test ()
@property (nonatomic, strong) NSString *name;
@property (nonatomic, assign) NSInteger age;
@end
@implementation Test
- (Test * _Nonnull (^)(NSString * _Nonnull))block1{
return ^(NSString *name){
self.name = name;
return self;
};
}
- (Test * _Nonnull (^)(NSInteger))block2{
return ^(NSInteger age){
self.age = age;
return self;
};
}
- (Test * _Nonnull (^)(void))block3{
return ^(void){
return self;
};
}
- (Test *(^)(void))handleData{
return ^(void){
NSLog(@"處理數(shù)據(jù)");
return self;
};
}
@end
<!--鏈?zhǔn)秸{(diào)用-->
Test *t = [[Test alloc] init];
t.block1(@"Tom").block2(3).block3().handleData();
所以思考實(shí)現(xiàn)鏈?zhǔn)骄幊坛恚彩侵鸩竭f進(jìn)的過程:方法調(diào)用 -> 方法通過點(diǎn)語法調(diào)用 -> 手寫getter方法 -> 點(diǎn)語法調(diào)用屬性 -> 實(shí)現(xiàn)點(diǎn)語法+block的鏈?zhǔn)骄幊?/strong>
通過屬性實(shí)現(xiàn)的鏈?zhǔn)骄幊叹志ⅲ趃etter方法中既完成了setter方法的賦值,也處理了邏輯關(guān)系珊膜,還能通過getter方法完成鏈?zhǔn)骄幊倘菸眨^一舉三得呀!