在 Kotlin 中,类描述的是一种通用结构,可以多次实例化,也可以用多种方式实例化。但有时我们只需要单个实例,不多不少。单例模式能帮你更好地组织代码,把相关的方法聚合在一起。

单例模式是什么?

单例模式是一种设计模式,保证一个类只有一个实例,并提供全局访问点。这意味着你可以在代码的任何地方获取这个单例类的实例。打个比方,就像大家一起玩棋盘游戏,所有玩家都在同一个“棋盘”上进行操作,这个棋盘就相当于游戏的全局状态。

单例的主要特征:

  • 单例类只有一个实例。

  • 单例类提供一个全局访问点。

Kotlin 的对象声明(object declaration)

单例模式非常有用,而 Kotlin 为单例提供了专门的语法结构:object 声明。这是一种特殊的类声明,使用关键字 object 创建单例。Kotlin 自动处理所有复杂步骤,你只需要用 object 声明即可,无需手动实现单例模式。

object PlayingField {fun getAllPlayers(): Array<Player> {/* ... */}fun isPlayerInGame(player: Player): Boolean {/* ... */}}

解释:

使用 object 声明时,构造函数不可用,因为 Kotlin 自动完成。你可以通过 PlayingField 直接访问这个单例实例,它在代码任何地方调用都指向同一个对象。

示例:

fun startNewGameTurn() {val players = PlayingField.getAllPlayers()if (players.size < 2) {return println("The game cannot be continued without players")}for (player in players) {nextPlayerTurn(player)}
}fun nextPlayerTurn(player: Player) {if (!PlayingField.isPlayerInGame(player)) {return println("Current player lost. Next...")}/* 玩家行动 */
}

嵌套对象(Nested object)

有时候你想创建一个和另一个类相关联的单例。例如,游戏中有 Player 类,代表不同的角色,这些角色有共享的属性,比如默认速度。你如何保存这些共享信息呢?

你可以简单地创建一个单例对象:

object PlayerProperties {/* 默认速度,每回合移动7格 */val defaultSpeed = 7fun calcMovePenalty(cell: Int): Int {/* 计算移动速度惩罚 */}
}

但如果项目里有许多类似的单例,代码会变得难读。更好的做法是将单例嵌套到相关类中。

class Player(val id: Int) {object Properties {val defaultSpeed = 7fun calcMovePenalty(cell: Int): Int {/* 计算移动惩罚 */}}
}/* 输出 7 */
println(Player.Properties.defaultSpeed)

Properties 对象作用域是 Player,只能通过 Player.Properties 访问。这种方式让单例和类有明确的关联。

你还可以在外部类中使用嵌套对象的属性:

class Player(val id: Int) {object Properties {val defaultSpeed = 7}val superSpeed = Properties.defaultSpeed * 2 // 14
}

但反过来是不行的——嵌套对象中不能访问外部类的实例成员:

class Player(val id: Int) {    val speed = 7object Properties {val defaultSpeed = speed // 错误,不能访问外部类实例属性}
}

这和其他语言的 static 类似,Kotlin 没有默认的静态成员,但可以用嵌套对象来达到类似效果。


编译时常量(Compile-time constants)

如果某个只读属性永远不会改变,我们称它为常量。可以使用 const 关键字声明编译时常量:

object Languages {const val FAVORITE_LANGUAGE = "Kotlin"
}

要求:

  • 必须是基本类型或 String。

  • 不能有自定义 getter。

