Skip to content

从 JavaScript 使用 Kotlin 代码

根据所选的 JavaScript 模块系统,Kotlin/JS 编译器会生成不同的输出。 但总的来说,Kotlin 编译器会生成普通的 JavaScript 类、函数和属性,你可以从 JavaScript 代码中自由使用它们。 不过,有一些细微之处需要注意。

在 plain 模式下将声明隔离在单独的 JavaScript 对象中

如果你显式地将模块类型设置为 plain,Kotlin 会创建一个包含当前模块中所有 Kotlin 声明的对象。这样做是为了防止污染全局对象。这意味着对于模块 myModule,所有声明都可以通过 myModule 对象提供给 JavaScript。例如:

kotlin
fun foo() = "Hello"

可以在 JavaScript 中像这样调用此函数:

javascript
alert(myModule.foo());

当你将 Kotlin 模块编译为 JavaScript 模块(如 UMDbrowsernodejs 目标的默认设置)、ESMCommonJSAMD)时,像这样直接调用函数是不适用的。在这些情况下,你的声明将根据所选的 JavaScript 模块系统进行公开。例如,当使用 UMD、ESM 或 CommonJS 时,你的调用站点如下所示:

javascript
alert(require('myModule').foo());

有关 JavaScript 模块系统的更多信息,请参阅 JavaScript 模块

软件包结构

对于大多数模块系统(CommonJS、Plain 和 UMD),Kotlin 会将其软件包结构公开给 JavaScript。除非你在根软件包中定义声明,否则必须在 JavaScript 中使用完全限定名称。例如:

kotlin
package my.qualified.packagename

fun foo() = "Hello"

例如,当使用 UMD 或 CommonJS 时,你的调用站点可能如下所示:

javascript
alert(require('myModule').my.qualified.packagename.foo())

当使用 plain 作为模块系统设置时,调用站点将是:

javascript
alert(myModule.my.qualified.packagename.foo());

当以 ECMAScript 模块 (ESM) 为目标时,为了减小应用包的大小并匹配 ESM 软件包的典型布局,软件包信息不会被保留。在这种情况下,通过 ES 模块使用 Kotlin 声明的方式如下所示:

javascript
import { foo } from 'myModule';

alert(foo());

@JsName 注解

在某些情况下(例如,为了支持重载),Kotlin 编译器会在 JavaScript 代码中对生成的函数和属性名称进行修饰。要控制生成的名称,可以使用 @JsName 注解:

kotlin
// 模块 'kjs'
class Person(val name: String) {
    fun hello() {
        println("Hello $name!")
    }

    @JsName("helloWithGreeting")
    fun hello(greeting: String) {
        println("$greeting $name!")
    }
}

现在,你可以按照以下方式从 JavaScript 使用这个类:

javascript
// 如有必要,根据所选的模块系统导入 'kjs'
var person = new kjs.Person("Dmitry");   // 引用模块 'kjs'
person.hello();                          // 打印 "Hello Dmitry!"
person.helloWithGreeting("Servus");      // 打印 "Servus Dmitry!"

如果我们没有指定 @JsName 注解,对应函数的名称将包含一个根据函数签名计算出的后缀,例如 hello_61zpoe

请注意,在某些情况下,Kotlin 编译器不会应用名称修饰:

  • external 声明不会被修饰。
  • 继承自 external 类的非 external 类中任何重写的方法都不会被修饰。

@JsName 的参数必须是有效的标识符且为常量字符串文字。如果尝试向 @JsName 传递非标识符字符串,编译器将报错。以下示例会产生编译时错误:

kotlin
@JsName("new C()")   // 此处报错
external fun newC()

@JsExport 注解

此功能是实验性的。其设计可能会在未来的版本中发生变化。

通过将 @JsExport 注解应用于顶级声明(如类或函数),可以使该 Kotlin 声明在 JavaScript 中可用。该注解会使用 Kotlin 中给出的名称导出所有嵌套声明。它也可以使用 @file:JsExport 在文件级应用。

为了解决导出中的歧义(如具有相同名称的函数的重载),可以将 @JsExport 注解与 @JsName 结合使用,以指定生成的和导出的函数的名称。

目前,@JsExport 注解是使你的函数从 Kotlin 可见的唯一方法。

对于多平台项目,@JsExport 在通用代码中也可用。它仅在针对 JavaScript 目标进行编译时生效,并允许你导出非平台特定的 Kotlin 声明。

@JsStatic

此功能是实验性的。它可能随时被删除或更改。请仅用于评估目的。我们欢迎你在 YouTrack 上提供反馈。

@JsStatic 注解指示编译器为目标声明生成额外的 static 方法。这有助于你直接在 JavaScript 中使用 Kotlin 代码中的 static 成员。

你可以将 @JsStatic 注解应用于具名对象中定义的函数,以及在类和接口内部声明的伴生对象中的函数。如果你使用此注解,编译器既会生成对象的 static 方法,也会生成对象本身的实例方法。例如:

