区别

  1. object修饰的类是单例的
  2. companion object用于修饰类中的伴生对象

    对象表达式、对象声明与伴生对象 - Kotlin 语言中文站

object关键字

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

对象表达式

继承一个匿名对象
也可以说是匿名表达式

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

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

对象声明

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

直接声明类

  1. object DemoManager {
  2. private val TAG = "DemoManager"
  3. fun a() {
  4. Log.e(TAG,"此时 object 表示 声明静态内部类")
  5. }
  6. }

声明静态内部类

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

  1. class DemoManager{
  2. object MyObject {
  3. fun a() {
  4. Log.e(TAG,"此时 object 表示 直接声明类")
  5. }
  6. }
  7. }

如果需要调用 a()方法

kotlin中调用
  1. fun init() {
  2. MyObject.a()
  3. }

java中调用
MyObject.INSTANCE.a();

companion object

类内部的对象声明可以用 companion 关键字标记,companion关键字不能使用在外部类上
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中如何调用外部的成员变量

为什么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表示 伴生对象")
    }
}

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

作者:会撒娇的犀犀利 链接:https://www.jianshu.com/p/14db81e1576a 来源:简书