Kotlin JVM 注解详解

前言

Kotlin 作为一门现代 JVM 语言,提供了出色的 Java 互操作性。为了更好地支持与 Java 代码的交互,Kotlin 提供了一系列 JVM 相关注解。这些注解不仅能帮助我们控制 Kotlin 代码编译成 Java 字节码的行为,还能让我们的 Kotlin 代码更好地被 Java 代码调用。虽然在日常开发中我们最常用的是 @JvmOverloads、@JvmStatic、@JvmName 和 @JvmField 这几个注解,但 Kotlin 其实还提供了更多强大的 JVM 注解。本文系统地整理下这些注解的作用、使用场景和具体示例,便于开发。
ps:以下整理基于kotlin-stdlib-1.7.10.jar!\kotlin\jvm\JvmInline.class

目录

  • @JvmOverloads
  • @JvmStatic
  • @JvmName
  • @JvmMultifileClass
  • @JvmPackageName
  • @JvmSynthetic
  • @Throws
  • @JvmField
  • @JvmSuppressWildcards
  • @JvmWildcard
  • @JvmInline
  • @JvmRecord

@JvmOverloads

作用

为带有默认参数值的函数生成重载方法。

使用场景

当 Kotlin 函数需要被 Java 代码调用时,特别是函数包含默认参数值的情况。

示例

@JvmOverloads
fun greet(name: String, greeting: String = "Hello") {println("$greeting, $name!")
}

编译后的 Java 代码

void greet(String name) {greet(name, "Hello");
}void greet(String name, String greeting) {System.out.println(greeting + ", " + name + "!");
}

@JvmStatic

作用

生成静态方法或静态属性访问器。

使用场景

在 companion object 中定义需要作为静态成员的方法或属性。

示例

class MyClass {companion object {@JvmStaticfun staticMethod() { }@JvmStaticvar staticProperty: String = ""}
}

@JvmName

作用

指定生成的 Java 类或方法的名称。

使用场景

  • 解决签名冲突
  • 自定义生成的 Java 代码名称
  • 改善 Java 代码的可读性

示例

@JvmName("filterString")
fun filter(list: List<String>) { }@JvmName("filterInt")
fun filter(list: List<Int>) { }

@JvmMultifileClass

作用

指示编译器生成多文件类,将多个文件中的顶级函数和属性合并到一个类中。

使用场景

需要将分散在多个文件中的相关功能组织在一起时。

示例

// File1.kt
@JvmName("Utils")
@JvmMultifileClass
fun function1() { }// File2.kt
@JvmName("Utils")
@JvmMultifileClass
fun function2() { }

@JvmPackageName

作用

更改生成的 .class 文件的 JVM 包名。

使用场景

需要自定义生成的 Java 代码的包名时。

注意

  • 内部注解,不推荐直接使用
  • 自 Kotlin 1.2 版本引入

@JvmSynthetic

作用

在 Java 字节码中设置 ACC_SYNTHETIC 标志,使目标对 Java 代码不可见。

使用场景

需要隐藏 Kotlin 特定的目标,使其对 Java 代码不可见,但保持对 Kotlin 代码可见。

示例

@JvmSynthetic
fun internalFunction() { }

@Throws

作用

指定函数编译为 JVM 方法时应声明的异常。

使用场景

需要从 Kotlin 代码中抛出 Java 检查异常时。

示例

@Throws(IOException::class)
fun readFile() { }

编译后的 Java 代码

void readFile() throws IOException { }

@JvmField

作用

指示编译器不要为属性生成 getter/setter,而是将其作为字段暴露。

使用场景

需要将 Kotlin 属性作为 Java 字段使用时。

示例

class MyClass {@JvmFieldvar field: String = ""
}

@JvmSuppressWildcards

作用

控制是否生成通配符。

使用场景

需要控制泛型类型参数的 Java 表示时。

示例

@JvmSuppressWildcards
fun process(list: List<String>) { }

@JvmWildcard

作用

为带声明点变异的类型参数生成通配符。

使用场景

需要控制泛型类型参数的 Java 表示时。

示例

fun process(@JvmWildcard list: List<String>) { }

@JvmInline

作用

指定值类为内联类。

使用场景

创建零开销的类型安全包装器。

示例

@JvmInline
value class Password(val value: String)

特点

  • 只能有一个主构造函数参数
  • 参数必须是不可变的(val)
  • 不能有 backing field
  • 不能有 init 块
  • 不能有 lateinit 属性

@JvmRecord

作用

指示编译器将类标记为记录类。

使用场景

