从Swift访问Kotlin标准库

Bri*_*ian 5 android ios kotlin swift kotlin-multiplatform

在我的Kotlin Multiplatform项目中,我正在尝试访问kotlin-stdlibSwift中定义的Kotlin类型.

TL; DR:StdLib类型/方法似乎没有导致头定义,我想要一个不涉及编写大量样板代码的解决方案

我的情景

我在Kotlin中定义了一个接口......

interface MyKotlinInterface {
  fun run() : Sequence<String>
}
Run Code Online (Sandbox Code Playgroud)

...并在Swift中实现了这个接口......

class MySwiftClass : MyKotlinInterface {
  func run() -> KotlinSequence {
    // return sequenceOf("foo")
  }
}
Run Code Online (Sandbox Code Playgroud)

...我正在尝试创建一个Sequence但是没有来自kotlin.sequences可用包的方法(例如generateSequence).

实际上是否可以访问我在代码中定义的Kotlin框架类型或方法 - 如果是,如何?此外,如何在不编写样板代码的情况下实现这一目标?

更多详情

看一下生成的Objective-C头文件,我看到了我的类(显然)和基本Kotlin类型的定义.遗漏的基本上是标准库功能的一切(我关心所有Sequence相关的东西).

build.gradle.kts看起来像:

plugins {
  kotlin("multiplatform") version "1.3.0"
}

kotlin {
    targets { /* ... */ }

    sourceSets {

        getByName("commonMain") {
            dependencies {
                api("org.jetbrains.kotlin:kotlin-stdlib-common")
            }
        }

        // ...

        getByName("iosMain") {
            dependencies {
                api("org.jetbrains.kotlin:kotlin-stdlib")
            }
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

kotlin-stdlib定义为iOS目标的依赖关系,我希望这些实际上可以从Swift获得.

最小的工作示例

https://github.com/panzerfahrer/so-mwe-kotlin-mpp-swift

目前的解决方案

我想出的唯一解决方案是为iOS目标编写所需的功能:

fun <T : kotlin.Any> generateSequence(nextFunction: () -> T?): kotlin.sequences.Sequence<T> = kotlin.sequences.generateSequence(nextFunction)
Run Code Online (Sandbox Code Playgroud)

很好用但是非常不满意,因为它需要大量的样板代码.此外,扩展功能无法以这种方式提供,并且需要更多样板代码甚至重写标准库的部分.

期望的解决方案

我希望尽可能避免编写样板代码.我实际上只关心的是,(在我的情况下)Sequence可以从Swift完全访问.我的感觉是,使编译器生成标准库功能的选定或所有头定义就足够了.

Eug*_*nko 1

您的 Kotlin 代码中真的需要惰性计算(又名序列)吗?如果没有,我建议使用List<T>(它直接映射到 Swift)。

对于Sequence实现,解决方法可能是从 Kotlin 库中导出工厂函数,例如,您可以声明如下函数

fun <T : kotlin.Any> generateSequence(nextFunction: () -> T?)
    = kotlin.sequences.generateSequence(nextFunction)
Run Code Online (Sandbox Code Playgroud)

Sequence您可以选择与您的用例相匹配的任何其他工厂函数。

总的来说,Kotlin 标准库中的函数太多了。将它们全部导出到 Swift 会在二进制文件中创建太多无用的符号并增加编译时间。