mirror of
https://github.com/mamoe/mirai.git
synced 2025-01-23 22:30:47 +08:00
308 lines
5.9 KiB
Markdown
308 lines
5.9 KiB
Markdown
# Mirai - Kotlin And Java
|
||
|
||
|
||
本章介绍部分 Kotlin 定义对应的 Java 定义,以帮助 Java 使用者理解 Mirai 的源代码。
|
||
|
||
每部分第一个代码块为 Kotlin 代码,第二个代码块为 Java 代码。
|
||
|
||
预计阅读时间:5 分钟
|
||
|
||
#### 通用
|
||
- Kotlin 的定义都默认是 `public` 和 `final`
|
||
- Kotlin 不需要句末分号,通常以换行作为一个语句的结束
|
||
|
||
#### `class`
|
||
```kotlin
|
||
class A
|
||
```
|
||
```java
|
||
public final class A {
|
||
}
|
||
```
|
||
|
||
#### 构造器定义
|
||
|
||
以下几种 Kotlin 定义是等价的。
|
||
|
||
```kotlin
|
||
class A {
|
||
private val value: String
|
||
|
||
constructor(value: String) {
|
||
this.value = value
|
||
}
|
||
|
||
constructor(integer: Int) {
|
||
this.value = integer.toString()
|
||
}
|
||
}
|
||
```
|
||
|
||
```kotlin
|
||
class A(val value: String) { // 类定义后面的括号表示主构造器
|
||
constructor(integer: Int) : this(integer.toString())
|
||
}
|
||
```
|
||
|
||
对应的 Java 定义为:
|
||
```java
|
||
public final class A {
|
||
private final String value;
|
||
|
||
public A(String value) {
|
||
this.value = value;
|
||
}
|
||
|
||
public A(int integer) {
|
||
this.value = String.valueOf(integer);
|
||
}
|
||
}
|
||
```
|
||
|
||
通常 Kotlin class 都会有一个主构造器。
|
||
|
||
#### 构造器调用
|
||
|
||
Kotlin 不需要 `new` 关键字。
|
||
|
||
```kotlin
|
||
val a = A("test")
|
||
```
|
||
|
||
```java
|
||
A a = new A("test");
|
||
```
|
||
|
||
#### 函数
|
||
```kotlin
|
||
class A {
|
||
fun test(string: String): Int = 1
|
||
}
|
||
```
|
||
|
||
```java
|
||
public final class A {
|
||
public int test(String string) {
|
||
return 1;
|
||
}
|
||
}
|
||
```
|
||
|
||
#### 属性 `val`
|
||
- Kotlin 的 `val` 是不可变的,只能被赋值一次
|
||
- 编译器为 `val` 创建 `getter`
|
||
|
||
```kotlin
|
||
class A {
|
||
val value: String = "test"
|
||
}
|
||
```
|
||
|
||
```java
|
||
public final class A {
|
||
private final String value = "test";
|
||
public final String getValue() {
|
||
return value;
|
||
}
|
||
}
|
||
```
|
||
|
||
#### 属性 `var`
|
||
- Kotlin 的 `var` 相较于 `val` 是可变的,可以被多次赋值。
|
||
- 编译器为 `var` 创建 `getter` 和 `setter`
|
||
|
||
```kotlin
|
||
class A {
|
||
var value: String = "test"
|
||
}
|
||
```
|
||
|
||
```java
|
||
public final class A {
|
||
private String value = "test";
|
||
public final String getValue() {
|
||
return value;
|
||
}
|
||
public final String setValue(String value) {
|
||
this.value = value;
|
||
}
|
||
}
|
||
```
|
||
|
||
#### 顶层定义和 `const`
|
||
- Kotlin 的定义不一定需要在 `class` 中,允许直接存在于文件中的「顶层函数」和「顶层属性」
|
||
- `XXX.kt` 中的顶层定义会被编译为名为 `XXXKt` 的 `class`
|
||
- 顶层定义会被编译为 `static`
|
||
- `const` 可以修饰一个属性,编译器会把它编译为 Java 静态字段。
|
||
|
||
```kotlin
|
||
// Test.kt
|
||
val x: String = "xx"
|
||
|
||
const val CONST_VALUE: String = "cc"
|
||
|
||
fun foo() { }
|
||
```
|
||
|
||
```java
|
||
// TestKt.java
|
||
public final class TestKt {
|
||
public static final String CONST_VALUE = "cc"; // const val 没有 getter
|
||
|
||
private static final String x = "xx";
|
||
public static String getX(){
|
||
return x;
|
||
}
|
||
|
||
public static void foo() { }
|
||
}
|
||
```
|
||
|
||
#### 单例对象
|
||
- Kotlin `object` 定义一个单例对象
|
||
|
||
```kotlin
|
||
object Test
|
||
```
|
||
|
||
```java
|
||
public final class Test {
|
||
public static final Test INSTANCE = new Test();
|
||
|
||
private Test() {}
|
||
}
|
||
```
|
||
|
||
#### 静态
|
||
|
||
```kotlin
|
||
object Test {
|
||
val x = "x" // public String getX()
|
||
@JvmField val y = "y" // public static final String y;
|
||
@JvmStatic val z = "z" // public static String getZ()
|
||
}
|
||
```
|
||
|
||
```java
|
||
public final class Test {
|
||
public static final Test INSTANCE = new Test();
|
||
private Test() {}
|
||
|
||
private final String x = "x"; // val
|
||
public String getX() {
|
||
return x;
|
||
}
|
||
|
||
public static final String y = "y"; // @JvmField val
|
||
|
||
private final String z = "z"; // @JvmStatic val
|
||
public static String getZ() {
|
||
return z;
|
||
}
|
||
}
|
||
```
|
||
|
||
|
||
#### 静态
|
||
|
||
- Kotlin 没有 `static` 关键字,但可以通过 `@JvmStatic` 将一个函数编译为 `static`
|
||
|
||
```kotlin
|
||
object Test {
|
||
fun a() { }
|
||
@JvmStatic
|
||
fun b() { }
|
||
}
|
||
```
|
||
|
||
```java
|
||
public final class Test {
|
||
public static final Test INSTANCE = new Test();
|
||
private Test() {}
|
||
|
||
public void a() { }
|
||
public static void b() { }
|
||
}
|
||
```
|
||
|
||
#### 伴生对象
|
||
|
||
- `class` 可以拥有 `companion object`
|
||
- 伴生对象内的 `@JvmField` 定义将会被编译到外部 `class`
|
||
- 伴生对象内的 `@JvmStatic` 函数以成员方法编译到伴生对象,然后以静态方法编译到外部 `class`
|
||
|
||
```kotlin
|
||
class Test {
|
||
companion object {
|
||
@JvmField
|
||
val CONST: String = ""
|
||
|
||
fun a() { }
|
||
@JvmStatic
|
||
fun b() { }
|
||
}
|
||
}
|
||
```
|
||
|
||
```java
|
||
public final class Test {
|
||
public static final Companion Companion = new Companion();
|
||
public static final String CONST = "";
|
||
|
||
public static void b() {
|
||
Companion.b();
|
||
}
|
||
|
||
public static final class Companion {
|
||
public void a() { }
|
||
public void b() { }
|
||
}
|
||
}
|
||
```
|
||
|
||
#### 协程
|
||
|
||
Kotlin 协程是语言级特性,`suspend` 修饰的函数会在编译期被处理。
|
||
|
||
```kotlin
|
||
class A {
|
||
suspend fun getValue(): String { /* ... */ }
|
||
}
|
||
```
|
||
|
||
```java
|
||
public final class A {
|
||
public Object getValue(Continuation<? super String> $completion) {
|
||
// 由 Kotlin 编译器生成非常复杂的方法体
|
||
}
|
||
}
|
||
```
|
||
|
||
`$completion` 参数类似于一个回调。需要熟悉 Kotlin 协程原理才能实现。为帮助 Java 用户,mirai 使用编译器插件处理 `suspend` 函数。
|
||
|
||
```kotlin
|
||
class A {
|
||
@JvmBlockingBridge
|
||
suspend fun getValue(): String { /* ... */ }
|
||
}
|
||
```
|
||
|
||
```java
|
||
public final class A {
|
||
public Object getValue(Continuation<? super String> $completion) {
|
||
// 由 Kotlin 编译器生成非常复杂的方法体
|
||
}
|
||
|
||
// 通过 @JvmBlockingBridge 生成的方法
|
||
public String getValue() {
|
||
// 由 @JvmBlockingBridge 的编译器生成方法体,调用 getValue(Continuation)
|
||
}
|
||
}
|
||
```
|
||
|
||
Java 使用者可以认为 `@JvmBlockingBridge suspend fun getValue(): String` 相当于 `fun getValue(): String`。
|
||
|
||
|
||
----
|
||
|
||
> [回到 Mirai 文档索引](README.md#jvm-平台-mirai-开发) |