像Ruby一樣寫ObjC暴构,用block實(shí)現(xiàn)鏈?zhǔn)椒椒ㄕ{(diào)用

Github源碼

引言

一切要從我加入了Codewars網(wǎng)站跪呈,開始與世界各地的Coder們一同刷題開始說起。在Codewars中取逾,有許多題目是支持多種不同語言的耗绿,比如下面這一道題,把字符串中的所有單詞根據(jù)空格分割反轉(zhuǎn):

#You need to write a function that reverses the words in a given string. A word can also fit an empty string. If this is not clear enough, here are some examples:

reverse('Hello World') === 'World Hello'
reverse('Hi There.') === 'There. Hi'

#Happy coding!

當(dāng)我們在Codewars的OJ系統(tǒng)中通過這道題目的時(shí)候砾隅,可以看到所有答案中误阻,大家評分最高的答案,對應(yīng)上面這個(gè)題目晴埂,ObjC的最佳答案是

#import <Foundation/Foundation.h>
NSString* reverse(NSString* text) {
    NSArray *words = [text componentsSeparatedByString:@" "];
    NSArray *reversed = [[words reverseObjectEnumerator] allObjects];
    return [reversed componentsJoinedByString:@" "];
}

這是一段中規(guī)中矩的ObjC代碼究反,跟我的答案是一樣的。
然后我們來看看Ruby版本的最佳答案儒洛,雖然是同樣的解題思路精耐,但表現(xiàn)出來的視覺效果卻完全不同:

def reverse(string)
  string.split.reverse.join(' ')
end

這里給沒有接觸過Ruby的朋友解釋一下這段代碼,首先琅锻,Ruby的方法中可以省略return關(guān)鍵字卦停,把方法中最后一個(gè)對象返回;其次split方法不傳參數(shù)時(shí)候默認(rèn)是以空格符分割恼蓬,這樣就有了這段簡介明晰的代碼惊完。

當(dāng)然,ObjC也是可以一句話寫完這段代碼的嘛:

NSString* reverse(NSString* text) {
    return [[[[text componentsSeparatedByString:@" "] reverseObjectEnumerator] allObjects] componentsJoinedByString:@" "];
}

但你會發(fā)現(xiàn)处硬,這樣書寫的ObjC代碼可讀性大打折扣小槐,一方面ObjC的方法名太長,引起代碼折行以后郁油,很難一眼看清整個(gè)過程本股;另一方面,ObjC的消息傳遞機(jī)制使用的中括號嵌套桐腌,嵌套多層時(shí)會增加額外的匹配括號的工作拄显,有些時(shí)候甚是煩人。

思考

上面的對比案站,引發(fā)了我對ObjC的種種思考躬审,是否有可能使用ObjCRuby一樣優(yōu)雅地寫鏈?zhǔn)胶瘮?shù)調(diào)用,實(shí)際上Swift中就采用了類似Ruby的寫法蟆盐。

在ObjC中采用 . 調(diào)用方法

我們知道ObjC中承边,點(diǎn)是用來獲取屬性(Property)的,例如我們給AppDelegate聲明了一個(gè)datas的數(shù)組屬性

@interface AppDelegate ()
@property (nonatomic, strong) NSArray* datas;
@end

當(dāng)前的編譯器會自動(dòng)給datas生成settergetter方法石挂,并在沒有使用@synthesize關(guān)鍵字合成的前提下博助,聲明了_datas這個(gè)內(nèi)部指針。

這時(shí)我們?nèi)绻命c(diǎn)方法調(diào)用self.datas痹愚,等同于傳遞了[self datas]消息富岳,實(shí)際是發(fā)送的getter消息蛔糯。如此一來,我們可以用property或者不帶參數(shù)的方法窖式,來模擬點(diǎn)方法蚁飒,例如數(shù)組反轉(zhuǎn):

@interface NSArray (Functional)
- (NSArray*)reverse;
@property (nonatomic, strong, readonly) NSArray* reverse;
@end

我們給NSArray增加一個(gè)名為Funcional的Category,增加reverse方法或者property都可以萝喘,二選一即可淮逻。這里的property聲明為readonly,從而禁止調(diào)用setter方法阁簸。

這兩種方法都可以實(shí)現(xiàn)self.reverse爬早,實(shí)際消息發(fā)送都是[self reverse],實(shí)現(xiàn)如下:

@implementation NSArray (Functional)
- (NSArray *)reverse //reverse屬性的getter方法 和 - (NSArray*)reverse; 相同
{
    return [[self reverseObjectEnumerator] allObjects];
}
@end

使用block閉包

然而當(dāng)我們需要改寫帶參數(shù)的方法時(shí)强窖,兩種方式實(shí)現(xiàn)都有些問題了凸椿。比如例子中的數(shù)組拼接字符串的方法componentsSeparatedByString,我們這里需要用到ObjC閉包(block)特性翅溺。

下面兩種方式也是等價(jià)的脑漫,原理同上面的reverse

@interface NSArray (Functional)
@property (nonatomic, strong, readonly) NSString* (^join)(NSString* join);
- (NSString* (^)(NSString*))join;
@end

實(shí)現(xiàn)代碼:

- (NSString *(^)(NSString * j))join
{
    return ^ (NSString* j) {
        return [self componentsJoinedByString:j];
    };
}

另外我們給NSString也增加一個(gè)Category實(shí)現(xiàn)字符串切割成數(shù)組:


@interface NSString (Functional)
@property (nonatomic, strong, readonly) NSArray* (^split)(NSString* s);
@end

