我正在开发一个小型 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/