Android Flutter中异常处理的方法总结

 更新时间:2023年06月28日 08:59:01   作者:97令山  
这篇文章主要为大家详细介绍了Android Flutter中异常处理的相关知识,文中的示例代码讲解详细,具有一定的学习价值,感兴趣的小伙伴可以跟随小编一起学习一下

Flutter如何捕获异常

程序异常会怎样

  • 在Java中,程序发生异常且没有被捕获,那么程序将会终止。
  • 但在Dart或JavaScript中则不会,究其原因,这和它们的运行机制有关系,Java是多线程模型的编程语言,任意一个线程触发异常且没被捕获时,整个进程就退出。
  • 但Dart和JavaScript不会,它们都是单线程模型,运行机制很相似(但有区别)。

flutter如何捕获异常

Dart中可以通过try/catch/finally来捕获代码块异常,这个和其它编程语言类似。

Flutter框架异常捕获

Flutter 框架在很多关键的方法进行了异常捕获。

举一个例子,当布局发生越界或不合规范时,Flutter就会自动弹出一个错误界面,这是因为Flutter已经在执行build方法时添加了异常捕获。

最终的源码如下,具体看 ComponentElement 类中的 performRebuild() 方法。

@override
void performRebuild() {
  try {
    //执行build方法  
    built = build();
  } catch (e, stack) {
    // 有异常时则弹出错误提示  
    built = ErrorWidget.builder(_debugReportException('building $this', e, stack));
  } 
}      

可以看到,在发生异常时,Flutter默认的处理方式是弹一个ErrorWidget

但如果想自己捕获异常并上报到报警平台的话应该怎么做?进入_debugReportException()方法看看:

FlutterErrorDetails _debugReportException(
  String context,
  dynamic exception,
  StackTrace stack, {
  InformationCollector informationCollector
}) {
  //构建错误详情对象  
  final FlutterErrorDetails details = FlutterErrorDetails(
    exception: exception,
    stack: stack,
    library: 'widgets library',
    context: context,
    informationCollector: informationCollector,
  );
  //报告错误 
  FlutterError.reportError(details);
  return details;
}

发现,错误是通过FlutterError.reportError方法上报的,继续跟踪:

static void reportError(FlutterErrorDetails details) {
  ...
  if (onError != null)
    onError(details); //调用了onError回调
}

发现onErrorFlutterError的一个静态属性,它有一个默认的处理方法 dumpErrorToConsole,到这里就清晰了,如果想自己上报异常,只需要提供一个自定义的错误处理回调即可,如:

void main() {
  FlutterError.onError = (FlutterErrorDetails details) {
    reportError(details);
  };
 ...
}

这样就可以处理那些Flutter为我们捕获的异常了,接下来看看如何捕获其它异常。

其它异常捕获与日志收集

在Flutter中,还有一些Flutter没有为我们捕获的异常,如调用空对象方法异常、Future中的异常。

在Dart中,异常分两类:同步异常和异步异常,同步异常可以通过try/catch捕获,而异步异常则比较麻烦,如下面的代码是捕获不了Future的异常的:

try{
    Future.delayed(Duration(seconds: 1)).then((e) => Future.error("xxx"));
}catch (e){
    print(e)
}

Dart中有一个runZoned(...) 方法,可以给执行对象指定一个Zone。

  • Zone表示一个代码执行的环境范围,为了方便理解,可以将Zone类比为一个代码执行沙箱,不同沙箱的之间是隔离的,沙箱可以捕获、拦截或修改一些代码行为。
  • 如Zone中可以捕获日志输出、Timer创建、微任务调度的行为,同时Zone也可以捕获所有未处理的异常。
  • 下面看看runZoned(...)方法定义:
R runZoned<R>(R body(), {
    Map zoneValues, 
    ZoneSpecification zoneSpecification,
    Function onError,
}) 

zoneValues: Zone 的私有数据,可以通过实例zone[key]获取,可以理解为每个“沙箱”的私有数据。

zoneSpecification:Zone的一些配置,可以自定义一些代码行为,比如拦截日志输出行为等,举个例子:

下面是拦截应用中所有调用print输出日志的行为。

main() {
    runZoned(() => runApp(MyApp()), zoneSpecification: new ZoneSpecification(
        print: (Zone self, ZoneDelegate parent, Zone zone, String line) {
          parent.print(zone, "Intercepted: $line");
        }),
    );
}

这样一来,APP中所有调用print方法输出日志的行为都会被拦截,通过这种方式,也可以在应用中记录日志,等到应用触发未捕获的异常时,将异常信息和日志统一上报。ZoneSpecification还可以自定义一些其他行为,读者可以查看API文档。

