5

在我的 Kotlin Multiplatform 项目中,我试图访问kotlin-stdlib从 Swift 中定义的 Kotlin 类型。

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

我的场景

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

interface MyKotlinInterface {
  fun run() : Sequence<String>
}

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

class MySwiftClass : MyKotlinInterface {
  func run() -> KotlinSequence {
    // return sequenceOf("foo")
  }
}

...我正在尝试创建一个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")
            }
        }
    }
}

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)

这工作正常,但非常不令人满意,因为它需要大量样板代码。此外,无法以这种方式提供扩展功能,并且需要更多样板代码甚至重写标准库的部分内容。

所需的解决方案

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

4

1 回答 1

1

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

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

fun <T : kotlin.Any> generateSequence(nextFunction: () -> T?)
    = kotlin.sequences.generateSequence(nextFunction)

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

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

于 2019-02-15T14:36:42.017 回答