@implementation NSString (Functional)
-(NSArray* (^)(NSString *))split
{
    return ^ (NSString* s) {
        return [self componentsSeparatedByString:s];
    };
}
@end

如此,我們就可以通過點(diǎn)語法來實(shí)現(xiàn)鏈?zhǔn)秸{(diào)用咙崎,來實(shí)現(xiàn)開篇說的問題优幸。

NSString* reverse(NSString* text) {
    return text.split(@" ").reverse.join(@" ");
}

是不是有感覺在用Ruby的錯(cuò)覺。

擴(kuò)展褪猛,函數(shù)式編程

ObjC作為一個(gè)比C++還要遙遠(yuǎn)的語言网杆,在某些方面還是缺少現(xiàn)代編程語言的特性。例如數(shù)組的Map伊滋、Filter碳却、Flatten等高級函數(shù),Cocoa框架都是沒有的笑旺。

而這些函數(shù)實(shí)在是太常用也太好用了昼浦,我們完全可以通過前面討論的方式,為NSArray增加這些方便的高級函數(shù).

//定義block
typedef id (^MapBlock)(id x);
@property (nonatomic, strong, readonly) NSArray* (^map)(MapBlock);
//或者
- (NSArray *(^)(id (^)(id)))map;

實(shí)現(xiàn)如下:

- (instancetype)map:(id (^)(id))map
{
    NSMutableArray* array = [NSMutableArray array];
    [self enumerateObjectsUsingBlock:^(id  _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) {
        id x = map(obj);
        if(x) [array addObject:x];
    }];
    return [array copy];
}

- (NSArray *(^)(id (^handle)(id)))map
{
    return ^(id (^handle)(id)) {
        return [self map:handle];
    };
}

如此筒主,我們可以通過兩種方式來調(diào)用map方法关噪,這兩種方式是等價(jià)的,數(shù)組[1,2,3]通過map方法變成了[3,6,9]:

[@[@1,@2,@3] map:^id(id x) {
   return @([x integerValue] * 3);
}];
    
@[@1,@2,@3].map(^id(id x) {
   return @([x integerValue] * 3);
});    

小結(jié)

有人可能認(rèn)為乌妙,這些代碼并沒有太多的簡化我們的開發(fā)ObjC的方式使兔,但請不要忽視這些微小的積累。代碼的簡化和優(yōu)化藤韵,帶來的是更低的耦合虐沥、更好的可讀性和更健壯的構(gòu)架,我們用了十幾分鐘的時(shí)間來討論ObjC的鏈?zhǔn)胶瘮?shù)調(diào)用方法泽艘,必定會在以后的開發(fā)中置蜀,節(jié)省大量的重復(fù)勞動(dòng)時(shí)間奈搜。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末悉盆,一起剝皮案震驚了整個(gè)濱河市盯荤,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌焕盟,老刑警劉巖秋秤,帶你破解...
    沈念sama閱讀 206,126評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異脚翘,居然都是意外死亡灼卢,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,254評論 2 382
  • 文/潘曉璐 我一進(jìn)店門来农,熙熙樓的掌柜王于貴愁眉苦臉地迎上來鞋真,“玉大人,你說我怎么就攤上這事沃于∩В” “怎么了?”我有些...
    開封第一講書人閱讀 152,445評論 0 341
  • 文/不壞的土叔 我叫張陵繁莹,是天一觀的道長檩互。 經(jīng)常有香客問我,道長咨演,這世上最難降的妖魔是什么闸昨? 我笑而不...
    開封第一講書人閱讀 55,185評論 1 278
  • 正文 為了忘掉前任,我火速辦了婚禮薄风,結(jié)果婚禮上饵较,老公的妹妹穿的比我還像新娘。我一直安慰自己遭赂,他們只是感情好循诉,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,178評論 5 371
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著嵌牺,像睡著了一般打洼。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上逆粹,一...
    開封第一講書人閱讀 48,970評論 1 284
  • 那天募疮,我揣著相機(jī)與錄音,去河邊找鬼僻弹。 笑死阿浓,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的蹋绽。 我是一名探鬼主播芭毙,決...
    沈念sama閱讀 38,276評論 3 399
  • 文/蒼蘭香墨 我猛地睜開眼筋蓖,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了退敦?” 一聲冷哼從身側(cè)響起粘咖,我...
    開封第一講書人閱讀 36,927評論 0 259
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎侈百,沒想到半個(gè)月后瓮下,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 43,400評論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡钝域,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 35,883評論 2 323
  • 正文 我和宋清朗相戀三年讽坏,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片例证。...
    茶點(diǎn)故事閱讀 37,997評論 1 333
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡路呜,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出织咧,到底是詐尸還是另有隱情胀葱,我是刑警寧澤,帶...
    沈念sama閱讀 33,646評論 4 322
  • 正文 年R本政府宣布烦感,位于F島的核電站巡社,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏手趣。R本人自食惡果不足惜晌该,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,213評論 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望绿渣。 院中可真熱鬧朝群,春花似錦、人聲如沸中符。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,204評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽淀散。三九已至右莱,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間档插,已是汗流浹背慢蜓。 一陣腳步聲響...
    開封第一講書人閱讀 31,423評論 1 260
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留郭膛,地道東北人晨抡。 一個(gè)月前我還...
    沈念sama閱讀 45,423評論 2 352
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親耘柱。 傳聞我的和親對象是個(gè)殘疾皇子如捅,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,722評論 2 345

推薦閱讀更多精彩內(nèi)容