Flutter(九)--Flutter中Widget刷新邏輯+源碼解讀

Flutter中Widget刷新邏輯+源碼解讀

前言

我們都知道StatefulWidget可以進行頁面刷新操作,而StatelessWidget并不具備這項功能被济,依舊在最開始拋出兩個問題:

  1. 為什么只有StatefulWidget可以做頁面更新操作?
  2. setState()之后是否是所有的組件都會重新創(chuàng)建涧团?

首先我們看一下setState(fn)都做了什么~

abstract class State<T extends StatefulWidget>...{
    void setState(VoidCallback fn) {
    ...
    final dynamic result = fn() as dynamic;
    _element.markNeedsBuild();
    }
}

//在Element類中
{
    void markNeedsBuild() {
    ...
    if (dirty)  //如果是已經(jīng)標(biāo)記為臟只磷,則直接結(jié)束
      return;
    _dirty = true;
    owner.scheduleBuildFor(this);
    }
}

//owner屬于BuildOwner類
class BuildOwner {
    void scheduleBuildFor(Element element) {
    ...
    _dirtyElements.add(element);
    }
    
    void buildScope(Element context, [ VoidCallback callback ]) {
        ...
        while (index < dirtyCount) {
            _dirtyElements[index].rebuild();
        }
    }
}

//而rebuild最終還是會回到我們熟悉的performRebuild方法里,除了最基本的build方法。
void performRebuild() {
    ...
    built = build();
    ...
    _child = updateChild(_child, built, slot);
}

目前還有一個問題buildScope這個方法是否是Flutter隱式調(diào)用的呢泌绣?有答案的同學(xué)可以指教指教钮追。目前沒找到調(diào)用的位置。

  • 經(jīng)過一系列調(diào)用阿迈,最終會到達到updateChild這個方法里元媚,目前為止當(dāng)前包含當(dāng)前Widget的Element就會進入到updateChild更新流程里。
  • _dirty這個參數(shù)的使用,我認(rèn)為是非常優(yōu)化的刊棕。即使你做出重復(fù)刷新的操作也不會導(dǎo)致頁面的重復(fù)刷新炭晒。
StatelessElement中并沒有找到setState等刷新方法,所以無法支持刷新甥角,回答了之前的問題一腰埂。雖然依舊可以以類似的方式實現(xiàn)為StatefulWidget的子類,但是會有問題蜈膨,這里就不具體說明屿笼,可以參考Flutter文檔Why is the build method on State, and not StatefulWidget?

現(xiàn)在來看看updateChild都做了什么~

// 只有類型相同且key相同的就是可以刷新的
static bool canUpdate(Widget oldWidget, Widget newWidget) {
    return oldWidget.runtimeType == newWidget.runtimeType
        && oldWidget.key == newWidget.key;
  }

Element updateChild(Element child, Widget newWidget, dynamic newSlot) {
    //如果在widgetTree中當(dāng)前widget被刪除則直接結(jié)束,并在ElementTree中也刪除它
    if (newWidget == null) {
        deactivateChild(child);
        return;
    }
    ...
    Element newChild;
    if (child != null) {
        ...
        if (hasSameSuperclass && child.widget == newWidget) {
        //如果相同則不進行updata操作
             newChild = child;
        } else if (hasSameSuperclass && Widget.canUpdate(child.widget, newWidget)) {
        //如果可以更新則進行update操作
             child.update(newWidget);
             newChild = child;
        }else{
        //inflateWidget中會執(zhí)行createElement這里就不多贅述了
            newChild = inflateWidget(newWidget, newSlot);
        }
    }
  }
  
void update(covariant Widget newWidget) {
...
    _widget = newWidget;
}

Element inflateWidget(Widget newWidget, dynamic newSlot) {
...
final Element newChild = newWidget.createElement();
newChild.mount(this, newSlot);
}

  • 其實update在很多類中都有實現(xiàn),但是基本上都是大差不差翁巍。
  • 通過調(diào)試發(fā)現(xiàn)widget的對比是通過widget的hash值來進行的驴一,所以任何改動都會導(dǎo)致hash值不同。
  • updateChild這個方法沒有什么好說的灶壶,只是在canUpdate中發(fā)現(xiàn)如果不使用key肝断,導(dǎo)致這個判斷oldWidget.key == newWidget.key默認(rèn)為true。如果不想要進行復(fù)用的Widget則使用不同的key就可以實現(xiàn)驰凛。
  • update要注意方法中的_widget = newWidget胸懈,更新后會持有newWidget。
  • inflateWidget在遇到需要創(chuàng)建新的Element的時候恰响,看到了上一篇遇到過的createElement,mount也佐證了之前的Widget創(chuàng)建到Element的創(chuàng)建過程趣钱。
