Kotlin中的object 与companion object的区别

之前写了一篇Kotlin中常量和静态方法的文章,最近有人提出一个问题,在companion object中调用外部的成员变量会调用不到,这才意识到问题,本篇文章会带着这个疑问来解决问题。

一、 object关键字

object 关键字可以表达两种含义:一种是对象表达式,另一种是 对象声明

1、对象表达式

继承一个匿名对象

val textView = findViewById<TextView>(R.id.tv)
textView.setOnClickListener(object : OnClickListener {
        override fun onClick(p0: View?) {
            Toast.makeText(this@TestActivity, "点击事件生效", Toast.LENGTH_LONG)
        }

})

上面代码其实就是我们经常要给 view 设置的点击事件,OnClickListener 事件是一个匿名类的对象,用object来修饰。

2、对象声明

object 修饰的类为静态类,里面的方法和变量都为静态的。

2.1 直接声明类
object DemoManager {
    private val TAG = "DemoManager"
        
    fun a() {
        Log.e(TAG,"此时 object 表示 声明静态内部类")
    }
    
}
2.2 声明静态内部类

类内部的对象声明,没有被inner 修饰的内部类都是静态的

class DemoManager{
    object MyObject {
        fun a() {
            Log.e(TAG,"此时 object 表示 直接声明类")
        }
    }
}

如果需要调用 a()方法

kotlin中调用
fun init() {
    MyObject.a()
}
java中调用
 MyObject.INSTANCE.a();
 

二、companion object

companion object 修饰为伴生对象,伴生对象在类中只能存在一个,类似于java中的静态方法 Java 中使用类访问静态成员,静态方法。

companion object {
    private val TAG = "DemoManager"

    fun b() {
        Log.e(TAG,"此时 companion objec t表示 伴生对象")
    }
}

kotlin 中调用

fun init(){
       b()
}

java 中调用

DemoManager.Companion.b();

companion object 相关的内容可以查阅 Kotlin中常量和静态方法 这篇文章,在这里不多在具体描述。

三、在companion object中如何调用外部的成员变量

3.1 为什么companion object 中调用不到外部成员变量
class DemoManager {
    private val MY_TAG = "DemoManager"
   
    fun init(){
       b()
   }

    companion object {
        fun b() {
            Log.e(MY_TAG,"此时 companion objec t表示 伴生对象")
        }
    }
}

在上面代码中MY_TAG 是不会被调用到的。

原理很简单:

在java中我们写一个静态方法,如果需要调用成员变量,是无法调用到的

private String TAG = "MainActivity";
  
public static void init(){
        Log.e(TAG,"init() ");
}

只有将 TAG 修改为静态成员变量才能调用到

private static String TAG = "MainActivity";
  
public static void init(){
        Log.e(TAG,"init() ");
}

由此可以看出来,java中静态方法调用成员变量,要求成员变量必须是静态的, 在kotlin 中也是一样,所以当companion object 中调用非静态的成员变量也是调用不到的。

3.2 怎样解决才能调用到呢?
companion object {
    private val MY_TAG = "DemoManager"
    fun b() {
        Log.e(MY_TAG,"此时 companion objec t表示 伴生对象")
    }
}

将所引用的成员变量也修饰静态的,这样就可以引用到了。

posted @   brave-sailor  阅读(1305)  评论(0编辑  收藏  举报
编辑推荐:
· SQL Server 2025 AI相关能力初探
· Linux系列:如何用 C#调用 C方法造成内存泄露
· AI与.NET技术实操系列(二):开始使用ML.NET
· 记一次.NET内存居高不下排查解决与启示
· 探究高空视频全景AR技术的实现原理
阅读排行:
· 阿里最新开源QwQ-32B,效果媲美deepseek-r1满血版,部署成本又又又降低了!
· AI编程工具终极对决:字节Trae VS Cursor,谁才是开发者新宠?
· 开源Multi-agent AI智能体框架aevatar.ai,欢迎大家贡献代码
· Manus重磅发布:全球首款通用AI代理技术深度解析与实战指南
· 被坑几百块钱后,我竟然真的恢复了删除的微信聊天记录!
历史上的今天:
2017-02-13 Android屏幕信息获取
2017-02-13 android 比较靠谱的图片压缩
2017-02-13 Android 图片压缩,基于比例和质量压缩
2017-02-13 android Bitmap类方法属性 详细说明
2017-02-13 Android入门——Bitmap和BitmapFactory
2017-02-13 Android使用BitmapFactory.Options解决加载大图片内存溢出问题
2017-02-13 Android中BitmapFactory.Options详解
点击右上角即可分享
微信分享提示