Dart Isolates As Workers

后端 未结 4 642
不思量自难忘°
不思量自难忘° 2021-02-09 20:35

Edited to make the question more clear.

I am trying to work with Isolates (or Web Workers) in Dart. The only ways I can find to communicate between the

相关标签:
4条回答
  • 2021-02-09 20:48

    As of Dart 1.0, you can use isolates like this:

    import 'dart:isolate';
    import 'dart:async';
    
    void doStuff(SendPort sendPort) {
      print('hi from inside isolate');
      ReceivePort receivePort = new ReceivePort();
      sendPort.send(receivePort.sendPort);
    
      receivePort.listen((msg) {
        print('Received in isolate: [$msg]');
        sendPort.send('ECHO: $msg');
      });
    
    }
    
    void main() {
      SendPort sendPort;
    
      ReceivePort receive = new ReceivePort();
      receive.listen((msg) {
        if (sendPort == null) {
          sendPort = msg;
        } else {
          print('From isolate: $msg');
        }
      });
    
      int counter = 0;
    
      Isolate.spawn(doStuff, receive.sendPort).then((isolate) {
        new Timer.periodic(const Duration(seconds:1), (t) {
          sendPort.send('Count is ${counter++}');
        });
      });
    }
    
    0 讨论(0)
  • 2021-02-09 20:48

    You can now use the MessageBox class to communicate the other way around. This code sends a message from the Isolate code as soon as it receives the Sink end of the MessageBox. Main thread receives the messages sent from the Isolate and prints it on the console of Dartium. Once you receive the Sink you can launch your game logic and send updates using the sink object received.

    import 'dart:html';
    import 'dart:isolate';
    
    void main() {
      IsolateSink isolateSink = streamSpawnFunction(myIsolateEntryPoint);
      MessageBox isolateMessageBox = new MessageBox();
      isolateSink.add(isolateMessageBox.sink);
      isolateMessageBox.stream.listen((String data) {
        print(data);
      });
    }
    
    void myIsolateEntryPoint() {
      stream.listen((IsolateSink messageBoxSink) {
        messageBoxSink.add("Test");
      });
    }
    
    0 讨论(0)
  • 2021-02-09 20:53

    WARNING: this code only works on very old versions of Dart. It does not work on Dart 1.0 or later.

    As you mention to post messages to a isolate you need to have a handle on it's sendport.

    #import('dart:isolate');
    
    main() {
      SendPort sendPort = spawnFunction(doWork);
      sendPort.call("hey 1").then((String res) => print("result was: [$res]"));
      sendPort.call("hey 2").then((String res) => print("result was: [$res]"));
    }
    
    doWork() {
      port.receive((msg, reply) {
        msg = "msg $msg";
        reply.send(msg);
      });
    }
    

    however since the Dart main thread is itself an isolate you can send data to it by using the global port function:

    #import('dart:isolate');
    #import('dart:io');
    
    main() {
       port.receive((data, reply) {
           // in here you can access objects created in the main thread
           print("handle [${data['text']}] for index ${data['index']}");
       });
    
       SendPort workPort = spawnFunction(doWork);
       workPort.send("msg", port.toSendPort());
    }
    
    doWork() {
       port.receive((msg, reply) {
          int i = 0;
          new Timer.repeating(1000, (Timer timer) {
             i++;
             var data = {
                "text": "$msg $i",
                "index": i
             };
             print("sending $data");
             reply.send(data);
          });
       });
    }
    

    Note there are certain limits about what can be send back and forth between isolates and also currently isolates act differently in JS and on the VM. The current limitations are well described here.

    0 讨论(0)
  • 2021-02-09 21:08

    Here is an example where parent creates two isolates and then two isolates also talk to each other along with the parent process.

    Parent Code:

    import 'dart:isolate';
    import 'dart:html';
    import 'dart:async';
    
    main() {
      querySelector('#output').text = 'Your Dart app is running.';
      int counter = 0;
    
      // Parent - Child 1
      SendPort csendPort1;
      ReceivePort receivePort1 = new ReceivePort();
      // Parent - Child 2
      SendPort csendPort2;
      ReceivePort receivePort2 = new ReceivePort();
      // Child1 - Child2
      SendPort csendPort11;
      SendPort csendPort12;
    
      // Child 1
      receivePort1.listen((msg) {
        if (csendPort1 == null) {
          csendPort1 = msg;
        } else if (csendPort11 == null) {
          csendPort11 = msg;
        } else {
          print('$msg');`enter code here`
        }
      });
    
      bool child1 = false;
      Isolate.spawnUri(Uri.parse('child.dart'), [], receivePort1.sendPort).then((isolate) {
        print('Child 1 isolate spawned');
        new Timer.periodic(const Duration(milliseconds: 500), (t) {
          if (csendPort11 != null && csendPort12 != null && child1 == false) {
            child1 = true;
            csendPort12.send(csendPort11);
          } else {
            csendPort1.send('Parent-Child1: ${counter++}');
          }
        });
      });
    
      // Child 2
      receivePort2.listen((msg) {
        if (csendPort2 == null) {
          csendPort2 = msg;
        } else if (csendPort12 == null) {
          csendPort12 = msg;
        } else {
          print('$msg');
        }
      });
    
      bool child2 = false;
      Isolate.spawnUri(Uri.parse('child.dart'), [], receivePort2.sendPort).then((isolate) {
        print('Child 2 isolate spawned');
        new Timer.periodic(const Duration(milliseconds: 500), (t) {
          if (csendPort11 != null && csendPort12 != null && child2 == false) {
            child2 = true;
            csendPort11.send(csendPort12);
          } else {
            csendPort2.send('Parent-Child2: ${counter++}');
          }
        });
      });
    }
    

    Child Code:

    import 'dart:isolate';
    import 'dart:async';
    
    int pcounter = 0;
    int ccounter = 0;
    
    SendPort csendPort;
    void handleTimeout() {
      csendPort.send("${ccounter++}");
    }
    
    main(List<String> args, SendPort psendPort) {
      // Parent Comm
      ReceivePort creceivePort1 = new ReceivePort();
      psendPort.send(creceivePort1.sendPort);
    
      creceivePort1.listen((msg) {
        psendPort.send('Child-Parent: ${pcounter++} - ${msg}');
      });
    
      // Child-Child Comm
      ReceivePort creceivePort2 = new ReceivePort();
      psendPort.send(creceivePort2.sendPort);
    
      creceivePort2.listen((msg) {
        if (csendPort == null) {
          csendPort = msg;
          csendPort.send("${ccounter++}");
        } else {
          print("Child-Child: $msg");
          var duration = const Duration(milliseconds: 2000);
          new Timer(duration, handleTimeout);
        }
      });
    }
    

    HTML Code:

    <!DOCTYPE html>
    <html>
    <head>
        <meta charset="utf-8">
        <meta http-equiv="X-UA-Compatible" content="IE=edge">
        <meta name="viewport" content="width=device-width, initial-scale=1.0">
        <meta name="scaffolded-by" content="https://github.com/google/stagehand">
        <title>WebIsolateTest</title>
        <link rel="stylesheet" href="styles.css">
        <script defer src="main.dart" type="application/dart"></script>
        <script defer src="packages/browser/dart.js"></script>
    </head>
    
    <body>
    
      <div id="output"></div>
    
    </body>
    </html>
    
    0 讨论(0)
提交回复
热议问题