Kotlin 中 companion object 扩展函数和普通函数区别

news2025/6/7 3:45:39

在 Kotlin 中,companion object 的扩展函数与普通函数(包括普通成员函数和普通扩展函数)有显著区别。以下是它们的核心差异和适用场景:

1. 定义位置与归属

特性companion object 扩展函数普通函数
定义位置在类外部为伴生对象添加在类内部(成员函数)或任意位置(扩展函数)
归属关系属于类的伴生对象,而非类实例成员函数属于实例,普通扩展函数属于接收者类型

示例对比

// companion object 扩展函数
class MyClass {
    companion object
}
fun MyClass.Companion.extFunc() = println("扩展函数")

// 普通成员函数
class MyClass {
    fun memberFunc() = println("成员函数")
}

// 普通扩展函数(非伴生对象)
fun MyClass.extFunc() = println("普通扩展函数")

2. 调用方式

特性companion object 扩展函数普通函数
调用主体通过类名直接调用成员函数需实例,普通扩展函数通过实例调用
语法ClassName.func()instance.func()

示例对比

// companion object 扩展函数
MyClass.extFunc()  // 直接通过类名调用

// 普通成员函数
val obj = MyClass()
obj.memberFunc()   // 需要实例

// 普通扩展函数
obj.extFunc()      // 需要实例

3. 访问权限

特性companion object 扩展函数普通函数
访问私有成员只能访问伴生对象的私有成员成员函数可访问类所有成员,普通扩展函数只能访问公有成员
上下文无类实例上下文(相当于静态上下文)普通成员函数有 this 指向实例

示例对比

class MyClass(private val secret: String) {
    companion object {
        private const val COMPANION_SECRET = "companion-secret"
    }
    
    fun memberFunc() {
        println(secret)          // 可访问实例私有属性
        println(COMPANION_SECRET) // 可访问伴生对象私有属性
    }
}

// companion object 扩展函数
fun MyClass.Companion.extFunc() {
    println(COMPANION_SECRET) // 只能访问伴生对象的私有成员
    // println(secret)        // 编译错误:无法访问实例成员
}

// 普通扩展函数
fun MyClass.extFunc() {
    // println(secret)        // 编译错误:无法访问私有成员
    // println(COMPANION_SECRET) // 编译错误:无法访问伴生对象私有成员
}

4. 使用场景