创建不可变的数据类。

示例

@JvmRecord
data class Person(val name: String, val age: Int)

特点

  • 自 Kotlin 1.5 版本引入
  • 生成 toString、equals、hashCode 方法
  • 适用于不可变数据模型

最佳实践

  1. 选择合适的注解

    • 根据具体需求选择合适的注解
    • 考虑 Java 互操作性的需求
    • 注意注解的版本兼容性
  2. 性能考虑

    • 使用 @JvmInline 减少运行时开销
    • 合理使用 @JvmField 避免不必要的 getter/setter
    • 注意 @JvmStatic 的使用场景
  3. 代码可维护性

    • 使用 @JvmName 提高代码可读性
    • 使用 @Throws 明确异常处理
    • 使用 @JvmSynthetic 控制 API 可见性
  4. 版本兼容性

    • 注意注解的引入版本
    • 考虑向后兼容性
    • 关注 Kotlin 版本更新

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.pswp.cn/pingmian/82934.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Starrocks 物化视图的实现以及在刷新期间能否读数据

背景 本司在用Starrocks做一些业务上的分析的时候&#xff0c;用到了物化视图&#xff0c;并且在高QPS的情况下&#xff0c;RT也没有很大的波动&#xff0c;所以在此研究一下Starrock的实现&#xff0c;以及在刷新的时候是不是原子性的 本文基于Starrocks 3.3.5 结论 Starro…

[网页五子棋][对战模块]前后端交互接口(建立连接、连接响应、落子请求/响应),客户端开发(实现棋盘/棋子绘制)

文章目录 约定前后端交互接口建立连接建立连接响应针对"落子"的请求和响应 客户端开发实现棋盘/棋子绘制部分逻辑解释 约定前后端交互接口 对战模块和匹配模块使用的是两套逻辑&#xff0c;使用不同的 websocket 的路径进行处理&#xff0c;做到更好的耦合 建立连接 …

电工基础【2】自锁、互锁、正反转电路

04 自锁、正反转电路 我们讲一下这个自锁和正反转。 自锁电路图示例图 加了一个这个 KM1 自锁。加了 KM1 的辅助触头&#xff0c;它怎么实现呢&#xff1f;它怎么就自锁了呢&#xff1f;没加它的时候为什么是点动&#xff1f;加它为什么自锁&#xff1f; 讲解一下。首先我们…

TypeScript 中感叹号(!)两种位置用法

这是一个非常好的问题&#xff01; 在 TypeScript 中&#xff0c;感叹号&#xff08;!&#xff09;有两种位置用法&#xff0c;它们含义完全不同&#xff1a; ✅ 一、后置感叹号 !&#xff08;非空断言&#xff09; process.env.ADMIN_PRIVATE_KEY! ✅ 作用&#xff1a; 告…

t014-项目申报管理系统 【springBoot 含源码】

项目演示视频 摘 要 传统信息的管理大部分依赖于管理人员的手工登记与管理&#xff0c;然而&#xff0c;随着近些年信息技术的迅猛发展&#xff0c;让许多比较老套的信息管理模式进行了更新迭代&#xff0c;项目信息因为其管理内容繁杂&#xff0c;管理数量繁多导致手工进行…

【C++】STL详解(四)---Stack和Queue

文章目录 Stack定义方式使用方式 Queue定义方式使用方式 Stack Stack是一种容器&#xff0c;是基本的数据结构之一&#xff0c;特点是先进后出。 定义方式 方式一&#xff1a;普通定义方式 stack<int> st1;方式二&#xff1a; stack<int,vector<int>> …

解决 xmlsec.InternalError: (-1, ‘lxml xmlsec libxml2 library version mismatch‘)

解决 xmlsec.InternalError: (-1, ‘lxml & xmlsec libxml2 library version mismatch’) 错误信息如下&#xff1a; Traceback (most recent call last):File "/home/mobsf/Mobile-Security-Framework-MobSF/manage.py", line 18, in <module>execute_f…

SpringBoot自定义实体类字段的校验注解

在Spring Boot项目中&#xff0c;我们经常需要对请求参数进行格式或业务规则的校验。虽然Spring Boot提供了如NotNull、Size等基础校验注解&#xff0c;但在实际开发中往往无法满足复杂的业务需求。但是在Controller层写大量的 if 语句的判断逻辑又实在不优雅&#xff0c;好在 …

实现单例模式的6种方法(Python)

