從同步到異步 - iOS轉RN最需轉變的思路

說是iOS轉React Native蹂随,不如說是iOS轉JavaScript最需要轉變的思路。

本質上來說JavaScript是單線程語言玉罐;同時屈嗤,可能因為它是為網絡而生,需要高頻度地向服務器讀取數(shù)據(jù)吊输;再者饶号,可能因為它所在的客戶端——瀏覽器——的執(zhí)行效率并不高;因此季蚂,它采取了大量的異步化操作茫船。例如,要裝載圖片扭屁,需要有一個異步讀取圖片的過程算谈;要讀取文件/數(shù)據(jù)庫,也必須異步讀确杞痢濒生;可能最令iOS程序員發(fā)指的是,如果要使用原生的iOS模塊幔欧,那么必然會要用異取讀取罪治。

所以丽声,原來簡單的iOS程序,就會變得支離破碎觉义。就拿讀文件來說雁社,在iOS程序中,你只需要在一個模塊中按順序寫下來就好:

1)獲得文件路徑
2)打開文件晒骇,讀取文件內容(若讀取時間長霉撵,頂多來一個UI轉圈界面展示)
3)使用文件內容

轉到RN中呢,你需要

1)獲得文件路徑
2)開一個子線程去讀文件
3)準備一個回調(callback)函數(shù)洪囤,等待子線程讀取后再返回

看起來好像沒有什么區(qū)別徒坡,但是,由于JS是單線程函數(shù)瘤缩,你還不能在主線程中單純等待返回喇完。真實的情況是,主線程執(zhí)行完了剥啤,才會去執(zhí)行子線程锦溪。因此,一個簡單的讀取文件的過程府怯,就不得不寫成好像從服務器異步讀取數(shù)據(jù)一樣麻煩刻诊。

單是讀取文件也罷了,若是你寫了一個iOS原生模塊牺丙,是各種工具函數(shù)的集合则涯,然后會在RN模塊中反復調用。那么赘被,你就需要在等待一個函數(shù)返回之后是整,再調用另一個函數(shù),于是民假,在多次調用之后浮入,你發(fā)現(xiàn)已陷入到傳說中的“回調地獄”!

說實話羊异,這是我從iOS編程轉向RN的過程中事秀,最不習慣的一道坎!比起編程環(huán)境搭建野舶,比起各種模塊的調用方式易迹,這個思路上的轉變最讓人鬧心。

最后平道,還是找到一些相對簡單的解決方法睹欲。以下這篇文章給出了最有用的一種方法:
http://blog.csdn.net/kunshan_shenbin/article/details/40425143

簡單說,就是用特定的語法方式,讓編寫代碼的過程窘疮,變得不再大括號套小括號袋哼。以下是這種方法使用的簡單流程。

1.在文件頭引入Generator闸衫、以及next控制函數(shù)的工具性代碼

// 當前的 Generator
let activeGenerator;

// 處理 g.next() 功能
function gNext() {
    return function(err, data) {
    if(err) {
        throw err;
    }
    //  g.next()涛贯,并把回調函數(shù)的結果作為參數(shù)傳遞給 yield
    activeGenerator.next(data)
  }
}

// 控制工具
function gQueue(generatorFunc) {
  activeGenerator= generatorFunc(gNext());
  activeGenerator.next();
}

