前端開(kāi)發(fā)者初次開(kāi)發(fā) Native 應(yīng)用的誤區(qū)

太長(zhǎng)不讀版

被動(dòng)態(tài)語(yǔ)言寵壞的前端在初次開(kāi)發(fā) Native 應(yīng)用的時(shí)候婆瓜,容易掉進(jìn)以下幾個(gè)坑:

  • 害怕程序崩潰
  • 滿篇 Optional

正文

首先說(shuō)一下我司的背景快集。我司大部分開(kāi)發(fā)人員是網(wǎng)頁(yè)前端開(kāi)發(fā)和 Java 后端開(kāi)發(fā)。公司最近搞了一個(gè) iOS 應(yīng)用開(kāi)發(fā)廉白,使用了蘋果的新開(kāi)發(fā)語(yǔ)言 Swift个初。

經(jīng)過(guò)了一段時(shí)間的開(kāi)發(fā),我發(fā)現(xiàn)前端開(kāi)發(fā)者在初次開(kāi)發(fā)原生應(yīng)用猴蹂,特別是使用靜態(tài)語(yǔ)言時(shí)容易走入幾個(gè)誤區(qū)院溺。

害怕程序崩潰

一個(gè)非常有趣的現(xiàn)象是,我發(fā)現(xiàn)很多開(kāi)發(fā)者非常害怕程序崩潰掉磅轻,因此在代碼中加入了大量的防御性代碼珍逸。

其實(shí)這個(gè)東西見(jiàn)仁見(jiàn)智,但是個(gè)人認(rèn)為聋溜,在函數(shù)遇到與預(yù)期不符的狀況的時(shí)候谆膳,優(yōu)雅地崩潰掉比為了不崩潰而隱藏錯(cuò)誤重要得多。

舉一個(gè)非常簡(jiǎn)單的例子:某個(gè)成員函數(shù)需要使用某個(gè)成員撮躁,而這個(gè)成員在這個(gè)類初始化的時(shí)候并不一定存在漱病。

很多人會(huì)把代碼寫成這樣:

class MyObject {
  var somethingToBeSetOutside: ImportData?

  func foo() {
    guard let sth = somethingToBeSetOutside else {
      return
    }

    // use 'sth' the import data
    ...
    ...
    ...
  }
}

這看起來(lái)并沒(méi)有什么問(wèn)題。但是如果將來(lái)某個(gè)時(shí)候,外部的調(diào)用者偶然沒(méi)有設(shè)置 somethingToBeSetOutside 這個(gè)成員的值杨帽,那么 foo 里面很大一部分代碼都將不會(huì)被調(diào)用漓穿。而且這個(gè) bug 將會(huì)非常難以查找

那么這個(gè)問(wèn)題怎么解決呢注盈?其實(shí)在“古老”的 C/C++ 里面晃危,有一種簡(jiǎn)單的處理方式,那就是 assert ——調(diào)用者如果想要調(diào)用某個(gè)函數(shù)老客,那么傳入的參數(shù)必須滿足指定的契約山害。

例子:

class MyObject {
  var somethingToBeSetOutside: ImportData!

  func foo() {
    assert(somethingToBeSetOutside != nil)

    // use the import data
    ...
    ...
    ...
  }
}

這個(gè)例子看起來(lái)和前面的代碼非常類似,僅有兩個(gè)區(qū)別:

  • somethingToBeSetOutside 的類型是 ImportData!
  • 加入斷言判空沿量,如果不滿足條件就讓程序崩潰

我們把類型修改為 ImportData! 的目的是為了 foo 下面的代碼能夠直接訪問(wèn)這個(gè) somethingToBeSetOutside 的內(nèi)容浪慌。而加入斷言的目的,是給這個(gè)函數(shù)加入了一個(gè)契約的概念朴则,就是說(shuō)我函數(shù)不負(fù)責(zé)檢查你傳入的參數(shù)权纤,你要調(diào)用我,那你得負(fù)責(zé)輸入的參數(shù)符合我的規(guī)則乌妒。

那么有人可能會(huì)說(shuō)了汹想,程序編譯成 release 以后,就不會(huì)做 assert 檢查了撤蚊,那你這程序交付給客戶的時(shí)候崩潰了掉了怎么辦古掏?

