为什么这个函数在 Jetpack Compose 中被多次调用?
Why this function is called multiple times in Jetpack Compose?
我目前正在尝试 Android Compose。我有一个显示加密硬币价格的文本。如果价格上涨,文本颜色应为绿色,但如果价格下跌,则应为红色。当用户单击按钮时调用该函数。问题是函数 showPrice()
被调用了多次(有时只调用一次,有时调用 2-4 次)。因此,用户会看到错误的颜色。我该怎么做才能确保只调用一次?
主要活动:
@Composable
fun MainScreen() {
val priceLiveData by viewModel.trackLiveData.observeAsState()
val price = priceLiveData ?: return
when (price) {
is ViewState.Response -> showPrice(price = price.data)
is ViewState.Error -> showError(price.text)
}
Button(onClick = {viewModel.post()} )
}
@Composable
private fun showPrice(price: Double) {
lastPrice = sharedPref.getFloat("eth", 0f).toDouble()
val color by animateColorAsState(if (price >= (lastPrice)) Color.Green else
Color.Red)
Log.v("TAG", "last=$lastPrice new = $price")
editor.putFloat("eth", price.toFloat()).apply()
Text(
text = price.toString(),
color = color,
fontSize = 28.sp,
fontFamily = fontFamily,
fontWeight = FontWeight.Bold
)
}
ViewModel:
@HiltViewModel
class MyViewModel @Inject constructor(
private val repository: Repository
): ViewModel() {
private val _trackLiveData: MutableLiveData<ViewState<Double>> = MutableLiveData()
val trackLiveData: LiveData<ViewState<Double>>
get() = _trackLiveData
fun post(
) = viewModelScope.launch(Dispatchers.Default) {
try {
val response = repository.post()
_trackLiveData.postValue(ViewState.Response(response.rate.round(7)))
} catch (e: Exception) {
_trackLiveData.postValue(ViewState.Error())
Log.v("TAG: viewmodelPost", e.message.toString())
}
}
}
视图状态:
sealed class ViewState<out T : Any> {
class Response<out T : Any>(val data: T): ViewState<T>()
class Error(val text:String = "Unknown error"): ViewState<Nothing>()
}
所以当我按下按钮调用 showPrice() 时。我可以在日志中看到这些行:
2021-06-10 16:39:18.407 16781-16781/com.myapp.myapp V/TAG: last=2532.375732421875 new = 2532.7403716
2021-06-10 16:39:18.438 16781-16781/com.myapp.myapp V/TAG: last=2532.740478515625 new = 2532.7403716
2021-06-10 16:39:18.520 16781-16781/com.myapp.myapp V/TAG: last=2532.740478515625 new = 2532.7403716
What can I do to ensure that it's only called once?
没什么,这就是它的工作方式。在 View 系统中你不会问“为什么我的视图失效了 3 次?”。框架根据需要使视图无效(重组),您不需要知道或关心这种情况何时发生。
您的代码的问题是您的可组合项正在从首选项中读取旧值,这不是它应该如何工作的,该值应该作为状态的一部分由视图模型提供。不要只提供新价格,而是公开一个包含新价格和旧价格的数据 Class,然后在您的可组合项中使用这 2 个值来确定要显示的颜色,或者公开要使用的价格和颜色。
我目前正在尝试 Android Compose。我有一个显示加密硬币价格的文本。如果价格上涨,文本颜色应为绿色,但如果价格下跌,则应为红色。当用户单击按钮时调用该函数。问题是函数 showPrice()
被调用了多次(有时只调用一次,有时调用 2-4 次)。因此,用户会看到错误的颜色。我该怎么做才能确保只调用一次?
主要活动:
@Composable
fun MainScreen() {
val priceLiveData by viewModel.trackLiveData.observeAsState()
val price = priceLiveData ?: return
when (price) {
is ViewState.Response -> showPrice(price = price.data)
is ViewState.Error -> showError(price.text)
}
Button(onClick = {viewModel.post()} )
}
@Composable
private fun showPrice(price: Double) {
lastPrice = sharedPref.getFloat("eth", 0f).toDouble()
val color by animateColorAsState(if (price >= (lastPrice)) Color.Green else
Color.Red)
Log.v("TAG", "last=$lastPrice new = $price")
editor.putFloat("eth", price.toFloat()).apply()
Text(
text = price.toString(),
color = color,
fontSize = 28.sp,
fontFamily = fontFamily,
fontWeight = FontWeight.Bold
)
}
ViewModel:
@HiltViewModel
class MyViewModel @Inject constructor(
private val repository: Repository
): ViewModel() {
private val _trackLiveData: MutableLiveData<ViewState<Double>> = MutableLiveData()
val trackLiveData: LiveData<ViewState<Double>>
get() = _trackLiveData
fun post(
) = viewModelScope.launch(Dispatchers.Default) {
try {
val response = repository.post()
_trackLiveData.postValue(ViewState.Response(response.rate.round(7)))
} catch (e: Exception) {
_trackLiveData.postValue(ViewState.Error())
Log.v("TAG: viewmodelPost", e.message.toString())
}
}
}
视图状态:
sealed class ViewState<out T : Any> {
class Response<out T : Any>(val data: T): ViewState<T>()
class Error(val text:String = "Unknown error"): ViewState<Nothing>()
}
所以当我按下按钮调用 showPrice() 时。我可以在日志中看到这些行:
2021-06-10 16:39:18.407 16781-16781/com.myapp.myapp V/TAG: last=2532.375732421875 new = 2532.7403716
2021-06-10 16:39:18.438 16781-16781/com.myapp.myapp V/TAG: last=2532.740478515625 new = 2532.7403716
2021-06-10 16:39:18.520 16781-16781/com.myapp.myapp V/TAG: last=2532.740478515625 new = 2532.7403716
What can I do to ensure that it's only called once?
没什么,这就是它的工作方式。在 View 系统中你不会问“为什么我的视图失效了 3 次?”。框架根据需要使视图无效(重组),您不需要知道或关心这种情况何时发生。
您的代码的问题是您的可组合项正在从首选项中读取旧值,这不是它应该如何工作的,该值应该作为状态的一部分由视图模型提供。不要只提供新价格,而是公开一个包含新价格和旧价格的数据 Class,然后在您的可组合项中使用这 2 个值来确定要显示的颜色,或者公开要使用的价格和颜色。