通過對刷新部分的源碼閱讀發(fā)現(xiàn),并不是所有的Widget都被會刷新胚宦、重新創(chuàng)建首有,某些可以更新的Widget還是可以update后復(fù)用的;某些hash值沒有發(fā)生變化的則直接復(fù)用枢劝。

后序

  • 整個源碼閱讀下來依舊發(fā)現(xiàn)Element這個中間者的設(shè)計是多么的巧妙井联,以及diff算法雖然看起來很簡單但是其中邏輯是非常嚴(yán)謹(jǐn)?shù)摹?/li>
  • 在這兩部分的源碼閱讀發(fā)現(xiàn),如果帶著問題去閱讀源碼您旁,不僅可以快速找到問題的原因烙常;還能提高源碼的閱讀速度,因為可以排除一些無關(guān)的方法鹤盒,不會毫無頭緒蚕脏。值得推薦。

傳送門:

Flutter-匯總

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末昨悼,一起剝皮案震驚了整個濱河市蝗锥,隨后出現(xiàn)的幾起案子跃洛,更是在濱河造成了極大的恐慌率触,老刑警劉巖,帶你破解...
    沈念sama閱讀 218,755評論 6 507
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件汇竭,死亡現(xiàn)場離奇詭異葱蝗,居然都是意外死亡穴张,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,305評論 3 395
  • 文/潘曉璐 我一進店門两曼,熙熙樓的掌柜王于貴愁眉苦臉地迎上來皂甘,“玉大人,你說我怎么就攤上這事悼凑〕フ恚” “怎么了?”我有些...
    開封第一講書人閱讀 165,138評論 0 355
  • 文/不壞的土叔 我叫張陵户辫,是天一觀的道長渐夸。 經(jīng)常有香客問我,道長渔欢,這世上最難降的妖魔是什么墓塌? 我笑而不...
    開封第一講書人閱讀 58,791評論 1 295
  • 正文 為了忘掉前任,我火速辦了婚禮奥额,結(jié)果婚禮上苫幢,老公的妹妹穿的比我還像新娘。我一直安慰自己垫挨,他們只是感情好韩肝,可當(dāng)我...
    茶點故事閱讀 67,794評論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著九榔,像睡著了一般伞梯。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上帚屉,一...
    開封第一講書人閱讀 51,631評論 1 305
  • 那天谜诫,我揣著相機與錄音,去河邊找鬼攻旦。 笑死喻旷,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的牢屋。 我是一名探鬼主播且预,決...
    沈念sama閱讀 40,362評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼烙无!你這毒婦竟也來了锋谐?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,264評論 0 276
  • 序言:老撾萬榮一對情侶失蹤截酷,失蹤者是張志新(化名)和其女友劉穎涮拗,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,724評論 1 315
  • 正文 獨居荒郊野嶺守林人離奇死亡三热,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,900評論 3 336
  • 正文 我和宋清朗相戀三年鼓择,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片就漾。...
    茶點故事閱讀 40,040評論 1 350
  • 序言:一個原本活蹦亂跳的男人離奇死亡呐能,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出抑堡,到底是詐尸還是另有隱情摆出,我是刑警寧澤,帶...
    沈念sama閱讀 35,742評論 5 346
  • 正文 年R本政府宣布首妖,位于F島的核電站懊蒸,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏悯搔。R本人自食惡果不足惜骑丸,卻給世界環(huán)境...
    茶點故事閱讀 41,364評論 3 330
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望妒貌。 院中可真熱鬧通危,春花似錦、人聲如沸灌曙。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,944評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽在刺。三九已至逆害,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間蚣驼,已是汗流浹背魄幕。 一陣腳步聲響...
    開封第一講書人閱讀 33,060評論 1 270
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留颖杏,地道東北人纯陨。 一個月前我還...
    沈念sama閱讀 48,247評論 3 371
  • 正文 我出身青樓,卻偏偏與公主長得像留储,于是被迫代替她去往敵國和親翼抠。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 44,979評論 2 355