尽管我有一个 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()
我对数据 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()