目录 一. 基于模块的实现(简单&#xff0c;易用) 二. 重新创建时报错(不好用) 三. 只靠方法获取实例(不好用) 四. 类装饰器 五. 重写__new__方法 六. 元类 七. 总结 单例模式&#xff08;Singleton Pattern&#xff09;是一种设计模式&#xff0c;其核心目标是确保一个类…

循环神经网络(RNN)全面教程:从原理到实践

循环神经网络(RNN)全面教程&#xff1a;从原理到实践 引言 循环神经网络(Recurrent Neural Network, RNN)是处理序列数据的经典神经网络架构&#xff0c;在自然语言处理、语音识别、时间序列预测等领域有着广泛应用。本文将系统介绍RNN的核心概念、常见变体、实现方法以及实际…

使用Vditor将Markdown文档渲染成网页(Vite+JS+Vditor)

1. 引言 编写Markdown文档现在可以说是程序员的必备技能了&#xff0c;因为Markdown很好地实现了内容与排版分离&#xff0c;可以让程序员更专注于内容的创作。现在很多技术文档&#xff0c;博客发布甚至AI文字输出的内容都是以Markdown格式的形式输出的。那么&#xff0c;Mar…

Day 40

单通道图片的规范写法 import torch import torch.nn as nn import torch.optim as optim from torch.utils.data import DataLoader , Dataset from torchvision import datasets, transforms import matplotlib.pyplot as plt import warnings warnings.filterwarnings(&q…

SPSS跨域分类:自监督知识+软模板优化

1. 图1:SPSS方法流程图 作用:展示了SPSS方法的整体流程,从数据预处理到模型预测的关键步骤。核心内容: 领域知识提取:使用三种词性标注工具(NLTK、spaCy、TextBlob)从源域和目标域提取名词或形容词(如例句中提取“excellent”“good”等形容词)。词汇交集与聚类:对提…

2025年通用 Linux 服务器操作系统该如何选择?

2025年通用 Linux 服务器操作系统该如何选择&#xff1f; 服务器操作系统的选择对一个企业IT和云服务影响很大&#xff0c;主推的操作系统在后期更换的成本很高&#xff0c;而且也有很大的迁移风险&#xff0c;所以企业在选择服务器操作系统时要尤为重视。 之前最流行的服务器…

如何在 Django 中集成 MCP Server

目录 背景说明第一步&#xff1a;使用 ASGI第二步&#xff1a;修改 asgi.py 中的应用第三步&#xff1a;Django 数据的异步查询 背景说明 有几个原因导致 Django 集成 MCP Server 比较麻烦 目前支持的 MCP 服务是 SSE 协议的&#xff0c;需要长连接&#xff0c;但一般来讲 Dj…

天拓四方工业互联网平台赋能:地铁电力配电室综合监控与无人巡检,实现效益与影响的双重显著提升

随着城市化进程的不断加快&#xff0c;城市轨道交通作为缓解交通压力、提升出行效率的重要方式&#xff0c;在全国各大城市中得到了迅猛发展。地铁电力配电室作为核心供电设施&#xff0c;其基础设施的安全性、稳定性和智能化水平也面临更高要求。 本文将围绕“工业物联网平台…

算法打卡第11天

36.有效的括号 &#xff08;力扣20题&#xff09; 示例 1&#xff1a; **输入&#xff1a;**s “()” **输出&#xff1a;**true 示例 2&#xff1a; **输入&#xff1a;**s “()[]{}” **输出&#xff1a;**true 示例 3&#xff1a; **输入&#xff1a;**s “(]”…

python 包管理工具uv

uv --version uv python find uv python list export UV_DEFAULT_INDEX"https://mirrors.tuna.tsinghua.edu.cn/pypi/web/simple" # 换成私有的repo export UV_HTTP_TIMEOUT120 uv python install 3.12 uv venv myenv --python 3.12 --seed uvhttps://docs.ast…

spring的多语言怎么实现?

1.创建springboot项目&#xff0c;并配置application.properties文件 spring.messages.basenamemessages spring.messages.encodingUTF-8 spring.messages.fallback-to-system-localefalsespring.thymeleaf.cachefalse spring.thymeleaf.prefixclasspath:/templates/ spring.t…

JAVA:Kafka 消息可靠性详解与实践样例

🧱 1、简述 Apache Kafka 是高吞吐、可扩展的流处理平台,在分布式架构中广泛应用于日志采集、事件驱动和微服务解耦场景。但在使用过程中,消息是否会丢?何时丢?如何防止丢? 是很多开发者关心的问题。 Kafka 提供了一套完整的机制来保障消息从生产者 ➜ Broker ➜ 消费…