Flutter里面錯誤捕獲的正確方法
背景
我們知道,在軟件開發(fā)過程中,錯誤和異常總是在所難免。
不管是客戶端的邏輯錯誤導(dǎo)致的,還是服務(wù)器的數(shù)據(jù)問題導(dǎo)致的,只要出現(xiàn)了異常,我們都需要一個機(jī)制來通知我們?nèi)ヌ幚怼?/p>
在 APP 的開發(fā)過程中,我們通過一些第三方的平臺,比如 Fabric、Bugly 等可以實現(xiàn)異常的日志上報。
Flutter 也有一些第三方的平臺,比如 Sentry 可以實現(xiàn)異常的日志上報。
但是為了更加通用一些,本篇不具體講解配合某個第三方平臺的異常日志捕獲,我們會告知大家如何在 Flutter 里面捕獲異常。
至于具體的上報途徑,不管是上報到自家的后臺服務(wù)器,還是通過第三方的 SDK API 接口進(jìn)行異常上報,都是可以的。
Demo 初始狀態(tài)
首先我們新建 Flutter 項目,修改 main.dart 代碼如下:
import 'package:flutter/material.dart'; void main() => runApp(MyApp()); class MyApp extends StatelessWidget { // This widget is the root of your application. @override Widget build(BuildContext context) { return MaterialApp( home: Scaffold( appBar: AppBar(title: Text('Flutter Crash Capture'),), body: MyHomePage(), ), ); } } class MyHomePage extends StatelessWidget { @override Widget build(BuildContext context) { return Container(); } }
效果如下:
捕獲錯誤
我們修改 MyHomePage,添加一個 List 然后進(jìn)行越界訪問,改動部分代碼如下:
class MyHomePage extends StatelessWidget { @override Widget build(BuildContext context) { List<String> numList = ['1', '2']; print(numList[6]); return Container(); } }
可以看到控制臺報錯如下:
flutter: ══╡ EXCEPTION CAUGHT BY WIDGETS LIBRARY ╞═══════════════════════════════════════════════════════════
flutter: The following RangeError was thrown building MyHomePage(dirty):
flutter: RangeError (index): Invalid value: Not in range 0..1, inclusive: 6
當(dāng)然這些錯誤信息在界面上也有顯示(debug 模式)。
那么我們?nèi)绾尾东@呢?
import 'dart:async'; import 'package:flutter/material.dart'; Future<Null> main() async { FlutterError.onError = (FlutterErrorDetails details) async { Zone.current.handleUncaughtError(details.exception, details.stack); }; runZoned<Future<void>>(() async { runApp(MyApp()); }, onError: (error, stackTrace) async { await _reportError(error, stackTrace); }); } Future<Null> _reportError(dynamic error, dynamic stackTrace) async { // TODO }
在 TODO 里面就可以執(zhí)行埋點上報操作或者其他處理了。
完整例子如下:
import 'dart:async'; import 'package:flutter/material.dart'; Future<Null> main() async { FlutterError.onError = (FlutterErrorDetails details) async { Zone.current.handleUncaughtError(details.exception, details.stack); }; runZoned<Future<void>>(() async { runApp(MyApp()); }, onError: (error, stackTrace) async { await _reportError(error, stackTrace); }); } Future<Null> _reportError(dynamic error, dynamic stackTrace) async { print('catch error='+error); } class MyApp extends StatelessWidget { // This widget is the root of your application. @override Widget build(BuildContext context) { return MaterialApp( home: Scaffold( appBar: AppBar(title: Text('Flutter Crash Capture'),), body: MyHomePage(), ), ); } } class MyHomePage extends StatelessWidget { @override Widget build(BuildContext context) { List<String> numList = ['1', '2']; print(numList[6]); return Container(); } }
運(yùn)行可以看到控制臺捕獲到錯誤如下:
flutter: catch error=RangeError (index): Invalid value: Not in range 0..1, inclusive: 6
assert 妙用
我們知道,一般錯誤上報都是在打包發(fā)布到市場后才需要。
平時調(diào)試的時候如果遇到錯誤,我們是會定位問題并修復(fù)的。
因此在 debug 模式下,我們不希望上報錯誤,而是希望直接打印到控制臺。
那么,這個時候就需要一種方式來區(qū)分現(xiàn)在是 debug 模式還是 release 模式,怎么區(qū)分呢?
這個時候就需要用到 assert 了。
bool get isInDebugMode { // Assume you're in production mode. bool inDebugMode = false; // Assert expressions are only evaluated during development. They are ignored // in production. Therefore, this code only sets `inDebugMode` to true // in a development environment. assert(inDebugMode = true); return inDebugMode; }
從注釋也可以知道,assert 表達(dá)式只在開發(fā)環(huán)境下會起作用,在生產(chǎn)環(huán)境下會被忽略。
因此利用這一個,我們就可以實現(xiàn)我們的需求。
上面的結(jié)論要驗證也很簡單,我們就不演示了。
完整模板
import 'dart:async'; import 'package:flutter/material.dart'; Future<Null> main() async { FlutterError.onError = (FlutterErrorDetails details) async { if (isInDebugMode) { FlutterError.dumpErrorToConsole(details); } else { Zone.current.handleUncaughtError(details.exception, details.stack); } }; runZoned<Future<void>>(() async { runApp(MyApp()); }, onError: (error, stackTrace) async { await _reportError(error, stackTrace); }); } Future<Null> _reportError(dynamic error, dynamic stackTrace) async { // TODO } bool get isInDebugMode { // Assume you're in production mode. bool inDebugMode = false; // Assert expressions are only evaluated during development. They are ignored // in production. Therefore, this code only sets `inDebugMode` to true // in a development environment. assert(inDebugMode = true); return inDebugMode; }
debug 模式下,直接將錯誤打印到控制臺,方便定位問題。
release 模式下,將錯誤信息收集起來,上傳到服務(wù)器。
參考鏈接:
Report errors to a service
總結(jié)
以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,謝謝大家對我們的支持。
上一篇:Android BSearchEdit 搜索結(jié)果選擇框的實例代碼
欄 目:Android
下一篇:Flutter 滾動監(jiān)聽及實戰(zhàn)appBar滾動漸變的實現(xiàn)
本文標(biāo)題:Flutter里面錯誤捕獲的正確方法
本文地址:http://mengdiqiu.com.cn/a1/Android/9165.html
您可能感興趣的文章
- 01-10如何給Flutter界面切換實現(xiàn)點特效
- 01-10Flutter適配深色模式的方法(DarkMode)
- 01-10Flutter 滾動監(jiān)聽及實戰(zhàn)appBar滾動漸變的實現(xiàn)
- 01-10如何使用Flutter實現(xiàn)58同城中的加載動畫詳解
- 01-10Flutter 假異步的實現(xiàn)示例
- 01-10使用Flutter實現(xiàn)一個走馬燈布局的示例代碼
- 01-10Flutter中如何加載并預(yù)覽本地的html文件的方法
- 01-10flutter 中監(jiān)聽滑動事件
- 01-10Flutter 實現(xiàn)下拉刷新上拉加載的示例代碼
- 01-10Windows實現(xiàn)Flutter環(huán)境搭建及配置這一篇就夠了


閱讀排行
本欄相關(guān)
- 01-10Android自定義View之繪制圓形頭像功能
- 01-10Android實現(xiàn)雙擊返回鍵退出應(yīng)用實現(xiàn)方
- 01-10android實現(xiàn)簡單計算器功能
- 01-10android實現(xiàn)記住用戶名和密碼以及自動
- 01-10C++自定義API函數(shù)實現(xiàn)大數(shù)相乘算法
- 01-10Android 友盟第三方登錄與分享的實現(xiàn)代
- 01-10android實現(xiàn)指紋識別功能
- 01-10如何給Flutter界面切換實現(xiàn)點特效
- 01-10Android實現(xiàn)圓形漸變加載進(jìn)度條
- 01-10Emoji表情在Android JNI中的兼容性問題詳
隨機(jī)閱讀
- 04-02jquery與jsp,用jquery
- 01-10delphi制作wav文件的方法
- 01-10C#中split用法實例總結(jié)
- 01-10使用C語言求解撲克牌的順子及n個骰子
- 01-11ajax實現(xiàn)頁面的局部加載
- 08-05dedecms(織夢)副欄目數(shù)量限制代碼修改
- 01-11Mac OSX 打開原生自帶讀寫NTFS功能(圖文
- 01-10SublimeText編譯C開發(fā)環(huán)境設(shè)置
- 08-05DEDE織夢data目錄下的sessions文件夾有什
- 08-05織夢dedecms什么時候用欄目交叉功能?