kotlin
// Kotlin
class C {
    companion object {
        @JsStatic
        fun callStatic() {}
        fun callNonStatic() {}
    }
}

现在,callStatic() 函数在 JavaScript 中是 static 的,而 callNonStatic() 函数则不是:

javascript
// JavaScript
C.callStatic();              // 正常运行,访问 static 方法
C.callNonStatic();           // 错误,在生成的 JavaScript 中不是 static 方法
C.Companion.callStatic();    // 实例方法仍然保留
C.Companion.callNonStatic(); // 唯一可行的方式

也可以将 @JsStatic 注解应用于对象或伴生对象的属性,使其 getter 和 setter 方法成为该对象或包含伴生对象的类中的 static 成员。

Experimental

使用 `BigInt` 类型表示 Kotlin 的 `Long` 类型

在编译为现代 JavaScript (ES2020) 时,Kotlin/JS 使用 JavaScript 的内置 BigInt 类型来表示 Kotlin Long 值。

要启用对 BigInt 类型的支持,你需要将以下编译器选项添加到你的 build.gradle(.kts) 文件中:

kotlin
// build.gradle.kts
kotlin {
    js {
        ...
        compilerOptions {
            freeCompilerArgs.add("-Xes-long-as-bigint")
        }
    }
}

此功能是实验性的。请在我们的问题跟踪器 YouTrack 中分享你的反馈。

在导出的声明中使用 Long

由于 Kotlin 的 Long 类型可以编译为 JavaScript 的 BigInt 类型,因此 Kotlin/JS 支持将 Long 值导出到 JavaScript。

要启用此功能:

  1. 允许在 Kotlin/JS 中导出 Long。将以下编译器选项添加到 build.gradle(.kts) 文件中的 freeCompilerArgs 属性中:
kotlin
// build.gradle.kts
kotlin {
   js {
       ...
       compilerOptions { 
           freeCompilerArgs.add("-XXLanguage:+JsAllowLongInExportedDeclarations")
       }
   }
}
  1. 启用 BigInt 类型。请参阅使用 BigInt 类型表示 Kotlin 的 Long 类型了解如何启用它。
Experimental

使用 `BigInt64Array` 类型表示 Kotlin 的 `LongArray` 类型

在编译为 JavaScript 时,Kotlin/JS 可以使用 JavaScript 的内置 BigInt64Array 类型来表示 Kotlin 的 LongArray 值。

要启用对 BigInt64Array 类型的支持,请将以下编译器选项添加到你的 build.gradle(.kts) 文件中:

kotlin
// build.gradle.kts
kotlin {
    js {
        ...
        compilerOptions {
            freeCompilerArgs.add("-Xes-long-as-bigint")
        }
    }
}

此功能是实验性的。请在我们的问题跟踪器 YouTrack 中分享你的反馈。

JavaScript 中的 Kotlin 类型

了解 Kotlin 类型如何映射到 JavaScript 类型:

KotlinJavaScript备注
Byte, Short, Int, Float, DoubleNumber
CharNumber数字表示字符的代码。
LongBigInt需要配置 -Xes-long-as-bigint 编译器选项
BooleanBoolean
StringString
ArrayArray
ByteArrayInt8Array
ShortArrayInt16Array
IntArrayInt32Array
CharArrayUInt16Array带有属性 $type$ == "CharArray"
FloatArrayFloat32Array
DoubleArrayFloat64Array
LongArrayBigInt64Array
BooleanArrayInt8Array带有属性 $type$ == "BooleanArray"
List, MutableListKtList, KtMutableList通过 KtList.asJsReadonlyArrayViewKtMutableList.asJsArrayView 公开一个 Array
Map, MutableMapKtMap, KtMutableMap通过 KtMap.asJsReadonlyMapViewKtMutableMap.asJsMapView 公开一个 ES2015 Map
Set, MutableSetKtSet, KtMutableSet通过 KtSet.asJsReadonlySetViewKtMutableSet.asJsSetView 公开一个 ES2015 Set
UnitUndefined当用作返回值类型时可导出,但用作形参类型时不可导出。
AnyObject
ThrowableError
enum class TypeType枚举条目公开为 static 类属性 (Type.ENTRY)。
可空 Type?`Typenull
除标记有 @JsExport 的类型外,所有其他 Kotlin 类型不支持包括 Kotlin 的无符号整数类型

此外,请务必了解:

  • Kotlin 为 kotlin.Intkotlin.Bytekotlin.Shortkotlin.Charkotlin.Long 保留了溢出语义。

  • Kotlin 在运行时无法区分数值类型(kotlin.Long 除外),因此以下代码可以正常运行:

    kotlin
    fun f() {
        val x: Int = 23
        val y: Any = x
        println(y as Float)
    }
  • Kotlin 在 JavaScript 中保留了延迟对象初始化。