其實(shí)這很簡(jiǎn)單,這個(gè) assert 的作用是在開(kāi)發(fā)的時(shí)候防呆侦啸,在 debug 時(shí)候給開(kāi)發(fā)人員提示錯(cuò)誤原因槽唾,而不是用來(lái)保證 release 代碼正常工作的。只要應(yīng)用程序經(jīng)過(guò)了適當(dāng)?shù)臏y(cè)試光涂,基本上所有錯(cuò)誤調(diào)用的地方都能在上線前被找出來(lái)庞萍。

另外,如果遇到了數(shù)據(jù)真正是可能沒(méi)有的情況忘闻,代碼也不應(yīng)該這樣寫了钝计。

滿篇的 Optional

先上代碼:

struct APIReturnedPassenger {
  let firstName: String?
  let middleName: String?
  let lastName: String?
}

因?yàn)椴荒艽_認(rèn) API 返回的數(shù)據(jù)是否包含了 Passenger 所有的信息,有些開(kāi)發(fā)人員就把結(jié)構(gòu)上的很多成員做成了 Optional 的齐佳。

這看起來(lái)似乎很好私恬。但是,如果我把這個(gè)翻譯成 C 的結(jié)構(gòu):

struct APIReturnedPassenger
{
  char *firstName;
  char *middleName;
  char *lastName;
};

看出什么沒(méi)有炼吴?這似乎就是 C 語(yǔ)言被新人詬病的指針滿天飛本鸣!

這種例子非常好解決。

首先把結(jié)構(gòu)設(shè)計(jì)成下面這樣:

struct APIReturnedPassenger {
  let firstName: String
  let middleName: String
  let lastName: String
}

然后在接收 API 返回內(nèi)容的地方缺厉,對(duì)返回的信息做驗(yàn)證即可永高。

如果 API 返回的信息不滿足我們預(yù)先設(shè)定好的格式,直接提示用戶服務(wù)器出現(xiàn)了故障或者使用預(yù)先設(shè)定的默認(rèn)值提针,后面的代碼寫起來(lái)不要舒服太多命爬。

再看一個(gè)例子:

假設(shè)這是一個(gè)點(diǎn)餐的系統(tǒng),客戶可以選擇不要甜點(diǎn)辐脖,一份甜點(diǎn)饲宛,或者兩份甜點(diǎn)∈燃郏恐怕下面這種代碼是很常見(jiàn)的:

func setDessert(firstDessert: Dessert?, secondDessert: Dessert?) {
   ...
}

可以想象艇抠,后面處理甜點(diǎn)的代碼將是大災(zāi)難,到處都需要判空久锥,各種復(fù)雜的邏輯處理家淤。

其實(shí) Swift 提供了一種機(jī)制,在處理類似的問(wèn)題上非常輕松愉快瑟由。

enum DessertType {
  case NoDessert,
  case OneDessert(Dessert),
  case FullDessert(Dessert, Dessert)
}

func setDessert(dessert: DessertType) {
  ...
}

程序這樣寫以后絮重,我們可以發(fā)現(xiàn)完全不需要處理各種奇葩的邏輯了。

總結(jié)

使用靜態(tài)語(yǔ)言的原生開(kāi)發(fā)和使用動(dòng)態(tài)語(yǔ)言的前端開(kāi)發(fā)在開(kāi)發(fā)理念上有一些不一樣的地方歹苦。不同領(lǐng)域的開(kāi)發(fā)人員確實(shí)有不同的習(xí)慣青伤。

