从 Swift 访问 Kotlin 标准库

Posted

技术标签:

【中文标题】从 Swift 访问 Kotlin 标准库【英文标题】:Access Kotlin Standard Library from Swift 【发布时间】:2019-04-05 20:03:03 【问题描述】:

在我的 Kotlin Multiplatform 项目中,我试图从 Swift 访问在 kotlin-stdlib 中定义的 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)

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

所需的解决方案

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

【问题讨论】:

同时,我找到了一个相关的Github issue,基本上涵盖了这个案例。 你的 Kotlin 代码中真的需要惰性计算(又名序列)吗?如果不是,我会推荐使用 List 代替(它映射到 Swift)。对于序列实现,一种解决方法可能是从 Kotlin 库中导出工厂函数,以帮助在 Swift 端构建序列 【参考方案1】:

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

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

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

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

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

【讨论】:

这是我目前的解决方案(见问题的下部):-P 是的,也许使用List&lt;T&gt; 而不是Sequence 是有意义的。使用 Sequence&lt;T&gt; 时,您的示例 repo 并没有赢得您的青睐

以上是关于从 Swift 访问 Kotlin 标准库的主要内容,如果未能解决你的问题,请参考以下文章

Kotlin标准库函数 ④ ( takeIf 标准库函数 | takeUnless 标准库函数 )

Kotlin标准库函数 ③ ( with 标准库函数 | also 标准库函数 )

Kotlin标准库函数总结 ( apply 函数 | let 函数 | run 函数 | with 函数 | also 函数 | takeIf 函数 | takeUnless 函数 )

Kotlin标准库函数总结 ( apply 函数 | let 函数 | run 函数 | with 函数 | also 函数 | takeIf 函数 | takeUnless 函数 )

Kotlin标准库函数 ② ( run 标准库函数 | run 函数传入 Lambda 表达式作为参数 | run 函数传入函数引用作为参数 )

如何在 Kotlin 标准库(多平台)上获取当前的 unixtime