NativeCallable<T extends Function>.listener 构造函数
- @DartRepresentationOf("T") Function callback
构建一个可以从任何线程调用的 NativeCallable。
当原生代码调用函数 nativeFunction 时,参数将通过 SendPort 发送到创建 NativeCallable 的 Isolate,并调用回调。
原生代码不会等待回调的响应,因此只支持返回 void 的函数。
回调将在未来某个时间被调用。原生调用者不能假设回调将立即运行。传递给回调的资源(例如 malloc 分配的内存指针或输出参数)必须在调用完成前有效。
当不再需要此回调时,必须调用 close。创建回调的 Isolate 将保持活动状态,直到调用 close。在调用 NativeCallable.close 之后,从原生代码调用 nativeFunction 将导致未定义的行为。
例如
import 'dart:async';
import 'dart:ffi';
import 'package:ffi/ffi.dart';
// Processes a simple HTTP GET request using a native HTTP library that
// processes the request on a background thread.
Future<String> httpGet(String uri) async {
final uriPointer = uri.toNativeUtf8();
// Create the NativeCallable.listener.
final completer = Completer<String>();
late final NativeCallable<NativeHttpCallback> callback;
void onResponse(Pointer<Utf8> responsePointer) {
completer.complete(responsePointer.toDartString());
calloc.free(responsePointer);
calloc.free(uriPointer);
// Remember to close the NativeCallable once the native API is
// finished with it, otherwise this isolate will stay alive
// indefinitely.
callback.close();
}
callback = NativeCallable<NativeHttpCallback>.listener(onResponse);
// Invoke the native HTTP API. Our example HTTP library processes our
// request on a background thread, and calls the callback on that same
// thread when it receives the response.
nativeHttpGet(uriPointer, callback.nativeFunction);
return completer.future;
}
// Load the native functions from a DynamicLibrary.
final DynamicLibrary dylib = DynamicLibrary.process();
typedef NativeHttpCallback = Void Function(Pointer<Utf8>);
typedef HttpGetFunction = void Function(
Pointer<Utf8>, Pointer<NativeFunction<NativeHttpCallback>>);
typedef HttpGetNativeFunction = Void Function(
Pointer<Utf8>, Pointer<NativeFunction<NativeHttpCallback>>);
final nativeHttpGet =
dylib.lookupFunction<HttpGetNativeFunction, HttpGetFunction>(
'http_get');
实现
factory NativeCallable.listener(
@DartRepresentationOf("T") Function callback) {
throw UnsupportedError("NativeCallable cannot be constructed dynamically.");
}