2.異步調用的時候使用這種語法開頭

  // 該語句實際產生了一個generator函數(shù),并定義了其中的next操作
   gQueue(function * flow(next) {

“flow”這個名字是隨便取的蔚出,這里表示它是一個流程控制工具弟翘,該名字不會被再次調用。接下來骄酗,我們在gQueue中稀余,就能大膽地像寫同步語法一樣,使用異步調用了:

        let result1 = yield (callBack => {
            someModule.someFunc1(var1, var2..., callBack);
        })(next);

        // 以下語句會在result1獲得值后繼續(xù)執(zhí)行
        let result2 = yield (callBack => {
            someModule.someFunc2(result, var2..., callBack);
        })(next);

實際上酥筝,由于generator函數(shù)的特性滚躯,每次.next()調用,都會來到新的一條yield語句嘿歌。然后,該函數(shù)所在線程會暫停茁影,開始調用yield語句中的匿名函數(shù)宙帝。在匿名函數(shù)中,我們放置了自己所想要調用的異步操作:someModule.someFunc募闲。

之后的方式非常巧妙步脓,注意,上述匿名函數(shù)所使用的形參callBack浩螺,在調用的時候靴患,被實參 next 替代。因此要出,callBack返回鸳君,實際上是調用了 next 函數(shù)。而根據(jù)最前面的工具定義患蹂,next 對應的是 gNext() 函數(shù)【JS的語法這個地方有點繞或颊,需要多看幾遍】,因此传于,callBack函數(shù)中囱挑,定義的第一個參數(shù) error 用來描述返回結果是否有誤,而第二個參數(shù)data沼溜,則被當成返回值傳給了yield平挑。那么,我們的 result 就獲得了這個data值系草。

更巧妙的是通熄,當 g.next(data)除了返回data值否淤,實質上還執(zhí)行了一次.next操作。那么棠隐,generator函數(shù)所有線程將繼續(xù)執(zhí)行石抡,這樣就保證了后續(xù)語句立刻會在返回后運行。實現(xiàn)了看似“同步連續(xù)執(zhí)行”的效果助泽。

如果對以上機制仍感模糊啰扛,阮一峰老師的這篇文章值得推薦:Generator 函數(shù)的語法

3.回調函數(shù)callBack的特殊規(guī)則

根據(jù)以上機制嗡贺,要正確地使用該方法隐解,必須對我們使用的callBack使用特定的返回機制。callBack的第一個參數(shù) error 必須用來描述返回結果是否有誤诫睬,而第二個參數(shù)data則用于返回真正需要的數(shù)據(jù)煞茫。假如你返回的數(shù)據(jù)有多個,那么必須打包成一個數(shù)據(jù)摄凡,然后在返回值里再做反向解析续徽。以下是這一callBack機制的正確使用方式:

    function someFunction(var1, var2, callback) {
        // some operation,最后得到兩個值要返回
        let result1 = ...;
        let result2 = ...;
        callback(null, [result1, result2]);
    }

最后的語句中亲澡,偷懶把錯誤值定成了null钦扭,待返回的兩個值,包裝成一個數(shù)組作為data返回過去了床绪。那么后面調用時客情,獲得這個兩個值,就必須分別從結果中以數(shù)組方式解析癞己。

4.其它

這篇文章只是描述了異步調用的一個最大障礙膀斋。相關的問題還有 setTimeout 函數(shù)的應用,以及針對異步模塊調用的監(jiān)聽等技術方案痹雅,將來有空再來細述仰担。

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市练慕,隨后出現(xiàn)的幾起案子惰匙,更是在濱河造成了極大的恐慌,老刑警劉巖铃将,帶你破解...
    沈念sama閱讀 207,248評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件项鬼,死亡現(xiàn)場離奇詭異,居然都是意外死亡劲阎,警方通過查閱死者的電腦和手機绘盟,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,681評論 2 381
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人龄毡,你說我怎么就攤上這事吠卷。” “怎么了沦零?”我有些...
    開封第一講書人閱讀 153,443評論 0 344
  • 文/不壞的土叔 我叫張陵祭隔,是天一觀的道長。 經常有香客問我路操,道長疾渴,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 55,475評論 1 279
  • 正文 為了忘掉前任屯仗,我火速辦了婚禮搞坝,結果婚禮上,老公的妹妹穿的比我還像新娘魁袜。我一直安慰自己桩撮,他們只是感情好,可當我...
    茶點故事閱讀 64,458評論 5 374
  • 文/花漫 我一把揭開白布峰弹。 她就那樣靜靜地躺著店量,像睡著了一般。 火紅的嫁衣襯著肌膚如雪垮卓。 梳的紋絲不亂的頭發(fā)上垫桂,一...
    開封第一講書人閱讀 49,185評論 1 284
  • 那天,我揣著相機與錄音粟按,去河邊找鬼。 笑死霹粥,一個胖子當著我的面吹牛灭将,可吹牛的內容都是我干的。 我是一名探鬼主播后控,決...
    沈念sama閱讀 38,451評論 3 401
  • 文/蒼蘭香墨 我猛地睜開眼庙曙,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了浩淘?” 一聲冷哼從身側響起捌朴,我...
    開封第一講書人閱讀 37,112評論 0 261
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎张抄,沒想到半個月后砂蔽,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經...
    沈念sama閱讀 43,609評論 1 300
  • 正文 獨居荒郊野嶺守林人離奇死亡署惯,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 36,083評論 2 325
  • 正文 我和宋清朗相戀三年左驾,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 38,163評論 1 334
  • 序言:一個原本活蹦亂跳的男人離奇死亡诡右,死狀恐怖安岂,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情帆吻,我是刑警寧澤域那,帶...
    沈念sama閱讀 33,803評論 4 323
  • 正文 年R本政府宣布,位于F島的核電站猜煮,受9級特大地震影響次员,放射性物質發(fā)生泄漏。R本人自食惡果不足惜友瘤,卻給世界環(huán)境...
    茶點故事閱讀 39,357評論 3 307
  • 文/蒙蒙 一翠肘、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧辫秧,春花似錦束倍、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,357評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至柿究,卻和暖如春邮旷,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背蝇摸。 一陣腳步聲響...
    開封第一講書人閱讀 31,590評論 1 261
  • 我被黑心中介騙來泰國打工婶肩, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人貌夕。 一個月前我還...
    沈念sama閱讀 45,636評論 2 355
  • 正文 我出身青樓律歼,卻偏偏與公主長得像,于是被迫代替她去往敵國和親啡专。 傳聞我的和親對象是個殘疾皇子险毁,可洞房花燭夜當晚...
    茶點故事閱讀 42,925評論 2 344

推薦閱讀更多精彩內容