Kotlin:2.0.20 的新特性

一、概述

Kotlin 2.0.20英文版官方文档

Kotlin 2.0.20发布了!这个版本包括对Kotlin 2.0.0的性能改进和bug修复,我们在其中宣布Kotlin K2编译器为Stable。以下是本次发布的一些亮点:

二、数据类复制函数与构造函数具有相同的可见性

Kotlin 2.0.20开始引入更改,以提高数据类的一致性,并替换实验性上下文接收器特性。
目前,如果使用私有构造函数创建数据类,则自动生成的copy()函数不具有相同的可见性。这可能会在稍后的代码中导致问题。在未来的Kotlin版本中,我们将引入copy()函数的默认可见性与构造函数相同的行为。此更改将逐步引入,以帮助您尽可能顺利地迁移代码。

我们的迁移计划从Kotlin 2.0.20开始,它会在代码中发出警告,将来可见性会发生变化。例如:

复制代码
// Triggers a warning in 2.0.20
data class PositiveInteger private constructor(val number: Int) {
    companion object {
        fun create(number: Int): PositiveInteger? = if (number > 0) PositiveInteger(number) else null
    }
}

/**
 *  数据类复制函数与构造函数具有相同的可见性
 *  
 *  @ConsistentCopyVisibility 用于标记一个类或接口,以表明其副本在可见性方面应该是一致的。换句话说,当你创建一个类的副本时,副本的可见性应该与原始类实例的可见性相匹配。
 */
fun testConsistentCopyVisibility(){
    val positiveNumber = PositiveInteger.create(42) ?: return
    // Triggers a warning in 2.0.20
    val negativeNumber = positiveNumber.copy(number = -1)
    // Warning: Non-public primary constructor is exposed via the generated 'copy()' method of the 'data' class.
    // The generated 'copy()' will change its visibility in future releases.
}


fun main() {
  testConsistentCopyVisibility()
}

运行结果

报错信息

复制代码
Non-public primary constructor is exposed via the generated 'copy()' method of the 'data' class.

The generated 'copy()' will change its visibility in future releases.

To suppress the warning do one of the following:
- Annotate the data class with the '@ConsistentCopyVisibility' annotation.
- Use the '-Xconsistent-data-class-copy-visibility' compiler flag.
- Annotate the data class with the '@ExposedCopyVisibility' annotation 
  (Discouraged, but can be used to keep binary compatibility).

To learn more, see the documentation of the '@ConsistentCopyVisibility' and '@ExposedCopyVisibility' annotations.



为了对这种行为有更多的控制,在Kotlin 2.0.20中,我们引入了两个注释:

  • @ConsistentCopyVisibility现在就可以选择加入该行为,否则我们会在以后的版本中将其设为默认值。
  • @ExposedCopyVisibility选择退出该行为,并在声明站点上抑制警告。注意,即使有了这个注释,当调用copy()函数时,编译器仍然会报告警告。

如果您想为整个模块而不是单个类选择2.0.20中已经有的新行为,您可以使用-Xconsistent-data-class-copy-visibility编译器选项。该选项与在模块中的所有数据类中添加@ConsistentCopyVisibility注释具有相同的效果。

三、标准库

标准库现在支持通用唯一标识符作为实验特性,并包括对Base64解码的一些更改。

3.1 在通用Kotlin标准库中支持uuid

本特性是实验性的。要选择加入,请使用@ExperimentalUuidApi注释或编译器选项-opt-in=kotlin.uuid.ExperimentalUuidApi。
Kotlin 2.0.20引入了一个类来表示通用Kotlin标准库中的uuid(通用唯一标识符),以解决唯一标识项的问题。

此外,该特性还为以下uuid相关操作提供了api:

  • 生成uuid。
  • 解析uid并将其格式化为字符串表示形式。
  • 从指定的128位值创建uuid。
  • 访问UUID的128位。

下面的代码示例演示了这些操作:

复制代码
@OptIn(ExperimentalUuidApi::class)
fun testUUIDs(){
    val byteArray = byteArrayOf(
        0x55, 0x0E, 0x84.toByte(), 0x00, 0xE2.toByte(), 0x9B.toByte(), 0x41, 0xD4.toByte(),
        0xA7.toByte(), 0x16, 0x44, 0x66, 0x55, 0x44, 0x00, 0x00
    )
    val uuid1 = Uuid.fromByteArray(byteArray)
    val uuid2 = Uuid.fromULongs(0x550E8400E29B41D4uL, 0xA716446655440000uL)
    val uuid3 = Uuid.parse("550e8400-e29b-41d4-a716-446655440000")
    println(uuid1)
// 550e8400-e29b-41d4-a716-446655440000
    println(uuid1 == uuid2)
// true
    println(uuid2 == uuid3)
// true

// Accesses UUID bits
    val version = uuid1.toLongs { mostSignificantBits, _ ->
        ((mostSignificantBits shr 12) and 0xF).toInt()
    }
    println(version)
// 4

// Generates a random UUID
    val randomUuid = Uuid.random()
    println(uuid1 == randomUuid)
// false
}

