Ускоряем работу RecyclerView. Лучшие практики оптимизации

Поделиться
HTML-код
  • Опубликовано: 23 ноя 2024

Комментарии • 65

  • @Maribobah81
    @Maribobah81 3 года назад +33

    Позитивный спикер, ему надо задуматься о карьере ютуб-блоггера - приятно смотреть

    • @pbKruasan
      @pbKruasan 3 года назад +4

      спасибо большое ^^

  • @Chernov1984
    @Chernov1984 3 года назад +3

    смотрел это видео в субботу,,,, еле сдержал себя что-бы комп не включить(выходные не для работы! (берегите глазки)).
    Павел спасибо,,, слушаю вас и вспоминаю зачем пошёл программистом работать (последние пару лет - как на каторгу хожу(и бабло не помогает)), видеть как человеку искренне (как мне показалось) интересно и вас заражает (жаль только не надолго).

  • @vladimirtishukov3746
    @vladimirtishukov3746 3 года назад +11

    Надо видео по CustomView! Однозначно! Другого пути нет!

  • @codemachine19
    @codemachine19 Год назад +3

    Паша, спасибо большое за твой труд. Доклад получился суперским, ничего подобного еще не видел. Собрал самые лучшие практики прямо как я и искал.
    Кирилл, спасибо тебе за контент и приглашенного гостя!

  • @dmitriymitroshin7525
    @dmitriymitroshin7525 2 года назад +5

    Огромное спасибо Паше - очень приятно слушать. Кирилл - спасибо за организацию. С Payload лично мне в голову не приходил вопрос про анимацию вставки элемента. Я там в коде увидел, что если не возвращать пейлоды, то не вызывается unbind() и как-то все встало на места. Если мы возвращаем пейлоды - значит хотим мальца сами подправить. А если мы сами берем на себя такую ответственность, то и за нас перетирать вьюшку не нужно. Мне было проще понять именно так.

  • @microdevices
    @microdevices 5 месяцев назад

    Еще не досмотрел, но было бы очень познавательно на каждой итерации оптимизации выполнять замеры производительности и отображать их

  • @tov3801
    @tov3801 10 месяцев назад +1

    Павел очень классный спикер.

  • @mikalaibabrou4515
    @mikalaibabrou4515 3 года назад +4

    По поводу динамического определения максимальной высоты horizontal recycler view относительно текста.
    Я решил эту задачу переопределив on measure где для каждого view item вычислял on measure с максимально длинным текстом, вставив самый длинный текст перед вызовом super.onMeasure. Ну и потом применял measure height для каждого элемента списка.
    P.S. Но во время разработки тоже были кое какие кастыли.
    Хотелось бы увидеть альтернативную реализацию.

  • @maksonic_official
    @maksonic_official 3 года назад +6

    Отличнейшее видео! Спасибо!!!

  • @sherzodnosirov1059
    @sherzodnosirov1059 3 года назад +5

    Замечательный практикум, узнал много нового, большое вам спасибо!

  • @ИгорьЛарцев-д4м
    @ИгорьЛарцев-д4м 3 года назад +6

    Спасибо, очень познавательно и по делу.

  • @МобильныйРазработчик-ъ3с

    Огромная благодарность за проделанный труд,многое для себя узнал!Спасибо, Кирилл! Спасибо, Павел!

  • @AsTaR75256
    @AsTaR75256 3 года назад +6

    Офигенное видео! Спасибо огромное, вы меня спасли :D

  • @sandroisu1274
    @sandroisu1274 2 года назад

    Пересматриваю второй раз, огромное спасибо, всё по полочкам, побольше бы таких видео на разные темы по андроиду

  • @summerwise
    @summerwise 2 года назад

    Поменял notifyDataSetChanged на diffUtils, пока экран не весь заполнен элементами списка, новые элементы добавляются нормально, в верх списка, а остальные сдвигаются вниз. Но как только элементы заполняют весь экран до низа, новые элементы добавляются вверх и их не видно, то есть надо проскроллить вверх, чтоб увидеть их.

  • @alexnerby1295
    @alexnerby1295 3 года назад +4

    Супер материал, хотя и делаю немного по другому. Побольше бы так технологии разбирать, на 2часа) Было бы очень удобно с таймингами на видео, напр. в данном случае по неймингам веток. Спасибо!)

    • @AndroidBroadcast
      @AndroidBroadcast  3 года назад

      Тайминг отрубил RUclips за нарушение (

  • @maxfir
    @maxfir Год назад +1

    Лучшее, что есть по RecyclerView на русском.

  • @КахарманБалтабаев-б2о

    Шикарная инфа и отличная подача!!!!

  • @IlayMMC
    @IlayMMC 2 года назад

    С композом есть свои проблемы, особенно с большим количеством элементов в LazyColumn с LazyRow. Пока это выражается в
    большем лагании, и расходовании памяти. Возможно надо примерять секретные оптимизаци или ждать отпимизаций от гугла. Но RecyclerView на тех же объемах данных работает заметно плавнее.

  • @СемёнНеизвестный-н2ж
    @СемёнНеизвестный-н2ж 3 года назад +1

    Огонь 🔥👏⭐

  • @nazariishostachuk6323
    @nazariishostachuk6323 3 года назад +1

    Вопрос: как правильно добавить тень на группу айтемов ?) Ну конечно вместе с закругленными углами для группы.
    С углами как правило нет проблем, а вот тень это боль.

  • @ИванИванов-в4н9п
    @ИванИванов-в4н9п 2 года назад

    Классный урок. Паша монстр!

  • @vitaliyashchuk8358
    @vitaliyashchuk8358 3 года назад +1

    Да, и видос отличный получился

  • @aldredo5543
    @aldredo5543 2 года назад

    Очень полезный выпуск 👍

  • @WoffkaG94
    @WoffkaG94 3 года назад +6

    Спасибо за видео, но очень затянуто.
    И мне кажется это решение не очень удобное: для простого примера надо по 3 класса по 100 строк кода, хотя большая часть из него (дифутилсы, сам холдер, создание холдера, проброс тайпов и т.д.) просто копипаста из класса в класс.
    Плюс в UI очень много поисков по мапе с кастами и прочего, что не сильно про оптимизацию. Ну и очень много логики с ластАдаптерПозишен, каррентАдаптерПозишен и прочим, что надо везде учитывать. + Один код во фрагменте (декораторы), второй в айтеме (ещё что-то), третий во вьюхолдере (биндинги), четвёртый ещё где-то. Сложно найти кто и где выставил этот декоратор и почему.
    Мы пошли по-другому пути: у нас ВМ в бэке создаёт "ячейки" c готовыми кликлистенерам, декораторам, данными и всем остальным, что понадобится для отображения или для уведомления ВМ о каких-то ивентах.
    Эти ячейки - это интерфейс, который имеют сразу дефолтный ViewHolder/onCreateViewHolder/DiffUtils/клик (но всё можно переопределить там же), так же все остальные методы вьюхолдера.
    А всё обрабатывает либовский адаптер (он просто делегирует всё, но без кастов, как здесь).
    То есть ячейка сразу рисует всё, что ей дали, без проверок "я последний айтем?" или "я после айтема типа Х?", т.к. ей ВМ сразу даст нужный декоратор (именнно декоратор, а не паддинги), что надо с таким отступом снизу или с таким разделителем сверху и т.д., т.к. она знает, какие айтемы находятся возле этой ячейки.
    Это всё без кодгена и т.д в либе на 100+- строк. (создание холдера для каждого айтема в списке можно сделать своё: databinding/viewbinding/customView/кодом).

    • @pbKruasan
      @pbKruasan 3 года назад +2

      я старался показать чуть более приземленные к RV вещи чтобы осветить как можно больше тем. возможно ваше решение действительно удобнее и гибче. а возможность посмотреть на эту либу?

    • @WoffkaG94
      @WoffkaG94 3 года назад +1

      @@pbKruasan Пока что либа приватная. Возможно опубликуем скоро, а может и расскажем, если попросят.

    • @pbKruasan
      @pbKruasan 3 года назад

      @@WoffkaG94 я бы послушал. я соглашусь с тем что размазывать код связанный элементами списка по приложению не самая удобная затея. касательно декораторов наверно даже оптимальнее будет прокидывать все необходимые отступы аналогично тому как я указал фиксированную длинну списка(не говоря уже о том что это и правда удобнее когда все рядом лежит). Про DiffUtil видимо у вас есть дефолтная имплементация где вы сравниваете по умолчанию хешкоды и по equals элементы. В любом случае очень интересно взглянуть на ваше решение! буду ждать новостей)

    • @WoffkaG94
      @WoffkaG94 3 года назад +5

      ​@@pbKruasan
      Я опубликовал версию
      github.com/netcosports/CompositeAdapter_Android
      DiffUtils основан на том, что у нас должен быть kotlin data class или класс с нормальным equals - а значит мы можем их сравнить по дефолту (и в редких случаях, если это надо, дать возможность переопределить это). А так же у класса должен быть какой-то ID. Раньше у нас был интерфейс, но это создавало неудобства. Теперь надо явно его передать.
      Хэшкод не подойдёт, т.к. нам надо знать, что это тот же объект, у которого поменялось какое-то поле.
      Примерно на таком принципе и строится всё остальное: создание вьюхолдера, передача типов, передача лаяутов, передача кликов и т.д.. Они в 99% одинаковые, а значит можно их не копипастить.
      А так же декораторы, которые "биндятся" к вьюхолдеру почти как "дата". Их высчитывает вьюмоделька при создании списка, а не фрагмент, который на момент создания вью не знает, какие данные придут в адаптер одной из его вьюх и навряд ли должен знать.
      Декораторы участвуют в диффах тоже, т.к. если мы удалим последний айтем, то у предпоследнего дата не поменяется, но поменяется декоратор (допустим, у последнего айтема отступ снизу больше).

  • @inquisitor4894
    @inquisitor4894 Год назад +1

    Ооочень сложно, но интересно). 60% всего не понял

  • @mukhtarbimurat5106
    @mukhtarbimurat5106 3 года назад +1

    Спасибо очень полезно

  • @torskandinav4634
    @torskandinav4634 3 года назад +5

    Хочу КастомВью ), только что нибудь сложное а не аватарку с анимашкой

    • @AndroidBroadcast
      @AndroidBroadcast  3 года назад +1

      Что например?

    • @torskandinav4634
      @torskandinav4634 3 года назад +3

      @@AndroidBroadcast ну можно что то типа графиков или диаграмм. Аля Диаграмма Ганта или еще что-нибудь

    • @vitalyraevsky
      @vitalyraevsky 3 года назад +1

      @@torskandinav4634 график свечей по акциям с интерактивом ?

    • @russabit7894
      @russabit7894 3 года назад

      @@AndroidBroadcast можно хотя бы click to expand

  • @alexeykrachkovsky8454
    @alexeykrachkovsky8454 2 года назад +1

    Custom view! :)

  • @hombre2355
    @hombre2355 2 года назад

    16:33 А, собственно, насколько правильно держать Drawable внутри каждого элемента списка, вместо его id? Мы ведь не знаем насколько большой у нас Drawable.

    • @MrPwnzrus
      @MrPwnzrus 2 года назад

      А ещё там при сравнении элементов с драваблами а не со ссылками / айдишниками будут проблемы

    • @alexeysokolov930
      @alexeysokolov930 Год назад

      После того, как вместо констант вью тайп он вставил id лейаутов, дальше можно не смотреть)

  • @sphinx489
    @sphinx489 3 года назад

    почему имя адаптера называется Fingerprint

  • @РоманШаров-ы7ч
    @РоманШаров-ы7ч 3 года назад +1

    очень полезное видео

  • @EvgeniyGooDiBunakov
    @EvgeniyGooDiBunakov 3 года назад

    Как корректно сравнивать в DiffUtil наследников Sealed класса?

    • @AndroidBroadcast
      @AndroidBroadcast  3 года назад

      Так вы можете вызывать equals, либо перебирать все sealed классы в when и собственным правилом сравнивать

  • @4004karp
    @4004karp 3 года назад

    10:30 no new line at end of file

  • @АлександрГребеньков-ы5й

    приложение падает при удалении последнего элемента

    • @pbKruasan
      @pbKruasan 3 года назад

      верно! спасибо за информацию! чуть позже поправлю

  • @СергейПанов-з3ц
    @СергейПанов-з3ц 3 года назад

    33:32
    Насколько правильно с точки зрения оптимизации делать так как в видео?
    1. Создавать новую копию Item'а с изменённым состоянием like'а
    2. Создавать 2 раза новые ArrayList'ы - один раз в методе toList, второй раз новая копия ArrayList'а создаётся внутри ListAdapter'a
    3. Считать для всего списка DiifCalback.
    Если список будет длинный, то данное действие может занять заметный пользователю промежуток времени.
    На мой взгляд оптимальнее было бы сделать поле like'а изменяемым (var) и вызвать метод notifyItemChanged(pos:Int, payload:Any?)
    private fun onSavePost(post: UserPost) {
    val postIndex = feed.indexOf(post)
    post.isSaved = !post.isSaved
    adapter.notifyItemChanged(postIndex,post.isSaved)
    }

    • @pbKruasan
      @pbKruasan 3 года назад +6

      1. Как я говорил, делал я копию в демонстрационнных целях. В этом случае мы будем терять в том что создаем новый объект и запускаем DiffUtil, в качестве альтернативы конечно можно делать чтобы при клике ViewHolder без посредников сам обновлял свое состояние а вам бы кидал коллбек что нужно отправить соответствующий запрос на сервер (ну и плюс както обрабатывать еще когда нам сервер вернет ошибку). Можно конечно сделать и так, но тут я бы задался вопросом - а точно ли обновление своего состояния задача ViewHolder? Ведь мы когда пишем фрагмент то абсолютно всю логику уносим во ViewModel/Presenter, тут кажется вполне аналогичная история. А говорить вам как делать я не могу, поэтому тут уже вам нужно выбрать что вам подходить больше.
      2. Да про дополнительный вызов `.toList()` это на самом деле мой косяк, в нем нет необходимости, не успел провести рефакторинг(
      3. Да, ваш пример вполне валидный. Но тут я бы исходил из необходимости, ведь если у вас в коде есть DiffUtil то другой разработчик не будет ожидать что вы еще гдето в обход DiffUtil обновляете список и если будут возникать какие-либо ошибки то это может увеличить время поиска проблемы. Я бы все же попробовал все изменения прокидывать через DiffUtil, и если вдруг пользователи начали бы жаловаться на проблемы тогда бы уже использовал `adapter.notifyItemChanged(postIndex,post.isSaved)` и чтоб совсем красиво было еще бы покрыл все это дело тестами, написал бы документацию/комментарии чтоб у другого разработчика было меньше шансов чтото поломать

    • @СергейПанов-з3ц
      @СергейПанов-з3ц 3 года назад +2

      @@pbKruasan спасибо за развернутый ответ.

  • @Polite_person_
    @Polite_person_ Год назад

    LazyColumn 🗿🗿🗿

  • @СергейПанов-з3ц
    @СергейПанов-з3ц 3 года назад

    1:45
    "Наклеечка символизирует что я из Краснодара"
    -на наклеечке изображена мусорка
    Как иронично...

    • @pbKruasan
      @pbKruasan 3 года назад +3

      ну это один из символов города) что у нас и мусорки красивые) Варламов рекомендует)

  • @dmytromarchuk3023
    @dmytromarchuk3023 3 года назад +1

    В закладки

  • @sphinx489
    @sphinx489 3 года назад +1

    contact adapter😂

  • @ЕвгенийМоскаленко-у9щ

    CustomView

  • @alexde6460
    @alexde6460 2 года назад

    Еще больше рекламы засунуть не пробовал????