Flutter计算功能的上下文/范围是什么

时间:2018-08-30 19:48:41

标签: flutter

我正在学习Flutter的compute函数。我对此的理解是,它用于将计算繁重的工作卸载到另一个线程,以免阻塞UI线程。我遇到了一些我不太了解的行为。考虑下面的应用程序。

writer = new PrintWriter(new OutputStreamWriter(new FileOutputStream("PF500.in"), "CHARSET_NAME"));

_testCompute函数的输出为:

import 'package:flutter/material.dart';
import 'package:flutter/foundation.dart';

void main() => runApp(new MyApp());

class StringValues {
  static String foo = 'Foo';
  static String bar;
}

String _calculate(String value) {
  return value + ' ' + (StringValues.foo ?? 'undefined') + ' ' + (StringValues.bar ?? 'undefined');
}

class MyApp extends StatelessWidget {

  @override
  Widget build(BuildContext context) {
    return new MaterialApp(
      title: 'Flutter Demo',
      theme: new ThemeData(
        primarySwatch: Colors.blue,
      ),
      home: new MyHomePage(title: 'Flutter Demo Home Page'),
    );
  }
}

class MyHomePage extends StatefulWidget {
  MyHomePage({Key key, this.title}) : super(key: key);
  final String title;

  @override
  _MyHomePageState createState() => new _MyHomePageState();
}

class _MyHomePageState extends State<MyHomePage> {

  void _testCompute() async {
    String result1 = _calculate('values are:');
    String result2 = await compute(_calculate, 'values are:');

    StringValues.bar = 'Bar';

    String result3 = _calculate('values are:');
    String result4 = await compute(_calculate, 'values are:');

    print(result1);
    print(result2);
    print(result3);
    print(result4);
  }

  @override
  Widget build(BuildContext context) {
    return new Scaffold(
      appBar: new AppBar(
        title: new Text(widget.title),
      ),
      floatingActionButton: new FloatingActionButton(
        onPressed: _testCompute,
        child: new Icon(Icons.print),
      ),
    );
  }
}

它可以编译。我希望values are: Foo undefined values are: Foo undefined values are: Foo Bar values are: Foo undefined result4相同。谁能解释为什么不是呢?我也尝试使用here中描述的使用'globals'的另一种方法,但是结果是相同的。

1 个答案:

答案 0 :(得分:2)

compute()创建一个新的隔离。 以这种方式启动的隔离程序除了代码外什么都不共享。
就像您将以_calculate(而不是main)作为入口点从同一项目开始新流程一样。

compute使用SendPort / ReceivePort组合将'values are:'传递到isolate,然后在隔离开始运行时将该值作为参数传递给_calculate

这些值是按值传递的,因此传递对象引用并在另一侧进行更改不会对发送方产生任何影响。

同样,隔离区不共享任何状态,只共享代码,它们使用SendPort / ReceivePort进行通信,以在主隔离区和“ compute”隔离区之间来回传递值。