幾種編程思想在iOS中的實(shí)現(xiàn):(一)鏈?zhǔn)骄幊?/h1>

前言

作為一個(gè)iOS程序員基本上都應(yīng)該接觸過(guò)Masonry這個(gè)自動(dòng)布局庫(kù)倘待。這個(gè)庫(kù)能夠幫助程序員極大程度的簡(jiǎn)化自動(dòng)布局的代碼拣技。使用這個(gè)庫(kù)讓我感到驚嘆的不是如何能夠?qū)⑤^為復(fù)雜的傳統(tǒng)自動(dòng)布局寫(xiě)法精簡(jiǎn)到如此程度批狐,而是精簡(jiǎn)后的代碼的書(shū)寫(xiě)方式:

make.left.right.bottom.mas_equalTo(0.f);

這種寫(xiě)法在做到簡(jiǎn)化的同時(shí)瓮下,通過(guò)點(diǎn)(.)調(diào)用的方式记劈,將代碼連接成一行盖腕,大大增加了代碼的可讀性茄菊,這就是本篇要提到的鏈?zhǔn)骄幊獭?/p>

關(guān)于鏈?zhǔn)骄幊叹唧w是種什么編程思想,這種概念性的東西赊堪,請(qǐng)自行百度面殖,這里不多做介紹,本篇主要是通過(guò)一個(gè)簡(jiǎn)單的例子來(lái)實(shí)現(xiàn)上述的鏈?zhǔn)骄幊獭?/p>

舉個(gè)例子

日常開(kāi)發(fā)中肯定少不了要與數(shù)據(jù)庫(kù)打交道哭廉,而數(shù)據(jù)庫(kù)相信不少項(xiàng)目是使用的splite3以及將其封裝的很好的FMDB脊僚。如果在開(kāi)發(fā)中直接使用FMDB,那么免不了要手動(dòng)寫(xiě)SQL語(yǔ)句遵绰。當(dāng)然現(xiàn)在網(wǎng)上也有不少的DAO辽幌,可以讓開(kāi)發(fā)者無(wú)需關(guān)心怎么去寫(xiě)SQL,甚至不需要懂SQL的語(yǔ)法椿访,通過(guò)對(duì)對(duì)象的操作就可以達(dá)到在數(shù)據(jù)庫(kù)的增刪改查乌企。

這里我們不討論這種DAO是怎么實(shí)現(xiàn)的,我們實(shí)現(xiàn)的是通過(guò)一個(gè)工具類(lèi)來(lái)幫助我們生成對(duì)應(yīng)的SQL成玫。免去我們手寫(xiě)“SELECT” “WHERE” 等關(guān)鍵字麻煩加酵,而且可以智能提示,避免寫(xiě)錯(cuò)哭当。

#不要問(wèn)為什么有好好的DAO不去用猪腕。這只是個(gè)例子,不要太在意細(xì)節(jié)钦勘。

分析

既然要智能提示陋葡,免去手動(dòng)書(shū)寫(xiě)“SELECT”等詞的話,很明顯是將其抽成一個(gè)方法供調(diào)用者調(diào)用彻采,其內(nèi)部將關(guān)鍵字拼接好腐缤。那么這個(gè)方法應(yīng)該是如下類(lèi)似的:

- (NSString *)select:(NSArray *)columns
                from:(NSString *)tableName;

但是,SELECT的情況不止這一種肛响,還有FROM岭粤,JOIN ON,GROUP BY终惑,ORDER BY等各種操作绍在,而且是可選的,那么就需要增加以下的一系列類(lèi)似方法:

- (NSString *)select:(NSArray *)columns
                from:(NSString *)tableName
               where:(NSString *)where;
               
- (NSString *)select:(NSArray *)columns
                from:(NSString *)tableName
               where:(NSString *)where
             orderBy:(NSString *)orderBy;
...

