gpt4 book ai didi

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

转载 作者:IT王子 更新时间:2023-10-29 07:16:18 34 4
gpt4 key购买 nike

我正在开发一个小型 Flutter 应用程序,我在其中使用 native 库进行一些计算。 dart 和 java(在 android 上)之间的通信是双向的,为此使用 methodChannels。我从 dart 调用 await in_channel.invokeMethod("someJavaMethod") 来开始计算。这会触发来自 Java 的 native 库的初始化。此 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 时,本地方法由本地 UI 线程处理。除非您使用线程池,否则这意味着 Dart 到 Native 方法是按顺序处理的。 等待每个本地方法的答案是没有意义的。或者,换句话说,同时启动两个 native 方法没有意义,因为它们将由单个 native 线程连续执行。 (请注意,您不应在 native 线程上执行耗时的操作,因为这会干扰它所做的其他事情,例如手势检测。)每个 Dart 到 native 方法都应返回其结果。

使用线程池

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

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

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

34 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com