4

我现在正在学习 Dart,并且正在玩 Dart 与 C 的互操作性。我可以使用带有两个 int 参数的 C 方法。下面的代码:

你好飞镖

import 'dart:ffi' as ffi;

typedef sum_func = ffi.Int32 Function(ffi.Int32 a, ffi.Int32 b);
typedef Sum = int Function(int a, int b);
...
final dylib = ffi.DynamicLibrary.open(path);
final sumPointer = dylib.lookup<ffi.NativeFunction<sum_func>>('sum');

final sum = sumPointer.asFunction<Sum>();
print('3 + 5 = ${sum(3, 5)}');

你好ç

int sum(int a, int b){
    return a + b;
}

你好.h

int add(int x, int y)

你好.def

LIBRARY   hello
EXPORTS
   sum

这一切都很好,但我也想要一个maxC 方法,它将一个 int 数组作为输入并返回最大的数字。我怎样才能做到这一点?我已经在 C 中实现了所有必需的代码,但我不确定如何将它与 Dart“链接”。有人可以帮我吗?

4

1 回答 1

4

首先,我真的想说我不是 C 程序员,尤其是在涉及指针时,我什至没有假装我完全理解如何以最佳方式完成这类事情。

有了这个,这里是我的解决方案,基于在此处找到的原语示例: https ://github.com/dart-lang/samples/tree/master/ffi/primitives

原语.dart

import 'dart:ffi';
import 'dart:io' show Platform;

import 'package:ffi/ffi.dart';

typedef max_func = Int32 Function(Pointer<Int32> list, Int32 size);
typedef Max = int Function(Pointer<Int32> list, int size);

void main() {
  var path = './primitives_library/libprimitives.so';
  if (Platform.isMacOS) path = './primitives_library/libprimtives.dylib';
  if (Platform.isWindows) path = r'primitives_library\Debug\primitives.dll';
  final dylib = DynamicLibrary.open(path);

  final list = [1, 5, 3, 59030, 131000, 0];
  final listPtr = intListToArray(list);

  final maxPointer = dylib.lookup<NativeFunction<max_func>>('max');
  final max = maxPointer.asFunction<Max>();
  print('${max(listPtr, list.length)}'); // 131000
  malloc.free(listPtr);
}

Pointer<Int32> intListToArray(List<int> list) {
  final ptr = malloc.allocate<Int32>(sizeOf<Int32>() * list.length);
  for (var i = 0; i < list.length; i++) {
    ptr.elementAt(i).value = list[i];
  }
  return ptr;
}

原语.h

int max(int *listPtr, int size);

原语.c

#include "primitives.h"

int max(int *listPtr, int size)
{
  int currentMax = *listPtr;

  for (int i = 0; i < size; i++)
  {
    if (currentMax < *listPtr)
    {
      currentMax = *listPtr;
    }
    listPtr++;
  }

  return currentMax;
}
于 2020-03-27T20:09:21.343 回答