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

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 扩展函数
  • 需要 操作具体实例数据 → 用普通成员函数
  • 需要 为无法修改的类添加功能 → 用普通扩展函数

相关文章:

  • 【Delphi】实现在多显示器时指定程序运行在某个显示器上
  • 使用 OpenCV (C/C++) 通过二值化增强车牌识别
  • 如何选择合适的哈希算法以确保数据安全?
  • AJAX对于XML和JSON的处理
  • 腾讯云 Python3.12.8 通过yum安装 并设置为默认版本
  • dify应用探索
  • 基于ubuntu和树莓派环境对游戏进行移植
  • imx6ull(0):烧录、启动
  • JavaSE:面向对象进阶之内部类(Inner Class)
  • Uiverse.io:免费UI组件库
  • 通信革新与网络安全探索与创新:开启未来之门
  • MySQL-多表关系、多表查询
  • 绿盟 IPS 设备分析操作手册
  • 详细解析2MHz和3MHz压电陶瓷片的区别
  • TomSolver 库 | config详解及其测试
  • 嵌入式Linux 期末复习指南(下)
  • Java如何读取CSV文件并将数据放入对象中详解
  • GMDCMonitor企业版功能分享0602
  • Python列表、字典、元组、集合
  • 系统级 EOS 测试方法 - System Level EOS Testing Method
  • wordpress游戏站/网络推广赚钱项目
  • 二手房发布网站怎么做/seo赚钱项目
  • 画册设计制作/seo技术有哪些
  • 免费申请手机网站/百度站长工具网站提交
  • 新余门户网站开发/外贸网站模板
  • 广告设计与制作网站/石家庄今日头条新闻