很明顯,要為每一種組合情況創(chuàng)建一個(gè)方法偿渡,這樣做顯得不太明智臼寄。因?yàn)椋哼@里的例子,SQL查詢(xún)語(yǔ)句的組合還算是有限的溜宽,目前確實(shí)可以窮舉吉拳,把每種可能性列出來(lái),添加相應(yīng)的方法适揉,但是留攒,要是哪天SQL支持ORDER BY可以寫(xiě)到前面去,那不就呵呵了嫉嘀,又要在增加對(duì)應(yīng)組合情況的方法炼邀。顯然這么做是不明智的。

那么就換一個(gè)方向剪侮,將SELECT等每一步操作拆分成一個(gè)個(gè)單獨(dú)的方法拭宁,這樣后續(xù)需要接WHERE條件還是直接ORDER BY都可以根據(jù)具體業(yè)務(wù)自由調(diào)用,比較靈活瓣俯,而且這樣更加符合面向?qū)ο蟮乃枷耄ㄔO(shè)計(jì)模式中的建造者模式)杰标。

傳統(tǒng)做法

假如我們的工具類(lèi)叫SQLTool,那么拆分后的方法定義應(yīng)該是以下類(lèi)似的:

@interface SQLTool : NSObject

//用于保存拼接后的SQL
@property (nonatomic, strong, readonly) NSString *sql;

//為了能后續(xù)接著調(diào)用彩匕,所以返回值還是該對(duì)象
- (SQLTool *)select:(NSArray *)columns;              
- (SQLTool *)from:(NSString *)tableName;
- (SQLTool *)where:(NSString *)where;
- (SQLTool *)orderBy:(NSString *)orderBy;
...

@end

實(shí)現(xiàn)部分這里就省略了腔剂,無(wú)非就是將傳進(jìn)來(lái)的參數(shù)按照各自的format拼接到self.sql后面,再返回對(duì)象自己驼仪。

使用起來(lái)是這樣的:

SQLTool *tool = [[SQLTool alloc] init];

NSString *testSQL1 = [[[tool select:nil] from:@"Table"] orderBy:@"Column DESC"].sql;
NSString *testSQL2 = [[[[tool select:nil] from:@"Table"] where:@"A = B"] orderBy:@"Column DESC"].sql;

拆分以后掸犬,使用起來(lái)確實(shí)是靈活了,這里 testSQL2 比 testSQL1 多了一個(gè)WHERE條件谅畅。當(dāng)然還可以添加leftJoin登渣,rightJoin等方法噪服,使用的時(shí)候可以跟業(yè)務(wù)需求調(diào)用毡泻。但是仔細(xì)一看testSQL1看上去還可以,但是testSQL2就感覺(jué)有點(diǎn)臃腫了粘优,要是再加上groupBy仇味,或者多個(gè)Join操作,那么這種臃腫就更加明顯了雹顺。而這種臃腫主要體現(xiàn)在方法調(diào)用時(shí)候的那一對(duì)“[]”上丹墨。就算是回車(chē)換行,頭尾的括號(hào)還是太多了嬉愧,各個(gè)方法之間的銜接也因?yàn)槔ㄌ?hào)的存在贩挣,閱讀性不是很強(qiáng)。

鏈?zhǔn)骄幊?/h3>

如果采用鏈?zhǔn)骄幊痰姆绞剑敲磪⒖枷翸asonry的實(shí)現(xiàn)王财,我們最終實(shí)現(xiàn)的目標(biāo)是以下類(lèi)似的:

SQLTool *tool = [[SQLTool alloc] init];
NSString *testSQL1 = tool.select(nil).from(@"Table").orderBy(@"Column DESC").sql;

這種方式明顯看上去清爽許多卵迂,就算免不了調(diào)用的方法次數(shù)較多,使得整行代碼較長(zhǎng)绒净,通過(guò)換行還是能保持一定的可閱讀性见咒。

那么我們來(lái)分析下這樣的書(shū)寫(xiě)方式是怎么實(shí)現(xiàn)的。

