Под катом адаптированный перевод статьи, в которой разработчики JetRuby Agency делятся впечатлениями о React: что использовали, что не использовали и что еще только планируют использовать.
Почти год прошел, как мы начали применять ReactJS и переваривать море сопутствующих технологий. И мы здесь, чтобы поделиться с вами своим опытом.
Недавно мы закончили проект, который использовал рендеринг на стороне клиента и API сервер на Rails. Общее впечатление – глубоко положительное.
Что нам понравилось:
- Хорошие и понятные исходники. Декларативный стиль React позволяет легко понимать код и избегать ненужной сложности. Которая так и норовит скопиться в UI.
- Явное разграничение на серверный и клиентский код. Можно легко разделять задачи между фронтенд и бекенд разработчиками, и они не будут друг с другом пересекаться.
- Благодаря технологии Virtual DOM изменения страницы происходят очень быстро (за исключением вырожденных случаев вроде игр или чего-то столь же интерактивного). Но стоит заметить, что в последнее время мы часто встречаем статьи (например, вот эта), авторы которых утверждают, что React медленный.
- Если вы хотите открыть API наружу, то разработка SPA приложения для него будет лучшим из возможных тестов. Причем код, который взаимодействует с бекендом, можно вынести в один или несколько отдельных модулей: это намного лучше, чем хардкодить AJAX запросы в Flux. К примеру, модуль для работы с настройками может делать разные запросы в зависимости от того, запущено приложение на проде или на стейдже.
- Большое количество разных реализаций Flux. Хорошо, если вы любите держать руку на пульсе технологий. Плохо, если просто хотите быстро что-то сделать и не знаете, что выбрать. Некоторые реализации используют несколько «сторов», другие – один глобальный объект с FRP или эвентами для координирования компонентов. Всегда можно выбрать то, что больше нравится. Мы выбрали «Fluxxor», но не используем его реализацию сторов, только actions. Для работы с состоянием мы выбрали подход с одним глобальным объектом и используем библиотеку «Baobab». Со стороны выглядит как переусложнение, но нам в компании нравится гибкость.
Естественно, в этом новом прекрасном мире не все так прекрасно. С рядом проблем мы все же столкнулись:
- Недостаток хорошо поддерживаемых, готовых к использованию компонентов. Конечно, это вопрос курицы и яйца – их количество увеличится вместе с ростом популярности самого React. Но мы-то живем здесь и сейчас.
- Формы и валидация. Их сложно делать. Плюс готовых высокоуровневых библиотек для создания форм не то чтобы много.
- Для своего приложения мы использовали кастомную тему с большим количеством кода jQuery. Оказалось, что её не так просто бесшовно интегрировать в React приложение! Технически возможно, но я бы не рекомендовал заниматься этим, если у вас есть уже готовые компонент или ресурсы, чтобы такой сделать с нуля. В нашем случае был нужен date time picker на bootstrap, и мы не нашли хорошего готового (этот не обладал всеми нужными нам функциями). Так что у нас не оставалось выбора, кроме как оформить подходящий jQuery плагин в виде компонента React. Если вам интересна эта тема, то вы можете прочитать больше здесь.
Несколько технических решений, которые нам кажутся правильными и которые мы хотим рекомендовать:
- Используйте один стор. И вот почему. Во-первых, основная проблема с несколькими сторами – их синхронизация. Если вам нужны данные из одного стора, для которых нужны данные из другого стора, вам придется прописывать это в явном виде с помощью функции waitFor(). С ростом приложения таких связей будет все больше и вам будет все труднее ими управлять. Использование одного стора для управления состояниям решает подобные проблемы. Хорошим кандидатом для такого стора является реализация из библиотеке «Baobab». Можете рассматривать его как разновидность локальной базы данных, которая живет на фронтенде.
- Используйте webpack для сборки модулей. Сейчас для такой сборки есть множество решений, к примеру, grunt или gulp совместно с browserify или requirejs, но webpack немного мощнее. Прямое включение css, картинок и шрифтов в React компоненты позволяет сделать css более управляемым.
- Используйте курсоры, чтобы передавать данные React компонентам. Эта мощная абстракция позволяет не передавать промежуточные данные вниз по «цепочке» компонентов, что сильно упрощает код. Мы использовали курсоры из Baobab, но есть много других хороших реализаций. Baobab был выбран из-за его простоты, прямолинейности и готовой интеграции с React, поддерживающей PureRenderMixin
Что мы хотим попробовать в следующем приложении:
Использовать Babel
Babel уже официально используется React для работы с JSX. Но, даже если вы не используете JSX, Babel все равно является великолепным инструментом для фронтенд разработки. Кто не хочет пользоваться всеми этими крутыми возможностями ES6 и ES7, не дожидаясь их поддержки в браузерах?
Изоморфные приложения
Прямо сейчас это тренд. Если еще не слышали о таком подходе, вот хороший обзор от команды Airbnb. Идея состоит в том, чтобы отрендерить HTML страницу на сервере, передать ее на клиент (с кешированием и CDN) чтобы она мгновенно отобразилась в браузере, а затем, когда загрузиться JavaScript, ReactJS «подцепится» к уже отрендеренной странице и приложение «оживет».
Использовать Functional Reactive Programming (FRP) для координации (как диспатчер Flux)
Есть несколько хороших библиотек для использования FRP в JavaScript (RxJS, Bacon, Kefir). Основная идея в том, чтобы представить переменные как последовательность их изменений, и комбинировать эти изменения с помощью таких функций как map, filter, reduce и функций высшего порядка (функция, которая принимает другую функцию в качестве аргумента). Для пользовательского интерфейса такой подход дает возможность преобразовать последовательность эвентов в поток эвентов и манипулировать такими потоками как объектами.
Один стор для управления состоянием
Дает всего одно, но огромное преимущество. Так как в компонентах React нет локального состояния, то можно рассматривать UI как одну чистую функцию (термин из функционального программирования) которая получает одно состояние на вход в виде аргумента и возвращает пользовательский интерфейс (точнее, его React-представление), соответствующий этому состоянию. На практике такой подход дает возможность отладки в реальном времени, undo и redo, time travel. Здесь можно посмотреть, как это все сделано:
- debug.elm-lang.org/edit/Stamps.elm ?– ?дебаггер для Elm
- www.youtube.com/watch?v=Fo86aiBoomE – то же самое, но имплементировано в JS, используется дебагер из библиотеки cerebraljs
- www.youtube.com/watch?v=5yHFTN-_mOo – т?о, что сделал парень из CircleCI, ребята используют ClojureScript Om, в котором также используется один объект для хранения состояния. Более детальное описание, как у них все это работает
Такой подход выводит тестирование и отладку веб приложений на принципиально новый уровень. Больше не нужны скриншоты и длинный список «шагов воспроизведения». Если вы встретили баг – просто скопируйте текущее состояние приложения и отдайте его разработчикам.
Иммутабельные структуры данных
В ряде случаев их использование позволяет написать более быстрый код. В котором, к тому же, будет меньше багов из-за отсутствия мутабельных объектов и значений. Несмотря на то, что Baobab не предлагает иммутабельных структур данных (только персистентные), он не рекомендует напрямую изменять дерево данных, а предлагает использовать для этого функции API.
Автор изображения (перед катом) – Stefpet, Сreative Commons 2.0.
Комментарии (78)
Miraage
22.08.2016 13:08+1Используйте один стор
А лучше — Redux.
Причем код, который взаимодействует с бекендом, можно вынести в один или несколько отдельных модулей
Это надо делать априори ввиду Single Responsibility.
… PureRenderMixin
Уже есть PureComponent.
Makaveli
22.08.2016 13:25+5Прямо сейчас это тренд… отрендерить HTML страницу на сервере, передать ее на клиент (с кешированием и CDN) чтобы она мгновенно отобразилась в браузере ...
Воистину история движется по спирали.orcy
22.08.2016 15:48Какой тогда будет следующий шаг спирали? Можно например переслать js код изоморфного сервера на клиент, чтобы клиент обращался к серверу как к локальной машиной, и тот бы ему возвращал либо отрендеренный HTML или HTML с JavaScript который все отрендерит через Ajax к локальному или удаленному серверу.
Makaveli
22.08.2016 16:01Ну пока я вижу поинг-понг в виде рендерим на сервере -> рендерим на клиенте -> рендерим на сервере и гибриды :)
Yozi
22.08.2016 13:52Замечание:
В статье:Несмотря на то, что Baobab не предлагает иммутабельных структур данных (только персистентные)
Тогда как на деле:
Baobab is a JavaScript persistent and immutable (at least by default) data tree supporting cursors and enabling developers to easily navigate and monitor nested data through events.
https://github.com/Yomguithereal/baobab
Конечно, это опционально, и настраивается. При разработке используется заморозка объектов, чтобы убедиться в корректности кода, а на продакшене её можно отключить, для производительности.
immutable boolean [true]: should the tree's data be immutable? Note that immutability is performed through Object.freeze and should be disabled in production for performance reasons.
de1m
22.08.2016 14:50У меня есть к вопрос к знающим. Я пишу для работы небольшие програмки в nodejs+espresss+jade/pug, типа вот такого.
Вопрос вот в чём, если отбросить период обучения, выиграю ли я применяя вместо espress+jade/pug nodejs+react?Yozi
22.08.2016 15:53Требования точнее бы, чтоб наверняка. Но в целом на React у меня писать получается побыстрее, чем на прежних JQuery. Дебажить легко, разделять на компоненты легко, расширений в принципе уже хватает
Точно могу посоветовать вместо express Koa с Babel: async/await очень упрощает код по сравнению с бесконечными
if(err) fail else success
, тут Вы точно выиграете и без React
NeXTs_od
22.08.2016 15:20Стек технологий у них уже явно устарел на год.
Отсутствие redux. Не использование babel с es 6/7, не использование пререндеринга на сервере…SkvPavel
22.08.2016 19:31Да, на сегодня она уже устарела. Оригинал был опубликован Dec 11, 2015.
AlexanderG
23.08.2016 10:21Что весьма характерно. Чуть больше полугода прошло и всё, что было cutting-edge на тот момет — старьё.
SkvPavel
23.08.2016 11:26Это нормально для технологии, которая развивается сейчас семимильными шагами. Несколько лет назад была такая же ситуация и с Rails.
SkvPavel
22.08.2016 19:29Ребята, не принимайте близко к сердцу перевод нашей статьи, на сегодня она уже устарела. Статья была опубликована Dec 11, 2015 на Medium.
easimonenko
23.08.2016 00:45Может кто объяснит, зачем нужны React и компания, если есть Elm в котором это всё из коробки, плюс статическая типизация, плюс ADT? (Кроме рендеринга на сервере и преемственности в языке.)
lega
23.08.2016 08:21Благодаря технологии Virtual DOM изменения страницы происходят очень быстро. Но стоит заметить, что в последнее время мы часто встречаем статьи ...
React хоть и быстр, все равно он медленнее конкурентов* в 2 (а то и больше раз), возможно вам этого хватит, но там где реакт начнет тромозить, другие будут нормально работать.
alexeyernest
23.08.2016 12:40-1Да, действительно устарела, redux для хранения единого состояние системы, причем это хорошо не только, чтобы легко воспроизвести баг, но можно и отмотать назад-вперед, такая машина времени.
ES6/ES7 позволяет писать компоненты реакта более локанично и правильно, но например jshint не поддерживает еще ES7, так как спецификация не готова, но нормально задавать handler'ы компонентов через => функцию можно только в ES7, так как она автомато биндит this, а в ES6 приходится вручную биндить внутренние обработчики в конструкторе, кроме того в ES6 нельзя внутри класса объявить статическое свойства, например propTypes (приходится их писать после объявления самого класса).
Короче развелся целый зоопарк технологий, но вроде как самое нормально сочетание сейчас react/redux/es6/es7 и jest для тестирования, после всяких ангуляров, что первого, что второго — это просто чистое счастье.
Что касается виртуального дома, то эта идея действительно крута и правильна, redux как последний кирпичик все расставил на места, не менялась ссылка на объект, значит не изменилось состояние, значит рендерить не надо, потому что состояние должно быть immutable. Виртуальный дом это очень хорошо, получше всяких ангуляров (я на ангулярах 3 года просидел). virtual dom решает почти все проблемы, если не решает то надо писать на нативном JS, но надо знать что делать, код должен быть оптимизированным. Кстати очень хорошая книга High Performance JS, без ее прочтения я бы даже не допустил человека к написанию кода на JS.nuit
23.08.2016 12:50>не менялась ссылка на объект, значит не изменилось состояние, значит рендерить не надо, потому что состояние должно быть immutable
время модификации данных раньше чем время последнего обновления компоненты, значит рендерить не надо и состояние может быть mutable.Yozi
29.08.2016 12:52Пример, пожалуйста. Я правильно понял, Вы предлагаете к каждому объекту прикреплять timestamp? Так:
const user = { name: "John" age: 23, contacts: { phone: 123456 email: foo@bar.com, socials: { vk: "vk.com/john", fb: "fb.com/john", _timestamp: 12345676 }, _timestamp: 12345677 } _timestamp: 12345678 }
nuit
29.08.2016 13:08Да, что-то вроде этого. В дискорде так пару узких мест разруливают https://news.ycombinator.com/item?id=11765477, так делали ещё до того как оптимизации с помощью immutable структур стали популярны. Просто иногда узким местом становятся immutable структуры, и для таких случаев можно просто добавить монотонно увеличивающиеся часики.
Yozi
29.08.2016 13:36Ага, ясно, тогда получается это удобство vs производительность. Я пока не столкнулся с необходимостью отказываться от иммутабельности, и больше склоняюсь к мысли, что не следует жонглировать оптимизациями раньше времени, хотя, конечно, когда прижмёт как Discord, придётся локально вырубать иммутабельность в узких местах, благо это не сложно
alexeyernest
23.08.2016 12:46P.S. Лучше использовать сразу es7 для классов компонентов, чтобы подружить sublime и linter с es7 надо поставить eslint + eslint-babel-plugin вместо jshint. Вот хорошая статья как это сделать https://medium.com/@dan_abramov/lint-like-it-s-2015-6987d44c5b48#.7fltq7vnz
G-M-A-X
25.08.2016 11:36+1Сайты, которым нужна индексация, не являются теми типами сайтов, для которых оправдано использовать ReactJS.
Это лишь дань моде.
Замечание относительно изоморфности.
То Вы говорите, что круто, что ничего не генерится на сервере, то вдруг круто, что все генерится еще и на сервере.
Как бы можно это понять, если все на том самом коде. Но в статье 2 разные языка.
Да и вряд ли реакт быстро сгенерирует страничку на сервере. Хотя c кешированием ОК.
Только странно, что сайт должен оживать, а не живой сразу.
Зачем 2 раза гонять те же самые данные? :) Сначала в HTML, потом в JSON (JS).lega
25.08.2016 20:11Сайты, которым нужна индексация, не являются теми типами сайтов, для которых оправдано использовать ReactJS.
А для каких типов сайтов оправдано использовать ReactJS?G-M-A-X
26.08.2016 09:04+1Мне кажется, для сайтов, которые по сути являются веб-интерфейсом, кабинетом по управлению чем-то.
i360u
Web Components — не хватало дробления на "изолированные кусочки", серьезно? Web Components — это и есть изолированные кусочки, и это самое крутое, что сейчас есть для декомпозиции интерфейса. И все, кто реализует поддержку веб-компонентов в своей инфраструктуре (тот же Angular например) — делают абсолютно правильно. Про скорость virtual DOM читать уже смешно, учитывая что обычный DOM никуда не делся, а отделение обсерверов от непосредственной работы с DOM и оптимизации вызова рендер-методов компонентов — это совсем не исключительно Реактовская "магия" а просто принцип хорошей архитектуры в любой экосистеме. Зачем вообще писать про изоморфность, когда "рельсовое" API на сервере? Потому, что "Изоморфность" — просто умное слово? Какую задачу оно может решить эффективно? На практике там сплошная боль. Надоели уже эти дифирамбы Реакту, которые исключительно дань моде, и не содержат никакой ценной информации в инженерном плане.
antipetrov
Airbnb топит за «изоморфность» в том же смысле что автор — один js-код рендерит html на сервере (в ноде) и на клиенте в зависимости от контектста вызова. Для поисковых crawler-ов это, как будто, хорошо — индексируют ровно то что видит пользователь и не нужно дублировать логику рендера на сервере. Почему вам не нравится?
i360u
Я прекрасно знаю что это такое, как и то, что в реальных проектах использовать реактовский рендер на сервере — ужасно медленно и в целом бессмысленно если у вас SPA: времени больше тратиться больше на запрос к серверу чем на загрузку бандла (это если цель — сделать пререндер а потом его "оживить"). В том то и дело, что необходимо дублировать логику на сервере (иначе кто и как там страницу отрендерит?), но продублировать ее полностью идентично клиентскому приложению у вас не получится, там огромное количество нюансов (роутинг, пользовательские данные и т. д.). Добавьте сюда отдельную песню с кэшированием (при работе с фильтрами и сортировками каталогов, к примеру) и получите разумные сомнения в целесообразности такого подхода и преимуществ его перед другими. Но допустим, я просто не встречал нормальной реализации изоморфности (хотя я считаю, что это лучше делает тот-же Meteor но там эта история больше про данные), может мне кто-то расскажет о своем успешном кейсе, как-то за рамками булшит-маркетинга? Чтобы было понятно какая задача решалась и насколько успешно решилась в итоге? Да еще и с Реактом? Что там на эту тему есть у Airbnb? С удовольствием такое почитаю.
nuit
Есть «успешный» кейс, в котором пользователи активно перезагружают страничку на тормозных коннектах. Когда страничка с сервера уже отдана, жаваскрипт её ещё не оживил, пользователи тыкают-тыкают, думают что ничего не работает и начинают рефрешить.
i360u
Вот я и говорю, сейчас модно говорить с умным видом "изоморфность", или там "персистентность", а в реале мало кто понимает зачем это все. Обидно, что стек технологий потом выбирается в соответствии с этой модой, и потом ты вдруг слышишь от реактовских фанбоев про то, что веб-компоненты это отстой или умение эффективно работать с DOM это не тру, есть ведь волшебный Virtual DOM! На этом фоне я постепенно становлюсь хейтером Реакта, хотя изначально работать мне с ним, в целом, нравилось.
pragmader
Обычно, когда люди говорят что-то подобное и знакомятся с Catberry.js им он нравится. Может и вам стоит попробовать.
Серверный рендеринг эффективнее чем у React, к тому же он работает прогрессивно на стримах.
Успешно используется на https://beta.flamp.ru/ по причине индексации и быстрой инициализации приложения.
Если вас в целом интересует вопрос зачем изоморфность, то я как-то делал доклад, который можно посмотреть тут.
Еще был более общий доклад на CodeFest.
i360u
https://twitter.com/TLobinger/status/710544845153107968/ — улыбнуло =) Вот как после такого относиться к этому фреймворку серьезно?
По пунктам:
Похоже ребята не очень хорошо понимают о чем говорят.
Хотите индексации? Делайте так: <my-component>Hello!</my-component> Хотите логики? Делайте так: <my-component component-data='{«myAttribute»: «Hello!»}'></my-component> Внутри — ванильный яваскрипт. Прогрессивная загрузка — не проблема, html-импорты дают мне полную гибкость в этом вопросе. В итоге мне нужен только внятный API для работы с данными. И тут я предпочту решение на чем-то более производительном, типа Go.
Признаюсь, доклад пока не смотрел, возможно ответ там есть.
pragmader
При всем уважении похоже, что вы не понимаете о чем говорите:
Слайд вырван из контекста и если вы сами не попробовали хотя бы посмотреть на готовый проект (1, 2, 3) то как вы можете понять что там написано?
Хочется сказать вам что-то вроде этого:
Прогрессивный рендеринг – вы видимо раньше не встречали такой термин и доклад не посмотрели по моей ссылке, а зря. Там говорится, что HTML с сервера грузится по мере готовности как только данные пришли с API и отрендерили часть страницы.
i360u
Аналогичный совет про рисеч в области веб-компонентов, у меня крепнет уверенность, что вы очень поверхностно с ними знакомы.
pragmader
Ну раз вы так говорите, пойду удалю репозиторий с Github, ведь никому не нужно и никто не использует (сарказм).
Я все же советую хотя бы ознакомиться с примерами и посмотреть доклад, чтобы все-таки представлять вообще о чем вы говорите, а то странный диалог получается.
А рисеч компонентов я как раз-таки провел — так появился Catberry 4.0. До февраля 2015 совсем другой подход был во фреймворке.
i360u
Я никоим образом не хотел вас задеть, респект вам и уважуха за ваш фреймворк. Милый котик, позитив, все дела. Однако, вы хорошо должны понимать, что фреймворков и подходов при использовании каждого из них сейчас довольно много. И далеко не каждый из них оправдает время, потраченное на изучение. Все во многом зависит от конкретных задач, потребностей, технических и бизнес-предпосылок. Веб-компоненты позволяют, лично мне, довольно комфортно заниматься разработкой онлайн-редакторов реального времени (для всяких замороченых, сложных и скучных корпоративных таблиц и форм). На текущий момент, именно они дают мне лучший экспириенс, хотя до этого я имел дело далеко не только с ними. И когда мне что-то советуют, я стараюсь получить детальное представление о том, что я получу и чем я за это заплачу. Открывая вашу ссылку я вижу следующее прямо в центре лендинга (уж не знаю насколько это вырвано из контекста): Cat-Components – similar to web-components but organized as directories, can be rendered on the server and published/installed as NPM packages. И это меня сильно смущает, потому как все абсолютно тоже самое можно делать и с веб-компонентами (и я это делаю каждый день), и это "but" — выглядит очень странно. Еще мне предлагают посмотреть как легко я могу все установить через npm — замечательно. Но зачем? Еще мне рассказывают про прогрессивный рендеринг — и я сравниваю это с возможность абсолютно свободно работать с API и слоем данных из любого своего модуля и снова не понимаю зачем мне для этого рисерчить другое решение. Если во главу угла поставить изоморфность, то чем ваш фреймворк лучше того-же Meteor? Мне это сходу не понятно, а вы говорите весьма сомнительные, с моей точки зрения, конечно, вещи. Может стоит все это как-то более вкусно подать? Так, чтобы зацепить чем-то осязаемо-рациональным (производительностью там, к примеру)?
Alukos
1) Подскажите, пожалуйста, а если хочу и индексацию, и логику, то как?
2) push из http2 используете?
indestructable
Веб компоненты не исключают реакта и можно их сочетать.
Преимущество виртуального DOM в том, что это встроено в библиотеку, а не реализуется каждый раз, как, например, при оптимизации директив в Ангуляре.
Немного преимуществ Реакта:
i360u
Веб-компоненты просто делают Реакт и всю его экосистему избыточной сущностью. Незачем сочетать, ибо делают одно и то-же по сути.
indestructable
Ну частично да, согласен. Но, например, умный патчинг DOM на чистых веб-компонентах не реализуешь.
i360u
Это вы про обращение через react-id? А что тут умного, поясните? Во первых, подобное можно легко реализовать в компонентах, а во вторых, я предпочитаю прямые подписки через коллекции ссылок на DOM-элементы, которые формируются при рендере моделей компонентов (да, это нечто вроде virtual-DOM). Это позволяет делать прямой вызов методов вложенных компонентов и, как и в Реакте, делать рендер данных без лишних перерисовок.
nuit
>Во первых, подобное можно легко реализовать в компонентах, а во вторых, я предпочитаю прямые подписки через коллекции ссылок на DOM-элементы, которые формируются при рендере моделей компонентов (да, это нечто вроде virtual-DOM)
Любопытно будет взглянуть на то как вы будете обновлять DOM прямыми вызовами при изменении формы поддерева, так чтоб не терять внутреннее состояние (css анимации итп).
i360u
А в чем именно вы видите тут проблему? Поясните пожалуйста подробнее.
nuit
В том что правильно изменять форму поддерева(двигать, добавлять, удалять элементы) без абстракций вроде виртуального дома достаточно сложно.
i360u
И где я предлагал это делать без абстракций? Вот прямо в той цитате из моего коммента, которую вы привели, именно об этих абстракциях и написано. И эти абстракции — довольно просты, ибо это просто коллекции элементов. Лично я — всегда так делаю, и не вижу особых проблем. Мне точно не нужен для этого Реакт и весь зоопарк его зависимостей.
nuit
>И эти абстракции — довольно просты
Можно взглянуть на пример?
Riim
Посмотрите Rionite, это как раз попытка прилепить реактивные биндинги к кастомным элементам. Есть ещё набор компонентов на этой библиотечке: jsfiddle. Всё это пока малость сырое конечно, но ощущения от использования на реальном проекте реально здоровские, даже после реакта. Плюс всё это ещё и довольно шустрое получается: js-repaint-perfs .
nuit
>Плюс всё это ещё и довольно шустрое получается
Гораздо тормознее виртуального дома http://localvoid.github.io/kivi-dbmonster/
Riim
Но быстрее его реализации в реакте)) И мне одному кажется, что этот тест очень удобен для виртуального дома? В реальном интерфейсе обычно несколько тысяч активных биндингов и при каком-то действии пользователя меняются лишь несколько (1-10) из них. Именно меняются, не создаются/удаляются. Либо пользователь переключает раздел и тогда как-раз происходит массовое создание/удаление биндингов и опять же обновление лишь нескольких из них. Думаю если приблизить этот тест к реальности, то библиотеки с "точечными" биндингами (Rionite, rivets, ...) оставят далеко позади даже самые быстрые реализации виртуального дома.
nuit
>И мне одному кажется, что этот тест очень удобен для виртуального дома?
Да, одному, можете выставить кол-во мутаций на 1% и будет worst case для виртуального дома. Этот бенчмарк плохой по совершенно другим причинам, например тк там упоротый счётчик фпс.
>Думаю если приблизить этот тест к реальности, то библиотеки с «точечными» биндингами (Rionite, rivets, ...) оставят далеко позади даже самые быстрые реализации виртуального дома.
Берём какой-нибудь mobx+react и получаем тысячи implicit granular per-value observer'ов. Просто те у кого опыта побольше, те знают как такие решения сосут на всяких дататэйблах, ну там где обычно узкие места в реальных приложениях ;)
Riim
В любом случае, скорости более чем хватает для реального проекта. Текущий проект, кстати, начинал на связке cellx+React, и уже на первом разделе были два места где немного притормаживало по вине реакта, позже начал с нуля на Rionite, доделываю второй раздел, пока всё летает)). Большой дататейбл тоже есть, как раз в том самом первом разделе. Проблема виртуального дома в том, что он вынужден считать дифф самого компонента и всех дочерних компонентов, и если рядом с обновляемым кусочком лежит какой-нибудь селект с несколькими тысячами опций (как-раз первое место с тормозами, селект был с городами) или что-нибудь в этом духе, то ему прийдётся всё это просмотреть. Тут, конечно, есть решения, но все они костыли по сути.
На счёт mobx не знаю, но на этом бенчмарке он в 10 раз медленнее cellx-а.
nuit
>Проблема виртуального дома в том, что он вынужден считать дифф самого компонента и всех дочерних компонентов
Ничего он не должен, если входящие данные не изменились, то можно ничего не диффать. А как уже узнавать о том что изменились — это уже выбирать в зависимости от проблемы. Если хочется вешать кучу обсерверов, то добавляем dirty флаг и обновляем только когда компонента грязная, хочется проверять входящие данные от парента, то либо используем immutable данные, либо используем внутренние часики и проверяем по дате модификации. Вобщем это не проблема виртуального дома, и всё очень сильно зависит от того что нужно реализовать.
Riim
вот я и говорю — костыли :), с точечными биндингами проблемы нет в принципе.
nuit
>с точечными биндингами проблемы нет в принципе.
кроме тысяч ненужных биндингов, которые вешаются как раз в тот момент когда происходят самые тормозные операции — добавление элементов.
ну а в случае со всякими фильтрациями, так вообще нужно вешать обсерверы на все фильтруемые значения, даже если отображается 10 из 10000.
Riim
добавить обработчик к эвентэмиттеру по сравнению с самим добавлением элемента — моментальная операция, не вижу проблемы. К тому же Rionite добавляет разметку компонентов фрагментами (один компонент — один фрагмент), а не по одному элементу и эти фрагменты, на момент добавления, уже обработаны, то есть куски типа
"{name}"
в них уже заменены на значения.так мы обсуждаем биндинги или работу РП? Если прикрутить РП к виртуальному дому, то будет тоже самое и единственный способ уйти от этого — отказаться от РП, на что я вообще никак не согласен и я не знаю людей хорошо понявших эту парадигму и согласных писать что-то серьёзное без неё. Вы, конечно, как хотите))
nuit
>и я не знаю людей хорошо понявших эту парадигму и согласных писать что-то серьёзное без неё
Желаю поскорее наконец вам встретить нормальных инженеров, которые умеют решать проблемы наиболее подходящими способами, а не пытаются решить все проблемы с помощью РП, тк они «хорошо» её поняли.
Riim
Вы уже второй раз скатываетесь на какие-то странные намёки, первый:
К чему это? Недостаток аргументов или раздутое самомнение? Я вроде пытаюсь нормально говорить и приводить нормальные аргументы. Может чего и не понимаю, так объясните без выпендрёжа.
Ну и вообще, сравнение VDOM-vs-RP — довольно странное, как Слон-vs-Табурет, это не конкурирующие вещи, но они могут неплохо дополнять друг друга. Лучше запилите коннектор вроде этого: cellx-react, а я с удовольствием поставлю ссылочку на него у себя. Заодно сможете попробовать непонятную вам парадигму не отрываясь от своих наработок, плюс убедитесь, что АПИ вашей реализации позволяет полноценно делать такие коннекторы, в ранних версиях реакта у меня, помнится, были какие-то проблемы с этим.
nuit
>Вы уже второй раз скатываетесь на какие-то странные намёки
>Если прикрутить РП к виртуальному дому, то будет тоже самое и единственный способ уйти от этого — отказаться от РП, на что я вообще никак не согласен и я не знаю людей хорошо понявших эту парадигму и согласных писать что-то серьёзное без неё."
От того что у меня будет одна подписка, которая будет кидаться в случае изменения любого фильтруемого значения, моя реализация не будет менее реактивной, но видимо я недостаточно «хорошо» что-то там понял.
>Ну и вообще, сравнение VDOM-vs-RP — довольно странное
Это вы там каким-то боком начали говорить про RP, видимо «хорошо» поняли, я лишь указывал на то что ваш датабайндинг подразумевает подписки на все значения, дата байндинг через виртуальный дом этого не требует и можно решать задачи наиболее подходящим способом.
Riim
видимо так, РП — это в первую очередь про потоки данных, а не про обновление разметки, как многие сейчас считают благодаря реакту.
так почему это плохо? Получается медленней при первичном построении DOM (использование фрагментов вполне может нивелировать эту разницу), но быстрее при точечном обновлении, в некоторых случаях в этом месте VDOM настолько медленный, что требует уже костылей. Вроде это всё или что-то забыл?
nuit
>видимо так, РП — это в первую очередь про потоки данных
Ну так у меня на входе поток данных, который посылает сообщения о том что табличка изменилась. Всё вполне реактивно.
>а не про обновление разметки, как многие сейчас считают благодаря реакту.
Это что-то новое. Видимо это были лично ваши заблуждения и вы теперь думаете что у остальных были такие же проблемы.
>так почему это плохо? Получается медленней при первичном построении DOM (использование фрагментов вполне может нивелировать эту разницу)
Получается медленей так где итак всё тормозит, зато пытаемся ускорить кэйсы на которых всё спокойно отрабатывает втечении одного кадра.
>но быстрее при точечном обновлении, в некоторых случаях в этом месте VDOM настолько медленный, что требует уже костылей.
пруфов видимо не будет, и лишь только ваш личный опыт с реактом, который вы переносите на все остальные вдом библиотеки :)
Riim
ok, раз вас всё устраивает, значит так и оставьте))
ну, можно относиться иначе: на 1-2% медленнее в тех местах, где и так тормозит и не плохо бы воткнуть какое-нибудь сообщение о загрузке.
Эх, если бы это всегда было так))
не только мой личный, это опыт большой команды в компании продуктами которой вы с большой вероятностью пользуетесь/овались. Думаю там очень не глупые люди сидят, да и я со своими 10+ годами продуктовой разработки уже чего-то стою. Плюс у меня есть свои реализации как VDOM, так и патчера без V составляющей. Мне тоже вся эта идея изначально очень нравилась и нахватавшись всяких кейсов вроде описанного выше я попытался исправить их в своём варианте, поняв, что это нормально сделать не получается, начал пилить morph-element, идея была такой: сравнение строк — идеально быстрая операция, причём не зависит от длинны строки, даже при очень длинных и почти совсем одинаковых строках сравнение происходит сверхбыстро где бы не находилось различие. Дальше вместо подстановки разметки дочернего компонента за счёт вызова его как функции, родитель должен был генерировать кастомные элементы разметка которых разворачивалась уже в attachedCallback (ныне connectedCallback) (он синхронный, полифилится на MutationObserver не синхронно, но это я тоже порешал). Что это давало? Получалось, что каждый компонент генерирует только свой собственный контент, но не контент потомков, то есть при первой генерации можно просто запомнить эту строку и при последующей сравнивать не два огромных объекта со всем контентом потомков, а две относительно короткие строки, что в 100500 раз быстрее. Ну и естественно не перегенерировать контент потомков при равности контента родителя с предыдущей вариантом. То есть лишняя перегенерация всё же происходила (родитель), но уходила не на всю глубину, а максимум на один уровень. Такой подход решал некоторые проблемы VDOM, но не все и в целом патчинг с реального DOM (без виртуальной составляющей) всё же получается медленней, как бы я не пытался его оптимизировать и что бы там не говорил разработчик оригинального morphdom.
В общем, пока я в тупике, а все другие варианты, что я посмотрел, предлагают костыли вроде:
А хочется то нормальный фреймворк, который позволяет просто спокойно кодить не расставляя эти самые костыли по всему коду.
Сейчас я для себя пришёл к выводу, что VDOM — тупиковая идея, но я не исключаю, что ошибаюсь и возможно ещё вернусь к ней))
Пока вернулся к точечным биндингам, тут тоже не всё в идеале, но всё же явно лучше.
nuit
Вообще советую пообщаться с Эваном, автором vue, ну чтоб он рассказал почему он выкинул примерно то что вы сейчас пытаетесь сделать в Rionite, и заменил на вдом во второй версии. Его легко можно найти в тематических чатиках.
Riim
Ну, я выше рассказал, почему я выкинул VDOM))
nuit
>Эх, если бы это всегда было так))
http://localvoid.github.io/kivi-dbmonster/?m=0.001&n=5000
10,000 рядов (~20,000 компонент, элементов хз сколько, но дохера, браузеру становится плохо)
0.1% мутаций
У меня на десктопе апдэйт с помощью вдома происходит за ~1.6мс. При этом всё невероятно тормозит из-за того что кол-во элементов на странице очень большое (лэйаут, лт апдэйт, пэинт >150мс).
Жду пруфов о том какой невероятной скорости вы добились в вашем фрэймворке на «create» и «clear» кэйсах в этом бэнчмарке: https://github.com/krausest/js-framework-benchmark
Должно быть не сложно продемонстрировать о том как всё замечательно, там всего-лишь будет два биндинга на каждый ряд, идеальные условия для вашего фрэймворка.
https://cdn.rawgit.com/krausest/js-framework-benchmark/4e47158a10e52122bf4244724cf99b6de4ef7ac1/webdriver-java/table.html
Riim
прежде чем предлагать кому-то повторить этот тест, доведите его до состояния, когда туда нормально можно добавить свою либу. Тайпскрипт (я не против, но всё же стандарт для таких тестов — VanillaJS), сборка, деплой, после сборки зачем-то требует SDK, дальше я не осилил. Сейчас это тест для одной библиотеки — вашей, и что вы там сравниваете непонятно.
кажется вы не слишком внимательно читаете, я как раз без проблем признаю, что в создании/удалении мой подход будет медленнее, а вот у вас с признанием недостатков похоже проблемы. Какой смысл использовать вашу реализацию, если вы не готовы признавать её очевидные подводные камни. Сколько их там ещё? Для реакта они хотя бы нормально описаны в Интернете, прощупаны многими самостоятельно и есть отработанные костыли для их обхода.
Давайте по пунктам:
Если есть нормальные контраргументы, я вас слушаю, но не надо кидаться в меня очередными тестами или мнением очередного "авторитета".
Если вы нашли хороший способ избавляться от лишнего рендера/диффа, расскажите как, то что вы предложили выше — костыли похлеще тех, что использую я, видимо специфичны для вашей реализации. По крайней мере такой жести:
я ещё ни у кого не видел.
nuit
>прежде чем предлагать кому-то повторить этот тест, доведите его до состояния, когда туда нормально можно добавить свою либу. Тайпскрипт (я не против, но всё же стандарт для таких тестов — VanillaJS), сборка, деплой, после сборки зачем-то требует SDK, дальше я не осилил. Сейчас это тест для одной библиотеки — вашей, и что вы там сравниваете непонятно.
Вы про этот? https://github.com/krausest/js-framework-benchmark
Это не мой бэнчмарк, там полным полно библиотек, закоммиченных различными людьми. Никаких проблем у меня не возникло с реализацией, где-то за 15 минут спортировал, глядя на реализацию реакта и сделал пулл риквест, автор потом прогнал и обновил табличку. Но видимо для разработчиков с 10+ летним опытом это всё слишком тяжело.
Или вы про дбмон? Не совсем понял, у вас вроде есть реализация дбмона, я тут просто выкрутил параметры N=5000, mutations=0.001. Худший кэйс для вдома, и ведь как-то справляется.
>кажется вы не слишком внимательно читаете, я как раз без проблем признаю, что в создании/удалении мой подход будет медленнее, а вот у вас с признанием недостатков похоже проблемы.
Возможно вас удивит насколько ваша реализация будет медленее :) Ато вы там про какие-то 1-2% фантазируете. Я конечно понимаю что наверно возможно сделать такую библиотечку с таким маленьким оверхэдом, просто я пока не встречал таких.
>есть ситуации в которых vdom начинает рендерить/диффать слишком много лишнего, что может приводить к тормозам и без костылей в коде проекта не решается.
пруф будет что это может приводить к тормозам?
>я ещё ни у кого не видел.
Использовалось ещё до того как immutable в реакте стало модным, так же используется в куче реальных проектов, например в дискорде. То что вы этого не видели, с вашим 10+летним опытом, отлично показывает что у вас за опыт.
А dirty флаг — возможно для вас это будет новостью, но веб браузеры именно так и работают. Да и большинство нативных ui библиотек последние лет 20 так же работали.
Riim
Общаться с вами стало просто неприятно, удачи вам.
nuit
Не понимаю зачем нужно было демонстрировать сколько вы там лет занимались продуктовой разработкой. В нашем маленьком сообществе новичков, мы таких обычно называем «программирую с 5 лет» :) Я вот веб-разработкой занимаюсь всего 2 года, наверное в этом вся проблема, нужно ещё 8 лет и я полюблю kvo биндинги :)
lega
Я решил погонять бенчмарки
С этим явно что-то не то, при 100% мутаций обновление идет раз в 2 сек, а это 0.5 FPS, но стат-бар в правой панели отображает 60 FPS, врет.Этот бенчмарк не удалось погонять, он падает при сборке «npm run build», попробую когда автор починит (или когда у меня появится время пофиксить это).
nuit
>С этим явно что-то не то, при 100% мутаций обновление идет раз в 2 сек, а это 0.5 FPS, но стат-бар в правой панели отображает 60 FPS, врет.
FPS счётчик использует EMA, и стартует с 60, так что если кадр рендерится раз в 2 сек, то он будет очень долго добираться до нужного значения.
Можно открыть dev tools timeline и там всё будет предельно ясно.
lega
Надо будет свои расчет вкуртить для проверки.
nuit
>Кстати на оригинальном dbmon, Angular 2 стартует с 100 FPS и за пару минут снижается до 30 FPS или ниже, это тоже с расчетом связано или баги в ангуляре...?
там вообще не fps, а то с какой частотой вызывается функция ping(), поэтому реализации которые успевают втечении одного кадра несколько раз проапдэйтить дом(грязный дом значительно быстрее обновляется) и вызвать функцию ping() начинают резко показывать >200fps :)
Вот angular2 реализация со счётчиком, который по rAFу считает: https://cdn.rawgit.com/lhorie/mithril.js/rewrite/examples/dbmonster/angular/index.html
nuit
>Этот бенчмарк не удалось погонять, он падает при сборке «npm run build», попробую когда автор починит (или когда у меня появится время пофиксить это).
Смотрю вы там пытаетесь всё собрать :) Я так и не смог там всё собрать и особенно запустить большой раннер, глянул исходники, понял что там пробиты маковские пути и забил на эту идею. Я просто тупо сделал так чтоб моя реализация локально собиралась внутри директории и предоставляла нужные npm run скрипты, тк глобальный npm build просто бегает по всем директориям и исполняет локальные установки, потестировал что всё правильно работает и отправил пулл риквест, всё остальное он там сам сделает :)
indestructable
Нет, причем тут react-id. Я об "умном" обновлении ДОМ дерева, том, что, собственно, и делает Реакт реактивным.
В Реакте дерево ДОМ элементов является функцией, или проекцией, состояния приложения. И при изменении этого состояния дерево элементов перестраивается не полностью, а частично, производя минимальные изменения, необходимые для приведения дерева в нужное состояние.
Если не использовать Реакт (и другие virtual DOM библиотеки), то придется либо перестраивать все дерево полностью — и забыть о производительности, либо, как в Ангуляре, оптимизировать перестроение дерева для каждой отдельной директивы.