如何将字节从 Swift (iOS) 传递到 Kotlin 通用模块?

How to pass bytes from Swift (iOS) to Kotlin common module?

为了在 android 和 iOS 应用程序之间共享协议的实现,我正在试验 Kotlin Multiplatform。我按照 here.

中的描述设置了一个基本的多平台项目

它定义了共享模块中的公共代码...

fun createApplicationScreenMessage() : String {
  return "Kotlin Rocks on ${platformName()}"
}

... 可用于 iOS 项目 CommonKt.createApplicationScreenMessage()

现在想在common模块做IO操作。为此我找到了 Kotlinx-io 并且可以在公共模块中使用它。

但是我如何正确设计 kotlin 代码和 swift 代码之间的 api,以便我可以将 Swift 中的 InputStream/ByteArray/ByteReadPacket 等价物传递给 kotlin 模块?

例如类似这样的 kotlinx-io 类型,如 ByteReadPacket:

Kotlin 通用模块:

class ProtocolReader{
    public fun parse(packet: ByteArray): ParsedMessage {
    //parse data
    } 
}

Swift iOS 应用

var byteArray = [UInt8](characteristicData)
let reader = ProtocolReader()
reader.parse(byteArray)

此示例不起作用,因为 swiftbyteArray 无法与 KotlinByteArray 互操作。

我该如何实现?我需要为每个平台定义 api 端点吗?在这种情况下,在 Kotlin 多平台项目的 ios 模块中?或者是否有辅助方法从 ios 数据类型创建 kotlinx-io 数据类型?

您传递给通用代​​码的所有内容都需要与平台无关,因此您要么使用 expect/actual mechanism 定义模型,要么将 swift 数据类型映射到 kotlin 数据类型。

我不流利 swift,但你可以这样做:

let swiftByteArray : [UInt8] = []
let intArray : [Int8] = swiftByteArray
    .map { Int8(bitPattern: [=10=]) }
let kotlinByteArray: KotlinByteArray = KotlinByteArray.init(size: Int32(swiftByteArray.count))
for (index, element) in intArray.enumerated() {
    kotlinByteArray.set(index: Int32(index), value: element)
}

查看生成的互操作性 headers 有时也有帮助。

科特林字节:

__attribute__((objc_runtime_name("KotlinByte")))
__attribute__((swift_name("KotlinByte")))
@interface MainByte : MainNumber
- (instancetype)initWithChar:(char)value;
+ (instancetype)numberWithChar:(char)value;
@end;

KotlinByteArray:

__attribute__((objc_subclassing_restricted))
__attribute__((swift_name("KotlinByteArray")))
@interface MainKotlinByteArray : KotlinBase
+ (instancetype)arrayWithSize:(int32_t)size 
__attribute__((swift_name("init(size:)")));
+ (instancetype)arrayWithSize:(int32_t)size init:(MainByte *(^)(MainInt *))init 
__attribute__((swift_name("init(size:init:)")));
+ (instancetype)alloc __attribute__((unavailable));
+ (instancetype)allocWithZone:(struct _NSZone *)zone. __attribute__((unavailable));
- (int8_t)getIndex:(int32_t)index __attribute__((swift_name("get(index:)")));
- (MainKotlinByteIterator *)iterator __attribute__((swift_name("iterator()")));
- (void)setIndex:(int32_t)index value:(int8_t)value 
__attribute__((swift_name("set(index:value:)")));
@property (readonly) int32_t size;
@end;