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

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.xdnf.cn/news/10506.html

相关文章:

  • Qt OpenGL 3D 编程入门
  • Grafana对接Prometheus数据源
  • JAVA学习-练习试用Java实现“使用JavaFX绘制散点图 :可视化数据集”
  • 【2025年B卷】华为OD-100分-字符串重新排列、字符串重新排序
  • 解锁 AI 大语言模型的“知识宝藏”:知识库的奥秘与优化之道
  • TDengine 的 AI 应用实战——电力需求预测
  • 秋招Day12 - 计算机网络 - UDP
  • 阿里云国际站,如何通过代理商邀请的链接注册账号
  • 多维度健康护理:为进行性核上性麻痹患者护航
  • Python基础入门:开启编程之旅
  • 数据资产评估进阶:精读资产评估专家指引第9号——数据资产评估指导【附全文阅读】
  • CppCon 2014 学习:Gamgee: A C++14 library for genomic data processing and analysis
  • 服务器间文件传输
  • Linux_T(Sticky Bit)粘滞位详解
  • Spring Boot中的WebSocket技术实现
  • Linux 权限管理入门:从基础到实践
  • 123网盘SDK-npm包已发布
  • MyBatisPlus--条件构造器及自定义SQL详解
  • 【Linux系列】Gunicorn 进程架构解析:主进程与工作进程
  • CppCon 2014 学习:Hardening Your Code
  • 3.RV1126-OPENCV 图像叠加
  • LM393红外避障电路Multisim仿真
  • (七)【Linux进程的创建、终止和等待】
  • 【AI论文】R2R:通过小型与大型模型之间的令牌路由高效导航发散推理路径
  • GpuGeek如何成为AI基础设施市场的中坚力量
  • C++11新特性包装器
  • 自然图像数据集
  • Lesson 26 The best art critics
  • 软考-数据库系统工程师-程序设计语言知识要点
  • 如何安装ojdbc6-12.1.0.1与je-5.0.58的mvn构建依赖jar包?