Как стать автором
Обновить

Метод setState() во Flutter может работать не так, как вы это представляете

Уровень сложностиПростой
Время на прочтение3 мин
Количество просмотров4.2K
Автор оригинала: Iiro Krankka

Вот несколько неловкое предположение, которое я сделал относительно setState, когда начал изучать Flutter почти 4 года назад.

Все мы знаем setState из примера со счетчиком:

class _MyWidgetState extends State<MyWidget> {
  int _counter = 0;

  void _incrementCounter() {
    setState(() {
      _counter = _counter + 1;
    });
  }

  @override
  Widget build(BuildContext context) {
    return Text("The current value is $_counter!");
  }
}

Вот, как я думал, это работало:

  1. Есть одно stateful поле под названием _counter.

  2. У нас также есть виджет Text, который отображает значение _counter.

  3. Каждый раз, когда мы хотим обновить _counter, мы также должны обернуть его в анонимную функцию, которую нужно передать в setState. В противном случае фреймворк не будет знать, что было обновлено.

Через четыре месяца моего путешествия по Flutter я обнаружил, что предположение номер три не соответствует действительности.

Мы действительно должны вызывать setState при обновлении _counter, но нет абсолютно никакой необходимости делать это в анонимном обратном вызове.

Это:

setState(() {
  _counter = _counter + 1;
});

ровно то же самое, что и:

_counter = _counter + 1;
setState(() {});

Фреймворк Flutter не проверяет магическим образом код внутри анонимной функции и не выполняет некоторые сравнения, чтобы увидеть, что изменилось.

Каждый раз, когда вы вызываете setState, виджет перестраивается – даже если ничего не изменилось и обратный вызов, переданный в setState, был пуст.

То, что я изначально предполагал о setState, было полностью выдумано в моей голове.

Заглянуть за кулисы

Давайте посмотрим, как выглядит код, лежащий в основе setState:

@protected
void setState(VoidCallback fn) {
  assert(fn != null);
  assert(() {
    if (_debugLifecycleState == _StateLifecycle.defunct) {
      throw FlutterError.fromParts([/* ... */]);
    }
    if (_debugLifecycleState == _StateLifecycle.created && !mounted) {
      throw FlutterError.fromParts([/* ... */]);
    }
    return true;
  }());
  final dynamic result = fn() as dynamic;
  assert(() {
    if (result is Future) {
      throw FlutterError.fromParts([/* ... */]);
    }
    return true;
  }());
  _element.markNeedsBuild();
}

Давайте уберем утверждения:

@protected
void setState(VoidCallback fn) {
  final dynamic result = fn() as dynamic;
  _element.markNeedsBuild();
}

На самом деле мы можем свести все к минимуму:

@protected
void setState(VoidCallback fn) {
  fn();
  _element.markNeedsBuild();
}

Всё, что делает setState – это вызывает предоставленный обратный вызов, после чего связанный элемент становится запланированным для новой сборки.

Здесь нет магической диффузии полей.

Так зачем же тогда нам setState?

Благодаря исследованиям UX, проведенным командой Flutter.

Они делают очень много.

Еще в 2017 и 2018 годах на GitHub существовал ярлык проблемы под названием "первый час". Команда Flutter провела UX-исследования для разработчиков и проследила, как новички будут использовать Flutter, если оставить их на час одних. Эти исследования будут определять будущие решения по API для Flutter.

Очевидно, что функция setState является одной из таких находок.

Из статьи на GitHub, которая помогла мне понять, что я жил во лжи:

Раньше у нас был просто метод markNeedsBuild, но мы обнаружили, что люди вызывают его как талисман на удачу – в любой момент, когда они не были уверены, нужно ли вызывать этот метод, они вызывали его.

Мы перешли на метод, который принимает (синхронно вызываемый) обратный вызов, и внезапно у людей стало гораздо меньше проблем с этим.

Есть также ответ на StackOverflow от Collin Jackson из команды Flutter:

Когда во Flutter была функция "markNeedsBuild", разработчики в итоге просто вызывали ее в произвольное время. Когда синтаксис изменился на setState((){ }), разработчики стали гораздо чаще использовать API правильно.

Таким образом, весь API setState – это просто умственный трюк. И, похоже, это работает – очевидно, это привело к тому, что люди стали реже перестраивать свои виджеты.

У меня никогда не было проблем с пониманием того, когда вызывать setState, но я определенно вижу, что задаюсь вопросом "кто такой Марк и что он строит?".

Тайна раскрыта.

Заключение

Что всё это значит?

Должны ли мы теперь преобразовать все наши вызовы setState в нечто подобное?

_counter = _counter + 1;
setState(() {});

Скорее всего, нет.

Несмотря на то, что нет никакой разницы, вызов setState с анонимным обратным вызовом кажется правильным.

На данный момент анонимный обратный вызов является устоявшейся традицией, и все остальное кажется странным.


Материал переведён Ruble.

Теги:
Хабы:
Всего голосов 9: ↑8 и ↓1+7
Комментарии2

Публикации

Истории

Работа

Ближайшие события

27 августа – 7 октября
Премия digital-кейсов «Проксима»
МоскваОнлайн
20 – 22 сентября
BCI Hack Moscow
Москва
24 сентября
Конференция Fin.Bot 2024
МоскваОнлайн
24 сентября
Astra DevConf 2024
МоскваОнлайн
25 сентября
Конференция Yandex Scale 2024
МоскваОнлайн
28 – 29 сентября
Конференция E-CODE
МоскваОнлайн
28 сентября – 5 октября
О! Хакатон
Онлайн
30 сентября – 1 октября
Конференция фронтенд-разработчиков FrontendConf 2024
МоскваОнлайн
3 – 18 октября
Kokoc Hackathon 2024
Онлайн
7 – 8 ноября
Конференция byteoilgas_conf 2024
МоскваОнлайн