尽管我有一个 var (kotlin),但无法重新分配 Val?

Val cannot be reassigned although I have a var (kotlin)?

我对数据 class 进行了 var 扩展 属性,如下所示:

var Element.bitmap: Bitmap?
        get() = downloadImg(PLACE_HOLDER_URL) //Default
        set(value) = this.bitmap.let {
            it = value ?: return@let
        }

但是,当我尝试将“它”重新分配给此行中的新值时,编译器抱怨“无法重新分配 Val”:

it = value ?: return@let  

我真的不明白为什么会这样??我有一个“var”位图 属性 而不是“val”,那么问题是什么?更重要的是什么是解决方案或替代方案?

为了创建自定义 setter,您需要通过 field 而不是 it 访问 property 的支持字段,如下所示:

set(value) =  {
            field = value
        }

lambdas里面的变量是不可变的,it变量有一个浅拷贝bitmap

您应该使用 field 分配到支持字段(实际变量)。

set(value) {
    value?.let { field = it }
}

let 函数创建变量的浅表副本,因此如果可见的可变变量 (var) 被另一个线程更改,则可以安全地使用它而不会冒突变的风险。

示例:

class Test {
    var prop: Int? = 5
}

fun main() {
    val test = Test()
    thread {
        Thread.sleep(100)
        test.prop = null
    }
    if (test.prop != null) {
        Thread.sleep(300)  // pretend we did something
        println(test.prop)  // prints null even inside if check
    }
}

为了解决这些情况,使用了浅拷贝,例如 let,它传递了这些的不可变浅拷贝。

class Test {
    var prop: Int? = 5
}

fun main() {
    val test = Test()
    thread {
        Thread.sleep(100)
        test.prop = null
    }
    test.prop?.let {  // `it` is a shallow copy, changes won't be reflected
        Thread.sleep(300)  // pretend we did something
        println(it)  // prints 5
    }
}

结论:it 本身不是实际变量,因此即使您可以将某些内容分配给 it,更改也不会反映到实际变量中.

编辑:扩展属性不能有支持字段,扩展实际上是 getter 和 setter。

  • 您可以做的一件事是制作一个具有唯一标识符的地图,您可以在其中存储值,但可能无法进行垃圾回收
  • 您可以做的另一件事(我推荐)是使用委派

委派示例:

class ElementBitmapDelegate {
    private var value: Bitmap? = null
    
    operator fun getValue(thisRef: Any?, property: KProperty<*>): Bitmap {
        return value ?: downloadImg(PLACE_HOLDER_URL).also { setValue(thisRef, property, it) }
        // return value or if value is null return from downloadImg() and set it to value
    }

    operator fun setValue(thisRef: Any?, property: KProperty<*>, v: Bitmap?) {
        v?.let { value = it }  // if v is not null then set value to v
    }
}


var Element.bitmap: Bitmap? by ElementBitmapDelegate()