当前位置: 首页 > backend >正文

谈谈 Kotlin 中的构造方法,有哪些注意事项?

在 Kotlin 中,构造方法分为主构造方法(Primary Constructor)和次构造方法(Secondary Constructor)。

1 主构造方法

主构造方法是类的核心构造方法,直接在类头声明,位于类名之后。

1.1 基本语法
class Person constructor(val name: String, val age: Int) {// 类体
}

如果主构造方法没有注解或可见性修饰符,constructor 关键字可以省略:

class Person(val name: String, val age: Int) {// 类体
}
1.2 特点
1.2.1 属性声明

参数直接作为属性:可通过 valvar 将主构造函数的参数声明为类的属性。

class Person(val name: String, age: Int) {// name 是属性,age 是构造方法参数
}
1.2.2 构造方法参数的作用域

主构造方法的参数可以在类体中直接使用,但未声明 val/ var 的参数仅在 init 代码块和属性初始化器中可见。

class User(private val id: String, name: String, age: Int) {// id 是类属性,在任何地方都可以用// name、age 仅在 init 代码块和属性初始化器中可见val displayName = "[$id] $name"init {println("age = $age")}
}

属性初始化器是用于在声明属性时直接赋值的语法,它允许在类体或主构造函数中直接为属性设置初始值。

属性初始化器可以在以下两种场景中使用:

  • 主构造函数中声明属性并初始化;
  • 类体中可以直接初始化属性;
// 通过主构造函数参数声明属性并初始化
class User(val name: String = "Unknown", var age: Int = 0)class User {val name: String = "Unknown" // 属性初始化器val age: Int = 0 // 属性初始化器
}
1.2.3 初始化代码

使用 init 代码块执行额外的初始化逻辑。

class User(name: String, age: Int) {val formattedName: Stringinit {formattedName = "Mr./Ms. $name"println("Person initialized: $formattedName")}
}
1.2.4 可见性修饰符

主构造函数的可见性默认是 public,可显式指定:

class User private constructor(val name: String) // 私有构造方法

2 次构造方法(Secondary Constructor)

次构造方法通过 constructor 关键字在类体内定义,必须直接或间接调用主构造函数。

2.1 基本语法
class Person(val name: String) {var age: Int = 0// 次构造方法必须直接或间接委托给主构造方法constructor(name: String, age: Int) : this(name) {this.age = age}
}
2.2 注意事项
2.2.1 必须委托

每个次构造函数必须通过 this() 调用主构造函数或其他次构造函数,确保所有初始化路径都经过主构造函数:

class User {constructor(name: String) : this(name, 0) // 错误:没有主构造方法constructor(name: String, age: Int) // 如果没有主构造方法,次构造方法无需委托
}
2.2.2 初始化顺序

主构造函数的参数初始化 —> init 代码块 —> 次构造函数体:

class User(val name: String) {init {println("主构造方法初始化")}constructor(name: String, age: Int) : this(name) {println("次构造方法执行")}
}fun main() {User("Eileen", 34)
}// 主构造方法初始化
// 次构造方法执行
2.2.3 避免与主构造方法参数冲突

次构造方法的参数名应避免与主构造方法的属性名重复:

class User(val name: String) {constructor(name: String, age: Int) : this(name) {// 此处的 name 参数会屏蔽类的 name 属性}
}

3 初始化顺序

  • 主构造函数参数初始化;
  • 类属性按声明顺序初始化;
  • init 块按出现顺序执行;
  • 次构造函数执行;
class User(val name: String = "Eileen") {val a = println("a 初始化")init {println("init 1")}val b = println("b 初始化")init {println("init 2")}
}fun main() {User()
}// a 初始化 -> init 1 -> b 初始化 -> init 2

4 默认参数替代次构造方法

Kotlin 支持构造函数参数默认值,可减少构造函数的数量:

class User(val name: String,val age: Int = 0, // 默认参数val country: String = "Unknown"
)fun main() {val user1 = User("Eileen")val user2 = User("Eileen", 34)val user3 = User("Eileen", 34, "China")
}

Java 互操作性:如果需要在 Java 中调用带默认参数的构造方法,需添加 @JvmOverloads 注解:

class Person @JvmOverloads constructor(val name: String,val age: Int = 0,val country: String = "Unknown"
)

5 继承中的构造方法

5.1 子类必须初始化父类构造方法

子类的主构造方法需初始化父类的主构造方法:

open class User(val name: String)class Student(name: String) : User(name)

如果父类只有次构造方法(无主类构造方法),子类需通过 super 调用父类的某个次构造方法:

open class User {constructor(name: String) {}
}class Student : User {constructor(name: String) : super(name)
}
5.2 抽象类的构造方法

抽象类的构造函数可由子类实现:

abstract class User(val name: String)class Student(name: String, val age: Int) : User(name)

6 数据类中的构造方法

数据类的主构造方法必须至少有一个参数,且所有的参数必须标记为 valvar

数据类的主构造方法

7 总结

场景注意事项
主构造方法参数可声明为属性,初始化顺序严格,支持默认参数替代次构造方法
次构造方法必须委托主构造方法,代码体在类初始化后执行
继承子类必须初始化父类构造方法,优先使用主构造方法
默认参数替代次构造方法,需要 @JvmOverloads 支持 Java 调用
初始化顺序init 块和属性初始化按照代码顺序执行,次构造方法体最后执行
http://www.xdnf.cn/news/7795.html

相关文章:

  • 【Django系统】Python+Django携程酒店评论情感分析系统
  • 【Java微服务组件】异步通信P2—Kafka与消息
  • [杂学笔记]浏览器多进程与多线程架构、wstring类型、哈希表、红黑树与哈希表的对比、C++标准库Random类
  • 影响镍钯金PCB表面处理价格的因素有哪些?
  • Spring事务简单操作
  • 【低代码】如何使用明道云调用 Flask 视图函数并传参(POST 方法实践)
  • vue-cli 构建打包优化(JeecgBoot-Vue2 配置优化篇)
  • Hadoop-HA高可用集群启动nameNode莫名挂掉,排错解决
  • digitalworld.local: FALL靶场
  • Mysql-数据闪回工具MyFlash
  • SQL查询, 响应体临时字段报: Unknown column ‘data_json_map‘ in ‘field list‘
  • leetcode 92. Reverse Linked List II
  • 张 Prompt Tuning--中文数据准确率提升:理性与冲动识别新突破
  • 分类算法 Kmeans、KNN、Meanshift 实战
  • maven之pom.xml
  • 【25软考网工】第七章(3) UOS Linux防火墙配置和Web应用服务配置
  • OpenHarmony外设驱动使用 (九),Pin_auth
  • 国产化Excel处理组件Spire.XLS for .NET系列教程:通过 C# 将 TXT 文本转换为 Excel 表格
  • 物业后勤小程序源码介绍
  • 【项目记录】准备工作及查询部门
  • python-leetcode 71.每日温度
  • Vue 3.0中核心的Composition API
  • 打造一个支持MySQL查询的MCP同步插件:Java实现
  • PCB智能报价系统——————仙盟创梦IDE
  • Python实例题:PyOt实现简易浏览器
  • leetcode字符串篇【公共前缀】:14-最长公共前缀
  • C语言-9.指针
  • “交互式“ PDF 与“静态“ PDF 表单的区别
  • liinux系统安装Helm
  • 系统数据对接-从获取到处理的全流程