Неожиданные рекомпозиции объектов Composable, которые принимают лямбда-функции в качестве аргумента.Android

Форум для тех, кто программирует под Android
Ответить Пред. темаСлед. тема
Anonymous
 Неожиданные рекомпозиции объектов Composable, которые принимают лямбда-функции в качестве аргумента.

Сообщение Anonymous »

У меня есть функция Composable, имеющая одну кнопку, которая принимает лямбда-выражение в качестве аргумента для параметра onClick кнопки. Я определяю пустую функцию в своей ViewModel и предоставляю ее этому Composable. Также во ViewModel есть список, в который периодически (каждую секунду) добавляется новый элемент. Список предоставляется другому Composable, который содержит LazyList.
Моя проблема в том, что всякий раз, когда элемент добавляется в список, Composable с кнопкой перекомпоновывается. , даже несмотря на то, что у этой функции только один аргумент — пустая лямбда-функция, которая ничего не делает.
Вот фрагмент MainActivity:

Код: Выделить всё

class MainActivity : ComponentActivity() {
private val mainViewModel by viewModels()

override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContent {
RecompositionTestTheme {
Surface(
modifier = Modifier.fillMaxSize(),
color = MaterialTheme.colorScheme.background
) {
MainScreen(mainViewModel)
}
}
}
}
}

@Composable
fun MainScreen(
viewModel: MainViewModel = viewModel(),
) {
Column {
TopBar(viewModel::doNothing)
CenterScreen(viewModel.deviceList)
}
}

@Composable
fun TopBar(
dummyAction: () -> Unit
) {
Row {
Button(onClick = dummyAction) {
Text("Do nothing")
}
}
}

@Composable
fun CenterScreen(deviceList: List) {
LazyColumn {
items(deviceList) {
Text("$it")
}
}
}
И MainViewModel:

Код: Выделить всё

data class DeviceData(
val name: String,
var value: Int
)

class MainViewModel : ViewModel() {

private val _deviceList = mutableStateListOf()
val deviceList: List
get() = _deviceList.toList()

init {
// Simulating reception of data (for example, during
// Bluetooth LE Scan) every second.
flow {
while (true) {
delay(1000L)
emit(Unit)
}
}.onEach {
_deviceList += DeviceData("", 1)
}.launchIn(viewModelScope)
}

fun doNothing() {
}
}
Я протестировал рекомпозицию с помощью LayourManager, а также добавив println() в функцию TopBar. Каждую секунду, когда в LazyList добавляется новый элемент, компонуемые элементы TopBar() и Button перестраиваются, хотя это не должно происходить.
DoNothing() Функция изначально что-то делала, но я упростил, потому что результат был тот же. Например, он может очистить _deviceList.
Странно то, что если вместо функции я определяю doNothing как переменную, вызывающую исходный doNothing , она внезапно начинает вести себя правильно и не перекомпоновывается.
Я имею в виду следующее:

Код: Выделить всё

fun doNothing() {
}

val doNothingVal: () -> Unit = {
doNothing()
}

Тогда я называю это так:

Код: Выделить всё

TopBar(viewModel.doNothingVal)
Итак, я не понимаю следующих двух вещей:
  • Почему оригинальный метод doNothing()< Функция /code> в первую очередь вызывает рекомпозицию.
Ее можно использовать как лямбда-выражение или как ссылку на член, без разницы:

Код: Выделить всё

TopBar({ viewModel.doNothing() })
// or
TopBar(viewModel::doNothing)
  • Почему вместо этого использование функциональной переменной не приводит к перекомпозиции?
Спасибо.

Подробнее здесь: https://stackoverflow.com/questions/768 ... as-argumen
Реклама
Ответить Пред. темаСлед. тема

Быстрый ответ

Изменение регистра текста: 
Смайлики
:) :( :oops: :roll: :wink: :muza: :clever: :sorry: :angel: :read: *x)
Ещё смайлики…
   
К этому ответу прикреплено по крайней мере одно вложение.

Если вы не хотите добавлять вложения, оставьте поля пустыми.

Максимально разрешённый размер вложения: 15 МБ.

  • Похожие темы
    Ответы
    Просмотры
    Последнее сообщение

Вернуться в «Android»