flutter - 在执行 invokeMethod() 时如何监听异步响应?

标签 flutter dart

我正在开发一个小型 flutter 应用程序,我使用 native 库进行一些计算。 dart 和 java(在 android 上)之间的通信是双向的,并使用 methodChannels 来实现此目的。 我从 dart 调用 wait in_channel.invokeMethod("someJavaMethod") 来开始计算。这会触发 Java 本地库的 init。此 init 的结果作为异步 JNI 调用返回,然后触发 out_channel.invokeMethod("someDartMethod")。

我的计划是将 out_channel 绑定(bind)到本地 dart 广播流,以便我可以调用 someJavaMethod,然后等待 myMethodStream.where((m) => m.method == "someDartMethod")...

问题是“someDartMethod”可能在“someJavaMethod”调用返回之前出现。

我所拥有的组合代码示例:

    static const MethodChannel _channel_in = const 
    MethodChannel('native_lib_wrapper_out'); 
    static const MethodChannel _channel_out = const 
    MethodChannel('native_lib_wrapper_in');
    final StreamController<MethodCall> _methodStreamController = new 
    StreamController.broadcast();

    NativeLibWrapper._() {
          _channel_in.setMethodCallHandler((MethodCall call) {
              _methodStreamController.add(call);
              return;
         });
    }


    Future<Map<dynamic,dynamic>> initLib(String id, String filePath) 
      async {
        Map<dynamic,dynamic> ret;

        ret = await _channel_out.invokeMethod("initLib",  <String, 
        dynamic> { // data to be passed to the function
                    'id': id,
                    'filePath': filePath,
                  });
        print('initLib - invokeMethod done. wait for stream');
        if(ret["status"] == 0) {
          await NativeLibWrapper.instance._methodStream
                .where((m) => m.method == "libInitEnded")
                .map((m){
                    var args = m.arguments;
                    ret = args;
                  }).first;
        }
        return ret;
    }

我希望代码在我的流上获取方法调用 libInitEnded,然后它应该在该点之后返回,但它持续卡在流上的等待中,从日志来看,libInitEnded 在打印之前被调用在中间。

那么有没有更好的方法来构建它?它不会是唯一来回的方法,所以我希望为此获得一个良好稳定的解决方案。

最佳答案

一个 channel

您应该只需要一个 channel 。无需输入和输出 channel 。两端都可以通过一个 channel 调用另一端的操作。

只有一个 UI 线程

当您从 Dart 调用 Native 时, native 方法由 native UI 线程处理。除非您使用线程池,否则这意味着 Dart 到 Native 方法是按顺序处理的。 等待每个 native 方法的答案是没有意义的。或者,换句话说,同时启动两个 native 方法是没有意义的,因为它们将由单个 native 线程连续执行。 (请注意,您不应在 native 线程上执行耗时的操作,因为这会干扰它执行的其他操作,例如手势检测。)每个 Dart 到 native 方法都应返回其结果。

使用线程池

如果一次单个线程/单个方法调用 Not Acceptable ,请考虑在 native 端使用线程池。现在您可以有多个正在运行的方法,因为有多个执行线程。现在您应该像通过套接字与服务器通信一样设计您的调用/响应。客户端为每个请求提供一个“调用 ID”。每个方法仅返回一个 bool 值,表明请求已排队。请求完成后,另一端调用“done”方法,传递原始 id 和结果。然后,调用者可以将响应 ID 与请求 ID 进行匹配,并适本地处理响应(并取消任何启动以检测超时的计时器)。请注意,响应可以按任何顺序到达,但通过 id 与其请求进行匹配。

在 Android 上,您必须在 UIThread 上调用 Dart 的 native 方法。如果您从工作线程调用“done”方法,则需要向主循环器发布一个Runnable lambda。

关于flutter - 在执行 invokeMethod() 时如何监听异步响应?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/57292064/

相关文章:

flutter - 在Flutter中异步加载图片

dart - NumberFormat 未找到 flutter

android - flutter View : AppBarLayout$ScrollingViewBehavior

Dart - 隔离跨窗口通信

dart - 在 AngularDart 中的 NgController 和 NgComponent 之间发送消息的标准方式是什么?

嵌套元素中的 Dartlang polymer 点击事件

flutter 如何获取最后的PageView滚动方向

android - VS 代码 : Failed to install the following Android SDK packages as some licences have not been accepted

flutter - 在 Flutter 中结合 SingleChildScrollView 和 PageView

dart - 以编程方式创建 Flex div