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

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

在 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.lryc.cn/news/2402425.html

相关文章:

  • 《汇编语言》第13章 int指令
  • Redis实战-基于redis和lua脚本实现分布式锁以及Redission源码解析【万字长文】
  • Ubuntu崩溃修复方案
  • 计算机网络 : 应用层自定义协议与序列化
  • Python Day42 学习(日志Day9复习)
  • CMake在VS中使用远程调试
  • 《图解技术体系》How Redis Architecture Evolves?
  • 从零搭建到 App Store 上架:跨平台开发者使用 Appuploader与其他工具的实战经验
  • Spring Cloud 2025 正式发布啦
  • 一文速通Python并行计算:12 Python多进程编程-进程池Pool
  • 相机Camera日志分析之二十五:高通相机Camx 基于预览1帧的process_capture_request四级日志分析详解
  • React从基础入门到高级实战:React 实战项目 - 项目一:在线待办事项应用
  • 云部署实战:基于AWS EC2/Aliyun ECS与GitHub Actions的CI/CD全流程指南
  • golang 如何定义一种能够与自身类型值进行比较的Interface
  • Web前端之原生表格动态复杂合并行、Vue
  • 『uniapp』把接口的内容下载为txt本地保存 / 读取本地保存的txt文件内容(详细图文注释)
  • C/C++ 面试复习笔记(2)
  • 宝马集团推进数字化转型:强化生产物流与财务流程,全面引入SAP现代架构
  • 【Redis技术进阶之路】「原理分析系列开篇」分析客户端和服务端网络诵信交互实现(服务端执行命令请求的过程 - 时间事件处理部分)
  • 【DAY40】训练和测试的规范写法
  • C语言 标准I/O函数全面指南
  • el-select 实现分页加载,切换也数滚回到顶部,自定义高度
  • Langchaine4j 流式输出 (6)
  • Jenkins:自动化流水线的基石,开启 DevOps 新时代
  • 学习经验分享【40】目标检测热力图制作
  • C#里与嵌入式系统W5500网络通讯(3)
  • 用OpenNI2获取奥比中光Astra Pro输出的深度图(win,linux arm64 x64平台)
  • Unity VR/MR开发-VR设备与适用场景分析
  • Linux: network: switch:arp cache更新规则 [chatGPT]
  • Java网络编程API 1