  • 命名用全大写加下划线(SCREAMING_SNAKE_CASE)。

比如游戏里的默认速度可以写成:

object Properties {const val DEFAULT_SPEED = 7
}

访问:

println(Properties.DEFAULT_SPEED) // 输出 7

为什么不都用顶层常量?因为大量无关联的顶层常量会让代码混乱,影响阅读。最好把和某个对象相关的常量放到对应的对象里。


对象与嵌套对象的扩展

你可以在一个类里声明多个对象,比如:

class Player(val id: Int) {object Properties {val defaultSpeed = 7fun calcMovePenalty(cell: Int): Int { /* ... */ }}object Factory {fun create(playerId: Int): Player {return Player(playerId)}}
}println(Player.Properties.defaultSpeed)    // 7
println(Player.Factory.create(13).id)      // 13

这里的 Factory 是工厂模式,用来创建 Player 实例。你也可以在一个对象内部声明多个对象,用来组织单例数据:

object Game {object Properties {val maxPlayersCount = 13val maxGameDurationInSec = 2400}object Info {val name = "My super game"}
}

数据对象(Data object)

普通的对象声明打印会显示类名和哈希码:

object MyObjectfun main() {println(MyObject) // MyObject@1f32e575
}

如果用 data 修饰单例对象,会生成更友好的方法:

data object MySingletonfun main() {println(MySingleton) // MySingleton
}

注意,data object 不是数据类,不能复制(没有 copy() 方法),也没有组件函数,因为单例不允许多实例。


总结

Kotlin 中,object 声明是创建单例的标准方式,也可以用嵌套对象关联类本身,而非类的实例。合理使用它们可以让代码结构更清晰,提升可读性和可维护性。

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

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

相关文章

Shell 编程基础入门从认识到实战

对于刚接触 Linux 或 Unix 系统的开发者来说&#xff0c;Shell 脚本往往是自动化操作的第一道门槛。它不像 Python 那样语法简洁&#xff0c;也不像 Java 那样有完善的面向对象体系&#xff0c;但却能以极少的代码实现强大的系统管理功能。本文将从 Shell 的基本概念讲起&#…

混合遗传粒子群算法在光伏系统MPPT中的应用研究

混合遗传粒子群算法在光伏系统MPPT中的应用研究 前些天发现了一个巨牛的人工智能学习网站&#xff0c;通俗易懂&#xff0c;风趣幽默&#xff0c;忍不住分享一下给大家&#xff0c;觉得好请收藏。点击跳转到网站。 摘要 本文针对光伏系统最大功率点跟踪(MPPT)问题&#xff0…

机器视觉的布料丝印应用

在纺织印染行业&#xff0c;布料丝印工艺的精度直接决定产品外观质量与市场竞争力。传统丝印设备依赖机械定位与人工校准&#xff0c;面对高密度图案、柔性面料或复杂纹理时&#xff0c;易出现套色偏移、油墨渗透不均等问题&#xff0c;导致良品率波动与生产成本攀升。 随着机…

前端常用类库

常用类库 类库作用 类库可以帮助我们快速实现项目业务的开发与功能的实现, 帮助我们解放劳动力提高生产效率, 前端中的类库与框架都是由原生javascript编写, 提供给其他开发者应用于某一业务环境或者需求。一般有开发者/团队开源维护. 优秀的类库需要具备高度封装可用, 稳定, …

通俗易懂循环神经网络(RNN)指南

本文用直观类比、图表和代码&#xff0c;带你轻松理解RNN及其变体&#xff08;LSTM、GRU、双向RNN&#xff09;的原理和应用。什么是循环神经网络 循环神经网络&#xff08;Recurrent Neural Network, RNN&#xff09;是一类专门用于处理序列数据的神经网络。与前馈神经网络不同…

【SVM】支持向量机实例合集

基于Java的SVM(支持向量机)实例合集 以下是一个基于Java的SVM(支持向量机)实例合集,包含核心代码示例和应用场景说明。这些例子基于流行的机器学习库(如LIBSVM、Weka、JSAT)实现。 数据准备与加载 使用LIBSVM格式加载数据集: // 加载LIBSVM格式数据 svm_problem pr…

Python100个库分享第38个—lxml(爬虫篇)

目录专栏导读&#x1f4da; 库简介&#x1f3af; 主要特点&#x1f6e0;️ 安装方法Windows安装Linux/macOS安装验证安装&#x1f680; 快速入门基本使用流程HTML vs XML解析&#x1f50d; 核心功能详解1. XPath选择器2. CSS选择器支持3. 元素操作&#x1f577;️ 实战爬虫案例…

imx6ull-系统移植篇17——linux顶层 Makefile(上)

目录 前言 顶层 Makefile 源码简析 版本号 MAKEFLAGS 变量 命令输出 静默输出 设置编译结果输出目录 代码检查 模块编译 设置目标架构和交叉编译器 调用 scripts/Kbuild.include 文件 交叉编译工具变量设置 头文件路径变量 导出变量 make xxx_defconfig 过程 …

OpenCV 官翻6 - Computational Photography

文章目录图像去噪目标理论OpenCV中的图像去噪1、cv.fastNlMeansDenoisingColored()2、cv.fastNlMeansDenoisingMulti()附加资源图像修复目标基础概念代码补充资源练习高动态范围成像&#xff08;HDR&#xff09;目标理论基础曝光序列HDR1、将曝光图像加载到列表中2、将曝光序列…

APT32F1732RBT8爱普特微电子 32位MCU国产芯片 智能家居/工业控制 首选

APT32F1732RBT8 爱普特微电子&#xff0c;32位MCU国产芯片一、产品简介APT32F1732RBT8 是爱普特微电子&#xff08;APT&#xff09;推出的高性能32位ARM Cortex-M0内核MCU&#xff0c;主频高达48MHz&#xff0c;内置64KB Flash8KB RAM&#xff0c;专为智能家居、工业控制、消费…

Smart Tomcat

本篇博客的内容是教你借助idea中的插件,把tomcat集成到idea中安装 Smart Tomcat 插件搜索下载 ,如果一直处于加载界面,就尝试一下科学上网配置 Smart Tomcat 插件 点击右上角的 "Add Configuration"选择左侧的 "Smart Tomcat" 在 Name 这一栏填写一个名字(…

Linux_shell编写

title: Linux_4 shell编写 shell pwd (/root/A/2025_7/19/myshell) 首先需要设计命令行提示 &#xff08;MakeCommandLine()&#xff09; 首先获取相关信息 getenv(“name”) // 获取用户名 const char* GetUserName() {const char* name getenv("USER");if (name …

【数据结构】栈的深入解析--用C语言实现

文章目录1.栈的概念2.栈的底层结构3.栈的功能4.栈的实现4.1.栈结构的定义4.2.栈的初始化4.3.栈的销毁4.4.入栈4.5.出栈4.6.取栈顶元素4.7.获取栈中有效元素个数5.完整代码Stack.hStack.cmain.c运行结果1.栈的概念 是一种特殊的线性表&#xff0c;只允许数据在固定的一段进行插…

Git仓库核心概念与工作流程详解:从入门到精通

Git仓库的基本概念版本库&#xff08;Repository&#xff09;是Git的核心概念&#xff0c;你可以简单理解为一个被Git管理的目录。这个目录里的所有文件都能被Git跟踪&#xff0c;记录每次修改和删除&#xff0c;让你可以随时追溯历史或在未来某个时刻"还原"文件。Gi…

Web开发 05

1 React库&#xff08;人话详解版&#xff09;别慌&#xff0c;React 刚接触时是会有点懵&#xff0c;咱们用 “人话 类比” 一步步拆&#xff1a;核心概念先抓牢组件&#xff08;Component&#xff09;把它想成 “乐高积木”&#xff0c;比如做个社交 App&#xff0c;顶部导航…

RustDesk 自建中继服务器教程(Mac mini)

&#x1f4d6; 教程目标 在家里的 Mac mini 上部署 RustDesk 中继服务器 (hbbs hbbr)&#xff0c;让你从办公室、笔电或手机 低延迟、安全 地远程控制家里的 Windows 和 Mac mini。 ✅ 不依赖第三方服务器 ✅ 支持 P2P 和中继双模式 ✅ 全流量可控、跨平台 &#x1f3d7;️ 架…

数据库—修改某字段默认值

前言有时候&#xff0c;数据库的字段默认值没有正确设置&#xff0c;这时候需要改默认值。以下是我做的改默认值的记录&#xff0c;希望对网友有所帮助。1.SQL SERVER下面的示例假设你要修改名为 YourColumnName 的字段&#xff0c;并为其设置一个新的默认值 NewDefaultValue。…

Spring快速整合Mybatis

MyBatis是一个优秀的持久层框架&#xff0c;Spring则是广泛使用的Java应用框架。可以将两者整合可以充分发挥各自的优势。 1、Spring整合MyBatis的基本配置 添加依赖&#xff1a; <dependency><groupId>org.springframework</groupId><artifactId>spri…

基于深度学习的语音识别:从音频信号到文本转录

前言 语音识别&#xff08;Automatic Speech Recognition, ASR&#xff09;是人工智能领域中一个极具挑战性和应用前景的研究方向。它通过将语音信号转换为文本&#xff0c;为人们提供了更加自然和便捷的人机交互方式。近年来&#xff0c;深度学习技术在语音识别领域取得了显著…

本地部署Nacos开源服务平台,并简单操作实现外部访问,Windows 版本

Nacos 是一款阿里开源的动态服务发现、配置、管理平台&#xff0c;拥有易于集成、高可用与可扩展等特点。它提供了动态服务注册和发现能力&#xff0c;使得服务自动注册到服务器并且消费真能够发现提供者。本文将详细介绍如何在本地安装 Nacos &#xff0c;以及结合nat123端口映…