Flutter Exception降到萬分之幾的秘密
閒魚技術團隊於2018年上半年率先引入了Flutter技術實現客戶端開發,到目前為止成功改造並上線了複雜的商品詳情和釋出業務。隨著flutter比重越來越多,我們開始大力治理flutter的exception,起初很長一段時間內閒魚內flutter的exception率一直在千分之幾左右。經過我們的整理和解決,解決了90%以上的flutter exception。
我們對exception進行了歸類,大頭主要分為兩大類,這兩大類堆疊數量很多,佔到整體90%左右:
第一大類的堆疊都指向了setstate
#0 State.setState (package:flutter/src/widgets/framework.dart:1141) #1 _DetailCommentWidgetState.replyInput.<anonymous closure>.<anonymous closure> (package:fwn_idlefish/biz/item_detail/fx_detail_comment.dart:479) #2 FXMtopReq.sendReq.<anonymous closure> (package:fwn_idlefish/common_lib/network/src/mtop_req.dart:32) #3 NetService.requestWithModel.<anonymous closure> (package:fwn_idlefish/common_lib/network/src/net_service.dart:58) #4 _rootRunUnary (dart:async/zone.dart:1132) #5 _CustomZone.runUnary (dart:async/zone.dart:1029) #6 _FutureListener.handleValue (dart:async/future_impl.dart:129)
第二大類堆疊都與buildContext相關
#0 Navigator.of (package:flutter/src/widgets/navigator.dart:1270) #1 Navigator.pop (package:flutter/src/widgets/navigator.dart:1166) #2 UploadProgressDialog.hide (package:fwn_idlefish/biz/publish/upload_progress_dialog.dart:35) #3 PublishSubmitReducer.doPost.<anonymous closure> (package:fwn_idlefish/biz/publish/reducers/publish_submit_reducer.dart:418) <asynchronous suspension> #4 FXMtopReq.sendReq.<anonymous closure> (package:fwn_idlefish/common_lib/network/src/mtop_req.dart:32) #5 NetService.requestWithModel.<anonymous closure> (package:fwn_idlefish/common_lib/network/src/net_service.dart:58) #6 _rootRunUnary (dart:async/zone.dart:1132) #7 _CustomZone.runUnary (dart:async/zone.dart:1029)
第一類明顯與element和sate的生命週期有關。第二類與buildContext有關。
buildContext是什麼?
下面是一段state中獲取buildContext的實現
Element get _currentElement => _registry[this]; BuildContext get currentContext => _currentElement;
很明顯buildContext其實就是element例項。buildContext是一個介面,element是buildContext的具體實現。所以上面的exception都指向了flutter element和state的生命週期。
state生命週期
element與state生命週期
element是由widget createElement所建立。state的生命週期狀態由element呼叫觸發。
最核心的是在new elment的時候element的state的雙向繫結正式建立。在umount的時候element和state的雙向繫結斷開。
activity生命週期與state關係
flutter提供WidgetsBindingObserver給開發者來監聽AppLifecycleState。 AppLifecycleState有4種狀態:
1.resumed
介面可見,比如應用從後臺到前臺
2.inactive
頁面退到後臺或者彈出dialog等情況下
這種狀態下接收不到很任何使用者輸入,但是還會有drawframe的回撥
3.paused
應用掛起,比如退到後臺。進入這種狀態代表不在有任何drawframe的回撥
4.suspending
ios中沒用,puased之後進入的狀態,進入這種狀態代表不在有任何drawframe的回撥
看下android生命週期和appLifecycleState、state關係,
1.建立
2.按home鍵退到後臺
3.從後臺回到前臺
4.back鍵退出當前頁面(route pop)
5.back鍵退出應用
常見的exception例子
工程開發中容易忽略state的dispose狀態
看一段例子:
這個例子可能會在某些情況下excetion。
在state dispose後,element會和state斷開相互引用,如果在這個時候開發者去拿element的位置資訊或者呼叫setstate 重新整理佈局時就會報異常。
最常見的是在一些timer、animate、網路請求等非同步邏輯後呼叫setstate導致的excetion。安全的做法是在呼叫setstate前判斷一下state是否是mounted狀態。如下:
buildContext使用錯誤
看一段錯誤使用buildcontext例子:
上面的錯誤在於在跨堆疊使用了buildcontext。由於outcontext的生命週期與buttomcontext不一致,在彈出bottomsheet的時候outcontext可以已經處於umount或者deactivite。上面例子正確的做法是使用bottomcontext獲取focusScopeNode。
我們在跨堆疊傳遞引數(如bottomsheet、dialog、alert、processdialog等)場景時特別要注意buildcontext的使用。
來自 “ ITPUB部落格 ” ,連結:http://blog.itpub.net/69900359/viewspace-2286381/,如需轉載,請註明出處,否則將追究法律責任。
相關文章
- 網易雲首席安全架構師談安全新形勢:DDOS兩三天,遊戲玩家數從幾萬降到幾百架構遊戲
- 首日售出250萬份《漫威蜘蛛俠 2》:系列成功的秘密
- 處理 Exception 的幾種實踐,很優雅,被很多團隊採納!Exception
- Exception和Error的理解ExceptionError
- Oracle 中的exception——(Raise)OracleExceptionAI
- Oracle exceptionOracleException
- 異常(Exception)Exception
- Exception和ErrorExceptionError
- Java中Error和Exception的異同以及執行時異常(Runtime exception)與檢查型異常(checked exception)的區別JavaErrorException
- Pooling與馬賽克的秘密
- 資料視覺化的秘密視覺化
- Flutter工具:Dart中幾種常用的Json轉Object方式FlutterDartJSONObject
- Sanic Exception – 異常Exception
- java exception and finally returnJavaException
- ch12_exceptionException
- MySQL百分比顯示和顯示前百分之幾的方法MySql
- Flutter從入門到寄幾玩兒Flutter
- Flutter(十一)之封裝幾個小WidgetFlutter封裝
- SQL奇遇記:解鎖 SQL 的秘密SQL
- Http歷險記(下)-- Struts的秘密HTTP
- CSS的秘密——背景和邊框(下)CSS
- TikTok 推薦引擎強大的秘密
- Mac的秘密資料夾:Secret FolderMac
- Java 中的Exception 有什麼用?JavaException
- 淺談Kotlin的Checked Exception機制KotlinException
- 400天活躍叢集從0到2萬+,什麼是TiDB Serverless快速增長的秘密?TiDBServer
- 小小的導航欄,有大大的秘密。
- oracle 使用異常exceptionOracleException
- 【譯】Exception Helper – Rethrown ExceptionsException
- Symfony\Component\Debug\Exception\FatalErrorExceptionExceptionError
- Sanic exception() 方法/函式Exception函式
- Java-Exception與RuntimeExceptionJavaException
- PHP 異常類 ExceptionPHPException
- BOOST應用 無法解析的外部符號 "void __cdecl boost::throw_exception(class std::exception const &)"符號Exception
- Deep和Cross不得不說的秘密ROS
- 拼多多:商品詳情背後的秘密
- 12.ThreadLocal的那點小秘密thread
- Flutter 頁面間資料傳遞(共享)的幾種常用方式Flutter