场景companion object 扩展函数普通函数
工具类方法✅ 适合(如 StringUtils.parse()❌ 需实例,不直观
工厂模式✅ 通过类名创建对象(MyClass.create()❌ 需先有工厂实例
实例操作❌ 无法操作实例✅ 主要用途
第三方库扩展✅ 为已有类添加静态方法✅ 为实例添加方法

典型用例

// companion object 扩展:为 Android 的 Toast 添加静态方法
fun Toast.Companion.showShort(context: Context, text: String) {
    makeText(context, text, Toast.LENGTH_SHORT).show()
}
// 调用:Toast.showShort(context, "Hello")

// 普通扩展:为 String 添加功能
fun String.addExclamation() = "$this!"
// 调用:"Hi".addExclamation()

5. 初始化时机

特性companion object 扩展函数普通函数
加载时机首次访问类时初始化伴生对象随实例创建或调用时执行
内存开销类级别共享实例级别(成员函数)或无状态(扩展函数)

总结对比表

维度companion object 扩展函数普通成员函数普通扩展函数
定义位置类外部类内部任意位置
调用方式ClassName.func()instance.func()instance.func()
访问权限仅伴生对象成员全实例成员仅公有成员
典型用途静态工具方法、工厂模式实例行为封装增强已有类功能
内存分配类级别(单次初始化)每实例占用无状态(不占内存)

选择建议

  • 需要 通过类名直接调用 且 不依赖实例状态 → 用 companion object 扩展函数
  • 需要 操作具体实例数据 → 用普通成员函数
  • 需要 为无法修改的类添加功能 → 用普通扩展函数

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

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

相关文章

《汇编语言》第13章 int指令

中断信息可以来自 CPU 的内部和外部,当 CPU 的内部有需要处理的事情发生的时候,将产生需要马上处理的中断信息,引发中断过程。在第12章中,我们讲解了中断过程和两种内中断的处理。 这一章中,我们讲解另一种重要的内中断…

Redis实战-基于redis和lua脚本实现分布式锁以及Redission源码解析【万字长文】

前言: 在上篇博客中,我们探讨了单机模式下如何通过悲观锁(synchronized)实现"一人一单"功能。然而,在分布式系统或集群环境下,单纯依赖JVM级别的锁机制会出现线程并发安全问题,因为这…

计算机网络 : 应用层自定义协议与序列化

计算机网络 : 应用层自定义协议与序列化 目录 计算机网络 : 应用层自定义协议与序列化引言1. 应用层协议1.1 再谈协议1.2 网络版计算器1.3 序列化与反序列化 2. 重新理解全双工3. socket和协议的封装4. 关于流失数据的处理5. Jsoncpp5.1 特性5.2 安装5.3…

Python Day42 学习(日志Day9复习)

补充:关于“箱线图”的阅读 以下图为例 浙大疏锦行 箱线图的基本组成 箱体(Box):中间的矩形,表示数据的中间50%(从下四分位数Q1到上四分位数Q3)。中位线(Median)&#…

CMake在VS中使用远程调试

选中CMakeLists.txt, 右键-添加调试配置-选中"C\C远程windows调试" 之后将 aunch.vs.json文件改为如下所示: CMake在VS中使用远程调试时,Launch.vs.json中远程调试设置 ,远程电脑开启VS专用的RemoteDebugger {"version": "0.2.1","defaul…

《图解技术体系》How Redis Architecture Evolves?

Redis架构的演进经历了多个关键阶段,从最初的内存数据库发展为支持分布式、多模型和持久化的高性能系统。以下为具体演进路径: 单线程模型与基础数据结构 Redis最初采用单线程架构,利用高效的I/O多路复用(如epoll)处…

一文速通Python并行计算:12 Python多进程编程-进程池Pool

一文速通 Python 并行计算:12 Python 多进程编程-进程池 Pool 摘要: 在Python多进程编程中,Pool类用于创建进程池,可并行执行多个任务。通过map、apply等方法,将函数和参数分发到子进程,提高CPU利用率&…

Web前端之原生表格动态复杂合并行、Vue

MENU 效果公共数据纯原生StyleJavaScript vue原生table 效果 原生的JavaScript原生table null 公共数据 const list [{id: "a1",title: "第一列",list: [{id: "a11",parentId: "a1",title: "第二列",list: [{ id: "…

『uniapp』把接口的内容下载为txt本地保存 / 读取本地保存的txt文件内容(详细图文注释)

目录 预览效果思路分析downloadTxt 方法readTxt 方法 完整代码总结 欢迎关注 『uniapp』 专栏,持续更新中 欢迎关注 『uniapp』 专栏,持续更新中 预览效果 思路分析 downloadTxt 方法 该方法主要完成两个任务: 下载 txt 文件:通…

【Redis技术进阶之路】「原理分析系列开篇」分析客户端和服务端网络诵信交互实现(服务端执行命令请求的过程 - 时间事件处理部分)

揭秘高效存储模型与数据结构底层实现 【专栏简介】【技术大纲】【专栏目标】【目标人群】1. Redis爱好者与社区成员2. 后端开发和系统架构师3. 计算机专业的本科生及研究生 时间事件:serverCron函数更新服务器时间缓存更新LRU时钟-lruclock更新服务器每秒执行命令次…

【DAY40】训练和测试的规范写法

内容来自浙大疏锦行python打卡训练营 浙大疏锦行 知识点: 彩色和灰度图片测试和训练的规范写法:封装在函数中展平操作:除第一个维度batchsize外全部展平dropout操作:训练阶段随机丢弃神经元,测试阶段eval模式关闭drop…

el-select 实现分页加载,切换也数滚回到顶部,自定义高度

el-select 实现分页加载&#xff0c;切换也数滚回到顶部&#xff0c;自定义高度 1.html <el-form-item label"俱乐部&#xff1a;" prop"club_id" label-width"120px"><el-select :disabled"Boolean(match_id)" style"w…

Langchaine4j 流式输出 (6)

Langchaine4j 流式输出 大模型的流式输出是指大模型在生成文本或其他类型的数据时&#xff0c;不是等到整个生成过程完成后再一次性 返回所有内容&#xff0c;而是生成一部分就立即发送一部分给用户或下游系统&#xff0c;以逐步、逐块的方式返回结果。 这样&#xff0c;用户…

学习经验分享【40】目标检测热力图制作

目标检测热力图在学术论文&#xff08;尤其是计算机视觉、深度学习领域&#xff09;中是重要的可视化分析工具和论证辅助手段&#xff0c;可以给论文加分不少。主要作用一是增强论文的可解释性与说服力&#xff1a;论文中常需解释模型 “如何” 或 “为何” 检测到目标&#xf…

C#里与嵌入式系统W5500网络通讯(3)

有与W5500通讯时,需要使用下面的寄存器: PHYCFGR (W5500 PHY Configuration Register) [R/W] [0x002E] [0b10111XXX] PHYCFGR configures PHY operation mode and resets PHY. In addition, PHYCFGR indicates the status of PHY such as duplex, Speed, Link. 这张表格详细…

用OpenNI2获取奥比中光Astra Pro输出的深度图(win,linux arm64 x64平台)

搞了一个奥比中光Astra Pro&#xff0c;想在windows平台&#xff0c;和linux rk3588 &#xff08;香橙派&#xff0c;ubuntu2404,debian)上获取深度信息&#xff0c;之前的驱动下载已经不好用了,参考如下 Astra 3D相机选型建议 - 知乎https://zhuanlan.zhihu.com/p/594485674 …

Unity VR/MR开发-VR设备与适用场景分析

视频讲解链接&#xff1a;【XR马斯维】VR/MR设备与适用场景分析&#xff1f;【UnityVR/MR开发教程--入门】_游戏热门视频

Linux: network: switch:arp cache更新规则 [chatGPT]

文章目录 介绍概念普通包带有不同的mac,是否更新arp cache?普通包带有相同的mac,是否刷新 aging timeswitch是否会主动学习介绍 关于arp cache在switch侧的行为。有很多问题需要理解。 概念 HP L3 - IP Services Configuration Guide 文档里有写:dynamic arp entry的解说…

Java网络编程API 1

Java中的网络编程API一共有两套&#xff1a;一套是UDP协议使用的API&#xff1b;另一套是TCP协议使用的API。这篇文章我们先来介绍UDP版本的API&#xff0c;并尝试来写一个回显服务器&#xff08;接收到的请求是什么&#xff0c;返回的响应就是什么&#xff09;。 UDP数据报套…

兰亭妙微 | 医疗软件的界面设计能有多专业?

从医疗影像系统到手术机器人控制界面&#xff0c;从便携式病原体检测设备到多平台协同操作系统&#xff0c;兰亭妙微为众多医疗设备研发企业&#xff0c;打造了兼具专业性与可用性的交互界面方案。 我们不仅做设计&#xff0c;更深入理解医疗场景的实际需求&#xff1a; 对精…