onError:Zone中未捕获异常处理回调,如果开发者提供了onError回调或者通过ZoneSpecification.handleUncaughtError指定了错误处理回调,那么这个zone将会变成一个error-zone,该error-zone中发生未捕获异常(无论同步还是异步)时都会调用开发者提供的回调,如:

runZoned(() {
    runApp(MyApp());
}, onError: (Object obj, StackTrace stack) {
    var details=makeDetails(obj,stack);
    reportError(details);
});

结合上面的FlutterError.onError就可以捕获我们Flutter应用中全部错误了!需要注意的是,error-zone内部发生的错误是不会跨越当前error-zone的边界的,如果想跨越error-zone边界去捕获异常,可以通过共同的“源”zone来捕获,如:

var future = new Future.value(499);
runZoned(() {
	var future2 = future.then((_) { throw "error in first error-zone"; });
	runZoned(() {
		var future3 = future2.catchError((e) { print("Never reached!"); });
	}, onError: (e) { print("unused error handler"); });
}, onError: (e) { print("catches error of first error-zone."); });

总结

最终异常捕获和上报代码如下:

void collectLog(String line){
    ... //收集日志
}
void reportErrorAndLog(FlutterErrorDetails details){
    ... //上报错误和日志逻辑
}
FlutterErrorDetails makeDetails(Object obj, StackTrace stack){
    ...// 构建错误信息
}
void main() {
  FlutterError.onError = (FlutterErrorDetails details) {
    reportErrorAndLog(details);
  };
  runZoned(
    () => runApp(MyApp()),
    zoneSpecification: ZoneSpecification(
      print: (Zone self, ZoneDelegate parent, Zone zone, String line) {
        collectLog(line); // 收集日志
      },
    ),
    onError: (Object obj, StackTrace stack) {
      var details = makeDetails(obj, stack);
      reportErrorAndLog(details);
    },
  );
}

到此这篇关于Android Flutter中异常处理的方法总结的文章就介绍到这了,更多相关Android Flutter异常处理内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • Android中控件GridView实现设置行列分割线的方法示例

    Android中控件GridView实现设置行列分割线的方法示例

    这篇文章主要介绍了利用Android中控件GridView实现设置行列分割线的方法,文中给出了详细的介绍与示例代码,相信对大家具有一定的参考价值,有需要的朋友们下面来一起看看吧。
    2017-01-01
  • android项目实现带进度条的系统通知栏消息

    android项目实现带进度条的系统通知栏消息

    本篇文章主要介绍了android项目实现带进度条的系统通知栏消息,就是实现在通知栏看到下载进度。具有一定的参考价值,感兴趣的小伙伴们可以参考一下。
    2016-10-10
  • Android7.0以上Uri转路径的方法实现(已验证)

    Android7.0以上Uri转路径的方法实现(已验证)

    这篇文章主要介绍了Android7.0以上Uri转路径的方法实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-03-03
  • 利用Kotlin的协程实现简单的异步加载详解

    利用Kotlin的协程实现简单的异步加载详解

    这篇文章主要给大家介绍了关于利用Kotlin的协程实现简单的异步加载的相关资料,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧。
    2018-03-03
  • Android自定义滑动解锁控件使用详解

    Android自定义滑动解锁控件使用详解

    这篇文章主要为大家详细介绍了Android自定义滑动解锁控件的使用方法,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-07-07
  • Kotlin中常见的符号详解

    Kotlin中常见的符号详解

    这篇文章主要介绍了Kotlin中常见的符号详解,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2020-03-03
  • Android动画之补间动画(Tween Animation)实例详解

    Android动画之补间动画(Tween Animation)实例详解

    这篇文章主要介绍了Android动画之补间动画(Tween Animation)用法,结合实例形式较为详细的分析了Android补间动画的定义,原理,注意事项与相关使用技巧,需要的朋友可以参考下
    2016-01-01
  • 使用runtime 实现weex 跳转原生页面

    使用runtime 实现weex 跳转原生页面

    这篇文章主要介绍了使用runtime 实现weex 跳转原生页面的相关资料,需要的朋友可以参考下
    2017-01-01
  • Android Studio Menu选择菜单的建立方法

    Android Studio Menu选择菜单的建立方法

    这篇文章主要介绍了Android Studio Menu选择菜单的建立,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2019-01-01
  • Flutter使用stack实现悬浮UI的示例代码

    Flutter使用stack实现悬浮UI的示例代码

    在Flutter中,你可以使用Stack和Positioned来创建悬浮 UI,这篇文章主要为大家详细介绍了Flutter使用stack实现悬浮UI的具体代码,希望对大家有所帮助
    2024-01-01

最新评论