首先挂疆,用點(diǎn)(.)的形式調(diào)用改览,那可以確定基本上是屬性(雖然沒(méi)有參數(shù)的方法也可以通過(guò)點(diǎn)(.)的形式調(diào)用,但是一般不推薦這么做缤言,有興趣的朋友可以查閱一些代碼規(guī)范或者相關(guān)帖子的說(shuō)明)宝当;然后,后面可以在括號(hào)里傳參數(shù)胆萧,那么進(jìn)一步確定今妄,這個(gè)屬性應(yīng)該是一個(gè)block;最后鸳碧,調(diào)完一次后盾鳞,還可以繼續(xù)調(diào)用,那么說(shuō)明block的返回值應(yīng)該還是這個(gè)對(duì)象瞻离。

那么以select方法為例腾仅,改造后的代碼如下:

.h文件

//定義select的block
typedef SQLTool *(^Select)(NSArray<NSString *> *columns);
@property (nonatomic, strong, readonly) Select select;

.m文件

- (Select)select {
    return ^(NSArray<NSString *> *columns) {
        if (columns.count > 0) {
            self.sql = [NSString stringWithFormat:@"SELECT %@", [columns componentsJoinedByString:@","]];
        } else {
            self.sql = @"SELECT *";
        }
        //這里將自己返回出去
        return self;
    }
}

看上面代碼應(yīng)該很清楚了,想要實(shí)現(xiàn)from套利、join等操作推励,也是用類(lèi)似的方式就可以了,這里就不一一貼出來(lái)了肉迫。

一些總結(jié)

1.什么時(shí)候使用鏈?zhǔn)骄幊蹋?br>
從上面的例子還有Masonry可以看出验辞,在面向一些過(guò)程化處理的時(shí)候(拼接SQL、給View加約束喊衫,都可以看成需要一步步完成的過(guò)程)跌造,需要將這些“過(guò)程”拆分,然后在“組合”這些“過(guò)程”的時(shí)候族购,就可以使用鏈?zhǔn)骄幊炭翘埃沟么a更加清晰,增加閱讀性寝杖。

2.鏈?zhǔn)骄幊痰暮诵膶?shí)現(xiàn)

實(shí)現(xiàn)鏈?zhǔn)骄幊痰年P(guān)鍵就是聲明一個(gè)block的屬性违施,而這個(gè)block返回值必須還是一個(gè)對(duì)象(根據(jù)業(yè)務(wù)需求不同,可以返回的是這個(gè)對(duì)象實(shí)例本身瑟幕,也可以是這個(gè)類(lèi)的另一個(gè)實(shí)例磕蒲,更可以是另一個(gè)類(lèi)的實(shí)例對(duì)象)留潦。而block中內(nèi)部的邏輯就是項(xiàng)目的業(yè)務(wù)邏輯(在這個(gè)例子中是拼接了SQL語(yǔ)句)。

#補(bǔ)充:如果在調(diào)用后不需要傳遞參數(shù)的話辣往,只需要聲明一個(gè)類(lèi)型是類(lèi)自己的屬性愤兵,并重寫(xiě)getter方法,在里面做相應(yīng)的操作就可以了排吴。以Masonry為例:

make.left.right.bottom.mas_equalTo(0.f);

前面部分的left秆乳,right,bottom都沒(méi)有傳遞參數(shù)钻哩,那么這部分的屬性是不需要聲明為block的屹堰,具體可以去查看Masonry源碼。

留在最后的思考

通過(guò)這個(gè)例子街氢,雖然是簡(jiǎn)單的實(shí)現(xiàn)了鏈?zhǔn)骄幊坛都沁€是有一些問(wèn)題值得我們?nèi)ニ伎嫉?

  • 在使用上跟Masonry還是有些區(qū)別的,為什么珊肃?
  • 就本篇舉的例子而言荣刑,還有些不足:在實(shí)際操作中,SELECT關(guān)鍵字及后面的列之后伦乔,應(yīng)該且只能是FROM厉亏,這是固定的,既不是WHERE烈和,也不是ORDER BY爱只,而現(xiàn)在的例子里,因?yàn)閎lock返回值是對(duì)象本身招刹,所以可能會(huì)出現(xiàn)這樣的情況:
