在 Kotlin 数据类中调用超类构造函数之前的访问函数

2022-09-04 08:43:28

我在 Kotlin 中使用数据类来显著减少我必须编写的 Java 代码量。

但是,在我的一个Java类中,我不确定该怎么做才能在Kotlin中实现相同的结果。

我的 Java 类看起来有点像这样:

public class DataObject {

    private int mId;
    private String mName;

    public DataObject(int id, String name) {
        mId = id;
        mName = name;
    }

    public DataObject(Context context, int id) {
        mId = id;
        Cursor cursor = ...
        cursor.moveToFirst();
        mName = cursor.getString(...);
        cursor.close();
    }

    public int getId() {
        return mId;
    }

    public String getName() {
        return mName;
    }

}

我试图在Kotlin中重写它,到目前为止,我有这个:

data class DataObject(val id: Int, val name: String) {

    constructor(context: Context, id: Int) : this(id, fetchName(context))

    private fun fetchName(context: Context): String {
        val cursor = ...
        cursor.moveToFirst()
        val name = cursor.getString(...)
        cursor.close()
        return name
    }

}

但是我的IDE(Android Studio)强调了我用红色调用的部分。它显示以下消息:fetchName(context)constructor

在调用超类构造函数之前无法访问fetchName

我应该如何解决此问题?


答案 1

只能在完全构造的对象上使用成员函数。解决此问题的一种方法是使用私有扩展函数或只是一个函数来获取名称:

private fun Context.fetchName(): String {
    ///...
    return cursor.getString(1)
}

data class DataObject(val id: Int, val name: String) {
    constructor(context: Context, id: Int) : this(id, context.fetchName())
}

虽然我确实认为使用对于.我会像这样使用单独的工厂Cursorconstructor

data class DataObject(val id: Int, val name: String) {
    object FromCursorFactory {
        fun create(id: Int, context: Context): DataObject {
            val name = context.fetchName()
            return DataObject(id, name)
        }
    }
}

延伸阅读:


答案 2

另一种方法是使用伴随对象。这将允许您在数据类之外调用函数(在您的特定情况下可能没有用)

data class DataObject(val id: Int, val name: String) {

  constructor(context: Context, id: Int) : this(id, fetchName(context))

  companion object {

    fun fetchName(context: Context): String {
      val cursor = ...
      ...
      return name
    }
  }
}

推荐