fun main() {
//    testConsistentCopyVisibility()
    testUUIDs()
}

运行结果

四、kt_2020.kt文件代码

复制代码
package com.example.test.ktversion

import kotlin.uuid.ExperimentalUuidApi
import kotlin.uuid.Uuid

//https://kotlinlang.org/docs/whatsnew2020.html
// Triggers a warning in 2.0.20

// 数据类复制函数与构造函数具有相同的可见性
@ConsistentCopyVisibility
//data class PositiveInteger private constructor(val number: Int) {
//报错信息:Cannot access 'fun copy(number: Int = ...): PositiveInteger': it is private in
data class PositiveInteger constructor(val number: Int) {
    companion object {
        fun create(number: Int): PositiveInteger? = if (number > 0) PositiveInteger(number) else null
    }
}

/**
 *  数据类复制函数与构造函数具有相同的可见性
 *
 *  @ConsistentCopyVisibility 用于标记一个类或接口,以表明其副本在可见性方面应该是一致的。换句话说,当你创建一个类的副本时,副本的可见性应该与原始类实例的可见性相匹配。
 */
fun testConsistentCopyVisibility(){
    val positiveNumber = PositiveInteger.create(42) ?: return
    // Triggers a warning in 2.0.20
    val negativeNumber = positiveNumber.copy(number = -1)
    // Warning: Non-public primary constructor is exposed via the generated 'copy()' method of the 'data' class.
    // The generated 'copy()' will change its visibility in future releases.
}

// https://kotlinlang.org/docs/whatsnew2020.html#standard-library
// 通用的标准库支持 UUIDs
// Constructs a byte array for UUID creation
@OptIn(ExperimentalUuidApi::class)
fun testUUIDs(){
    val byteArray = byteArrayOf(
        0x55, 0x0E, 0x84.toByte(), 0x00, 0xE2.toByte(), 0x9B.toByte(), 0x41, 0xD4.toByte(),
        0xA7.toByte(), 0x16, 0x44, 0x66, 0x55, 0x44, 0x00, 0x00
    )
    val uuid1 = Uuid.fromByteArray(byteArray)
    val uuid2 = Uuid.fromULongs(0x550E8400E29B41D4uL, 0xA716446655440000uL)
    val uuid3 = Uuid.parse("550e8400-e29b-41d4-a716-446655440000")
    println(uuid1)
// 550e8400-e29b-41d4-a716-446655440000
    println(uuid1 == uuid2)
// true
    println(uuid2 == uuid3)
// true

// Accesses UUID bits
    val version = uuid1.toLongs { mostSignificantBits, _ ->
        ((mostSignificantBits shr 12) and 0xF).toInt()
    }
    println(version)
// 4

// Generates a random UUID
    val randomUuid = Uuid.random()
    println(uuid1 == randomUuid)
// false
}
fun main() {
    testConsistentCopyVisibility()
    testUUIDs()
}
相关推荐
inferno3 分钟前
Maven基础(一)
java·开发语言·maven
csbysj20209 分钟前
SQLite Truncate Table: 完全删除表中的数据
开发语言
tung tung tung sahur34 分钟前
领略 Rust 抽象之美:自定义迭代器实现全解析
开发语言·后端·rust
ftpeak40 分钟前
《Rust MP4视频技术开发》第八章:生成MP4
开发语言·rust·音视频·mp4
好学且牛逼的马1 小时前
【SSM框架 | day25 spring IOC 与 DI 注解开发】
java·开发语言
_OP_CHEN2 小时前
C++进阶:(四)set系列容器的全面指南
开发语言·c++·stl·set·multiset·关联式容器·setoj题
不惑_2 小时前
Java 使用 FileOutputStream 写 Excel 文件不落盘?
开发语言·python
十五年专注C++开发2 小时前
Qt-VLC: 一个集成VLC的开源跨平台媒体播放库
开发语言·qt·媒体·libvlc·vlc-qt
陈大头铃儿响叮当2 小时前
Android Studio升级后,Flutter运行android设备报错
android·flutter·android studio
勤劳打代码2 小时前
isar_flutter_libs 引发 Namespace not specified
android·flutter·groovy