tool.select(nil).from(@"table").from(@"table").select(nil);

也就是說(shuō)恬试,不但同一個(gè)操作可以重復(fù)調(diào)用,而且會(huì)出現(xiàn)不符合順序的調(diào)用疯暑。怎么樣才能避免這種情況呢训柴,這些問(wèn)題會(huì)在后續(xù)的文章中講解。

謝謝閱讀妇拯!

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者

  • 序言:七十年代末幻馁,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子乖阵,更是在濱河造成了極大的恐慌宣赔,老刑警劉巖,帶你破解...
    沈念sama閱讀 211,423評(píng)論 6 491
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件瞪浸,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡吏祸,警方通過(guò)查閱死者的電腦和手機(jī)对蒲,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,147評(píng)論 2 385
  • 文/潘曉璐 我一進(jìn)店門(mén)钩蚊,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人蹈矮,你說(shuō)我怎么就攤上這事砰逻。” “怎么了泛鸟?”我有些...
    開(kāi)封第一講書(shū)人閱讀 157,019評(píng)論 0 348
  • 文/不壞的土叔 我叫張陵蝠咆,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我北滥,道長(zhǎng)刚操,這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 56,443評(píng)論 1 283
  • 正文 為了忘掉前任再芋,我火速辦了婚禮菊霜,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘济赎。我一直安慰自己鉴逞,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,535評(píng)論 6 385
  • 文/花漫 我一把揭開(kāi)白布司训。 她就那樣靜靜地躺著构捡,像睡著了一般。 火紅的嫁衣襯著肌膚如雪壳猜。 梳的紋絲不亂的頭發(fā)上叭喜,一...
    開(kāi)封第一講書(shū)人閱讀 49,798評(píng)論 1 290
  • 那天,我揣著相機(jī)與錄音蓖谢,去河邊找鬼捂蕴。 笑死,一個(gè)胖子當(dāng)著我的面吹牛闪幽,可吹牛的內(nèi)容都是我干的啥辨。 我是一名探鬼主播,決...
    沈念sama閱讀 38,941評(píng)論 3 407
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼盯腌,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼溉知!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起腕够,我...
    開(kāi)封第一講書(shū)人閱讀 37,704評(píng)論 0 266
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤级乍,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后帚湘,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體玫荣,經(jīng)...
    沈念sama閱讀 44,152評(píng)論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,494評(píng)論 2 327
  • 正文 我和宋清朗相戀三年大诸,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了捅厂。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片贯卦。...
    茶點(diǎn)故事閱讀 38,629評(píng)論 1 340
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖焙贷,靈堂內(nèi)的尸體忽然破棺而出撵割,到底是詐尸還是另有隱情,我是刑警寧澤辙芍,帶...
    沈念sama閱讀 34,295評(píng)論 4 329
  • 正文 年R本政府宣布啡彬,位于F島的核電站,受9級(jí)特大地震影響故硅,放射性物質(zhì)發(fā)生泄漏庶灿。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,901評(píng)論 3 313
  • 文/蒙蒙 一契吉、第九天 我趴在偏房一處隱蔽的房頂上張望跳仿。 院中可真熱鬧,春花似錦捐晶、人聲如沸菲语。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 30,742評(píng)論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)山上。三九已至,卻和暖如春英支,著一層夾襖步出監(jiān)牢的瞬間佩憾,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 31,978評(píng)論 1 266
  • 我被黑心中介騙來(lái)泰國(guó)打工干花, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留妄帘,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 46,333評(píng)論 2 360
  • 正文 我出身青樓池凄,卻偏偏與公主長(zhǎng)得像抡驼,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子肿仑,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,499評(píng)論 2 348

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