美文网首页
Flutter之-dart多线程isolate(二)

Flutter之-dart多线程isolate(二)

作者: 仙人掌__ | 来源:发表于2021-03-08 10:26 被阅读0次

    dart中的isolate

    isolate可以理解为dart中的线程,但它又不同于线程,准确的说应该叫做协程,协程最大的优势就是它具有极高的执行效率,因为携程中子程序的调用不需要线程的切换,所以对于线程数量越大的程序来说协程的优势就越明显。每个isolate都有自己独立的执行线程和事件循环,以及内存,所以isolate之间不存在锁竞争的问题,各个isolate之间通过消息通信。

    root isolate

    对于每一个flutter应用,当应用被启动时都会有一个默认的isolate,称为root isolate。我们自己的代码默认情况下都在这个isolate中执行。

    消息循环

    类似于iOS的NSRunLoop或者安卓的Looper,每个isolate内部都有一个消息循环,它随着isolate的创建自动开启,存在两个队列 event queue和microtask queue,后者优先级高于前者,提交到队列中的任务按照顺序依次执行。引用官方的一张图片


    image.png

    先看如下代码:

    void function_main() async
    {
      print("开始执行");
      Timer.run((){
        print("timer event 立即执行1");
      });
      Future.delayed(Duration(seconds:1),(){
        print("event 延迟2秒执行");
      });
      scheduleMicrotask((){
        print("micro task 立即执行1");
      });
      print("结束执行");
    }
    

    执行结果为:
    flutter: 开始执行
    flutter: 结束执行
    flutter: micro task 立即执行1
    flutter: timer event 立即执行1
    flutter: event 延迟2秒执行
    符合预期

    接下来看如下一段代码:

    void function_main() async
    {
      print("开始执行");
      Timer.run((){
        print("timer event 执行1");
        scheduleMicrotask((){
          print("微任务2");
        });
      });
      scheduleMicrotask((){
        print("micro task 执行1");
        Timer.run(() {
          print("time event 执行2");
        });
      });
      print("结束执行");
    }
    

    运行结果如下:
    flutter: 开始执行
    flutter: 结束执行
    flutter: micro task 执行1
    flutter: timer event 执行1
    flutter: 微任务2
    flutter: time event 执行2
    可以看到microtask 优先级比event要高,而且提交到队列中的任务是顺序执行的,一个任务执行完成后才会执行下一个任务

    Future

    Future用于异步任务,通过它可以创建到event loop和microtask队列中的任务去执行,请看如下一段代码

    void function_main() async
    {
      print("开始执行");
    
      Future future1 = new Future(() => null);
      future1.then((value){
        print("future1 执行then2");
      });
      future1.then((_) {
        print("future1 执行then");
      }).catchError((e) {
        print("future1 执行catchError");
      }).whenComplete(() {
        print("future1 执行whenComplete");
      });
    
      Future future2 = new Future((){
        print("future2 初始化任务");
      });
    
      future2.then((_) {
        print("future2 执行then");
        future1.then((_){
          print("future1 执行第三个then");
        });
      }).catchError((e) {
        print("future2 执行catchError");
      }).whenComplete(() {
        print("future2 执行whenComplete");
      });
    
      future1.then((_) {
        print("future1 执行第二个then");
      });
    
      Future future3 = Future((){
        print("future3 初始化");
      });
    
      Future future4 = Future.value("立即执行").then((value){
        print("future4 执行then");
      }).whenComplete((){
        print("future4 执行whenComplete");
      });
      print("main 结束");
    }
    

    执行结果为:
    flutter: 开始执行
    flutter: main 结束
    flutter: future4 执行then
    flutter: future4 执行whenComplete
    flutter: future1 执行then2
    flutter: future1 执行then
    flutter: future1 执行whenComplete
    flutter: future1 执行第二个then
    flutter: future2 初始化任务
    flutter: future2 执行then
    flutter: future2 执行whenComplete
    flutter: future1 执行第三个then
    flutter: future3 初始化

    分析
    1、通过Future.value()函数创建的任务是立即执行的,所以就立即执行future4 执行then和future4 执行whenComplete
    2、每一个Future可以通过then()注册多个回调,他们按照注册的顺序依次执行。当所有注册的then()执行完毕后接着会回调whenComplete。(备注:如果是在whenComplete之后注册的then,那么这个then的任务将放在microtask执行,所以这也就是为什么"future1 执行第三个then"在"future3 初始化"前执行的原因),参考官方文档then的注释

    * When this future completes with a value,
    * the [onValue] callback will be called with that value.
    * If this future is already completed, the callback will not be called
    * immediately, but will be scheduled in a later microtask.
    

    3、catchError是then函数里抛出异常后会被执行

    async和await

    任何一个函数都可以加上async关键字,它会将函数的返回值自动转换成Future类型,await关键字只能用于返回值为Future类型的函数前面,如下代码:

    String _data = "";
    int clickCount = 0;
    Future<void> _buttonClick() async {
      var data = _data + "Started $clickCount: ${DateTime.now().toString()}";
      print("开始啦");
      await Future.delayed(new Duration(seconds: 2));
      data += " End $clickCount: ${DateTime.now().toString()} ";
      clickCount += 1;
      _data = data;
      print("结束啦 $_data");
    }
    
    // ignore: non_constant_identifier_names
    void function_main() async
    {
      for (int i=0;i<5;i++) {
        _buttonClick();
      }
      print("最终值 $_data");
    }
    

    最终的输出结果为:
    flutter: 开始啦
    flutter: 开始啦
    flutter: 开始啦
    flutter: 开始啦
    flutter: 开始啦
    flutter: 最终值
    flutter: 结束啦 Started 0: 2021-03-08 08:21:18.766893 End 0: 2021-03-08 08:21:20.772541
    flutter: 结束啦 Started 0: 2021-03-08 08:21:18.769211 End 1: 2021-03-08 08:21:20.772872
    flutter: 结束啦 Started 0: 2021-03-08 08:21:18.769887 End 2: 2021-03-08 08:21:20.773108
    flutter: 结束啦 Started 0: 2021-03-08 08:21:18.770458 End 3: 2021-03-08 08:21:20.773354
    flutter: 结束啦 Started 0: 2021-03-08 08:21:18.770836 End 4: 2021-03-08 08:21:20.773580
    这里分析一下代码的执行顺序,function_main()函数中连续调用_buttonClick()五次,按照之前的分析await之前的语句顺序执行五次,等待2秒之后,await之后的语句再顺序执行五次,所以await之后的data变量的值固定为Started 0: 2021-03-08 08:21:18.766893,clickCount由于每次执行后会加1,所以它的值会变化。上面函数其实等价于如下:

    Future<void> _buttonClick() async {
      var data = _data + "Started $clickCount: ${DateTime.now().toString()}";
      print("开始啦");
      return Future.delayed(new Duration(seconds: 2)).then((value) {
        data += " End $clickCount: ${DateTime.now().toString()} ";
        clickCount += 1;
        _data = data;
        print("结束啦 $_data");
      });
    }
    

    创建新的isolate

    当flutter应用启动时会创建一个默认的Root isolate(也可以理解为Main isolate),根据官方的说法,如果在Root isolate中做大量的耗时任务有可能被系统wathdog强杀,所以对于一些耗时任务可以创建一个新的iso去完成,当做完任务后再通过Port和Main isolate进行通信,请看如下代码:

    void function_main() async
    {
      print("开始啦");
      var receivePort = new ReceivePort();
      var sp1 = receivePort.sendPort;
      var sp2 = receivePort.sendPort;
      print("值 ${sp1==sp2}");
    
      await Isolate.spawn(entryPoint, receivePort.sendPort);
      print("结束啦");
      // 两个isolate之间通过SendPort来进行通信
      SendPort sendPort = await receivePort.first;
      print("结束啦1");
      // Send a message to the Isolate
      sendPort.send("hello");
      print("结束啦2");
    }
    entryPoint(SendPort sendPort) async {
      // Open the ReceivePort to listen for incoming messages (optional)
      print("进来啦1");
      // 创建跟当前线程关联的ReceivePort对象,一个线程由
      var port = new ReceivePort();
    
      // Send messages to other Isolates
      sendPort.send(port.sendPort);
      print("进来啦2");
    
      // Listen for messages (optional)
      await for (var data in port) {
        print("data $data");
      }
      print("进来啦3");
    }
    

    打印结果:
    flutter: 开始啦
    flutter: 值 true
    flutter: 结束啦
    flutter: 进来啦1
    flutter: 进来啦2
    flutter: 结束啦1
    flutter: 结束啦2
    flutter: data hello

    两个isolate之间基于Port实现的通信基于如下流程:


    image.png

    SendPort
    只能通过ReceivePort获取,用于Isolate之间的数据通信
    ReceivePort
    通过所在Isolate的函数内通过ReceivePort()创建。每一个ReceivePort可以有多个SendPort与之对应

    相关文章

      网友评论

          本文标题:Flutter之-dart多线程isolate(二)

          本文链接:https://www.haomeiwen.com/subject/qgzfqltx.html