希望這篇簡(jiǎn)單的文章能夠幫助一些轉(zhuǎn)到原生開(kāi)發(fā)的開(kāi)發(fā)人員,我就很滿足了殴瘦。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末狠角,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子蚪腋,更是在濱河造成了極大的恐慌丰歌,老刑警劉巖,帶你破解...
    沈念sama閱讀 217,734評(píng)論 6 505
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件屉凯,死亡現(xiàn)場(chǎng)離奇詭異动遭,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)神得,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,931評(píng)論 3 394
  • 文/潘曉璐 我一進(jìn)店門厘惦,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人哩簿,你說(shuō)我怎么就攤上這事宵蕉。” “怎么了节榜?”我有些...
    開(kāi)封第一講書人閱讀 164,133評(píng)論 0 354
  • 文/不壞的土叔 我叫張陵羡玛,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我宗苍,道長(zhǎng)稼稿,這世上最難降的妖魔是什么薄榛? 我笑而不...
    開(kāi)封第一講書人閱讀 58,532評(píng)論 1 293
  • 正文 為了忘掉前任,我火速辦了婚禮让歼,結(jié)果婚禮上敞恋,老公的妹妹穿的比我還像新娘。我一直安慰自己谋右,他們只是感情好硬猫,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,585評(píng)論 6 392
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著改执,像睡著了一般啸蜜。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上辈挂,一...
    開(kāi)封第一講書人閱讀 51,462評(píng)論 1 302
  • 那天衬横,我揣著相機(jī)與錄音,去河邊找鬼终蒂。 笑死冕香,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的后豫。 我是一名探鬼主播悉尾,決...
    沈念sama閱讀 40,262評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼挫酿!你這毒婦竟也來(lái)了构眯?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書人閱讀 39,153評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤早龟,失蹤者是張志新(化名)和其女友劉穎惫霸,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體葱弟,經(jīng)...
    沈念sama閱讀 45,587評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡壹店,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,792評(píng)論 3 336
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了芝加。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片硅卢。...
    茶點(diǎn)故事閱讀 39,919評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖将塑,靈堂內(nèi)的尸體忽然破棺而出蝌麸,到底是詐尸還是另有隱情,我是刑警寧澤敢辩,帶...
    沈念sama閱讀 35,635評(píng)論 5 345
  • 正文 年R本政府宣布戚长,位于F島的核電站,受9級(jí)特大地震影響历葛,放射性物質(zhì)發(fā)生泄漏恤溶。R本人自食惡果不足惜帜羊,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,237評(píng)論 3 329
  • 文/蒙蒙 一讼育、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧奶段,春花似錦痹籍、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書人閱讀 31,855評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)娜谊。三九已至斤讥,卻和暖如春纱皆,著一層夾襖步出監(jiān)牢的瞬間芭商,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書人閱讀 32,983評(píng)論 1 269
  • 我被黑心中介騙來(lái)泰國(guó)打工澳眷, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留蛉艾,地道東北人衷敌。 一個(gè)月前我還...
    沈念sama閱讀 48,048評(píng)論 3 370
  • 正文 我出身青樓缴罗,卻偏偏與公主長(zhǎng)得像祭埂,于是被迫代替她去往敵國(guó)和親蛆橡。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,864評(píng)論 2 354

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

  • Android 自定義View的各種姿勢(shì)1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 172,116評(píng)論 25 707
  • 在前面一篇文章中垃喊,我們分析了Android應(yīng)用程序窗口的繪圖表面的創(chuàng)建過(guò)程本谜。Android應(yīng)用程序窗口的繪圖表面在...
    一個(gè)不掉頭發(fā)的開(kāi)發(fā)閱讀 4,570評(píng)論 0 14
  • Spring Cloud為開(kāi)發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見(jiàn)模式的工具(例如配置管理,服務(wù)發(fā)現(xiàn)笤成,斷路器,智...
    卡卡羅2017閱讀 134,656評(píng)論 18 139
  • 1. Java基礎(chǔ)部分 基礎(chǔ)部分的順序:基本語(yǔ)法眷茁,類相關(guān)的語(yǔ)法炕泳,內(nèi)部類的語(yǔ)法上祈,繼承相關(guān)的語(yǔ)法培遵,異常的語(yǔ)法,線程的語(yǔ)...
    子非魚_t_閱讀 31,631評(píng)論 18 399
  • 世上沒(méi)有一件工作不辛苦籽腕,沒(méi)有一處人事不復(fù)雜。不要隨意發(fā)脾氣纸俭,誰(shuí)都不欠你的皇耗。學(xué)會(huì)低調(diào),取舍間必有得失郎楼,不用太計(jì)較万伤。學(xué)...
    千里馬_ec84閱讀 245評(píng)論 0 0