"Если хочешь идти быстро — иди один. Если хочешь идти далеко — идите вместе." (с)


С этой лирической строки в данной статье я буду рассуждать о том, как правильно организовать код в вашем приложении, чтобы оно могло расти в высоту и в ширь. Если вы хотите, чтобы продукт вашей мозговой активности был мощнее, чем у ваших конкурентов, то вам неизбежно придется приглашать новых программистов в вашу команду. А если не положить вектор масштабируемости, то порывы энтузиазма через год превратятся в лапшу-код и командная работа превратит каждого сотрудника от злости в маленького сатану.


Так вот… Для того, чтобы ваши бойцы чувствовали себя комфортно вместе в одном проекте, надо чтобы они не мешали друг другу и писали свои буквы в разных не пересекающихся участках кода. Для этого им нужно будет писать "Самостоятельные" компоненты.


"Самостоятельные" — это такие компоненты, которые сами управляют своим поведением, ориентируясь на события из внешней среды. При знании о том, как работает ваше приложение и какие события в нем протекают, можно легко писать такие "самостоятельные" компоненты, не затрагивая старые и чужие участки кода.


"Несамостоятельные" — компоненты, которые ничего не знают о внешней среде, но у них есть очень развернутое api. Этому компоненту нужно объяснить, как себя вести в вашем приложении. Такие компоненты, в отличие от "самостоятельных" пишутся ради многоразового использования в вашем или публичном проектах, как например открытые библиотеки на github и др.


Как определить какие компоненты нужны в вашем приложении? Очень просто. Если компонент применим только к одной задаче и не многоразовый, то его нужно писать так, чтобы он был "самостоятельным".
Вот например, рассмотрим компонент олицетворяющий поля ввода сообщения в ленте чата. Скорее всего такое поле ввода в вашем приложении вы будете использовать только по прямому назначению и не будете его использовать, скажем, в форме ввода никнейма или пароля при авторизации, ибо там у компонентов будет своя специфика.


Не будем тянуть кота за то, что не стоит оттягивать и разберем конкретный пример. Пускай это будет имитация чатика.


Представим, что у вас два программиста в команде. Петька и Толик. И у них есть ядро масштабируемого приложения. Простое, как два пальца у двупалого человека. В ядре есть сетевой транспорт, хранилище ленты сообщений в виде массива(в этом примере не будем выделять его в отдельный файл с методами) и event emitter, который в этом случае является залогом масштабируемости.
В качестве event emitter в этом примере я взял Backbone.Events, хотя этим и ограничимся в использовании Backbone, чтобы показать пример как можно проще.


<html>
<head>
    <script src="http://underscorejs.org/underscore-min.js">
    <script src="http://backbonejs.org/backbone-min.js">
    <script src="connection.js"/>
    <script src="app.js"/>
</head>
<body>
    <div id="header" style="padding:10px; border:1px solid #ddd"></div>
    <div id="container" style="margin-top:20px;"></div>

    <script>
        var app = new App();
        app.init();
        app.on('new_message', function(text){
            console.info('new message', text);
            console.info('messages length', app.messages.length);
        });
    </script>
</body>
</html>

//app.js
var App = function(){
    var app = _.extend({
            init: function(){
                this.connection.connect();
            }
        }, Backbone.Events);
    app.connection = new Connection(),
    app.messages = [];

    app.connection.on('connected', function(){
        console.info('App connected!');
    });

    app.connection.on('incoming_message', function(text){
        app.messages.push(text);
        app.trigger('new_message', text)
    });

    return app;
}

//connection.js
var Connection = function(){
    return _.extend({
        connect: function(){
            /*просто имитируем то, что наш сетевой транспорт принимает сообщения от сервера и отдает какие то сигналы каждую секунду во внешнюю среду*/
            var i=0;
            setInterval(_.bind(function(){
                i++;
                var text = 'message_' + i;
                this.trigger('incoming_message', text);
            },this),1000);
            this.trigger('connected');
        },
    }, Backbone.Events);
}

Ну вот, у нас есть приложение, в котором пока ни одной вьюхи и работу которого можно протестировать через консоль браузера. Кстати, если из вашего приложения удалить все вспомогательные компоненты и вьюхи, и оно сможет работать через консоль, то это очень хорошо. Значит у вас достигнута в какой-то мере слабая связанность между компонентами и код можно покрыть автоматизированными тестами. Погнали дальше.


Теперь сведущий в стратегических планах человек ставит задачу Петьке и Толику, мол, надо, чтобы приложение показывало ленту сообщений, а в шапке был счетчик всех сообщений в из ленты. У вас мог возникнуть вопрос… кому вообще нужен этот, блин, счетчик сообщений в шапке в реальной жизни? Это просто для примера.


Ок, думают Петька и Толик, ок. Они решают одновременно написать два разных компонента для приложения.


Петька взял на себя задачу по ленте сообщений


Но он не слышал о том, как программировать масштабируемое приложение и начал писать код:


//list-view.js - "несамостоятельный" компонент
var ListView = function(){
    var el = document.createElement('div');
    return {
        el: el,
        addMessage: function(text){
            var row = document.createElement('div');
            row.innerHTML = 'message: ' + text;
            el.appendChild(row);
        }
    }
}

//app.js изменение кода
var App = function(){
    var app = _.extend({
            init: function(){
                connection.connect();
            }
        }, Backbone.Events);
    app.connection = new Connection(),
    app.messages = [];

    //добавил код
    app.listView = ListView();
    document.getElementById('container').appendChild(app.listView.el);
    //

    app.connection.on('connected', function(){
        console.info('App connected!');
    });

    app.connection.on('incoming_message', function(text){
        app.messages.push(text);
        app.trigger('new_message', text);
        app.listView.addMessage(text); //добавил код
    });

    return app;
}

Петя создал компонент, которым приходится управлять посредством методов на более высоком уровне и, как следствие, помимо простого объявления компонента, пришлось копаться в коде app.js и добавить строки в обработчик incoming_message. Теперь вы не сможете просто закомментировать строки "app.listView = .." и "...appendChild(app.listView.el)" так, чтобы ваше приложение не сломалось. Ибо app.listView.addMessage(text); выдаст Exception. Приложение начало обрастать связанностью. Ядро начало зависеть от view.


Посмотрим, как справился Толик с задачей по счетчику сообщений в шапке


Он знает, как писать код так, чтобы не мешать другим:


//header-view.js
var HeaderView = function(app) {
    var el = document.createElement('div'),
        span = document.createElement('span'),
        view = {
            el: el,
            setCounter: function(num){
                span.innerHTML = num;
            }
        }

    el.innerHTML = 'Кол-во сообщений: ';
    el.appendChild(span);
    view.setCounter(0);

    app.on('new_message', function(){
        view.setCounter(app.messages.length);
    });

    return view;
}

//app.js изменение кода
...
    app.connection = new Connection(),
    app.messages = [];

    //добавил код
    app.headerView = HeaderView(app);
    document.getElementById('head').appendChild(app.headerView.el)
    //
...

Что сделал Толик за пределами своего компонента — это только объявил компонент в области переменных app, отрендерил и все. Компонент остается также доступным для ручного тестирования через консоль или для модульного тестирования, так как он все же возвращает api.
Зона ответственности за код Толика ограничивается по сути всего одним файлом header-view.js и эти правки легче ревьюить, ведь надо смотреть всего в один файл.


Писать "самостоятельные" компоненты выгодно


Если бы Толик тоже написал несамостоятельный компонент, то в app.js он затронул бы те же куски кода, что и Петя. Сложно мержить, связанность между компонентами увеличивается. На таком маленьком примере может этого не сильно будет заметно, но если у вас суммарно многотысячный код и пишутся большие сложные фичи, то это можно будет хорошо почувствовать.
В процессе написания кода всегда будет выбор, либо управлять компонентом на более высоком уровне иерархии, либо дать компоненту управляться самостоятельно.
Разделяйте и властвуйте господа, пишите для своих приложений "самостоятельные" компоненты.


p.s. Хотя примеры кода в данной статье были написаны на голом JS без использования фреймворков, данная философия слабой связанности справедлива и при их использовании, будь то Backbone или React с хитрыми методологиями изоморфных приложений типа Flux и Redux, или еще каких других фреймворков.
Всегда стремитесь ограничивать зону ответственности в коде ваших программистов, когда они пилят новые фичи. Если вам дали такой гаечный ключ, как React, то им нужно закручивать гайки, а не бить себе им по пальцам.


Команда разработчиков JivoSite.ru желает вам чистого и понятного кода.

Поделиться с друзьями
-->

Комментарии (7)


  1. Nadoedalo
    18.05.2016 14:43
    -7

    >> header-view.js
    >> container.appendChild(el);
    Плохая практика. Тот кто будет вызывать не должен передавать контейнер а должен сделать

    var view = new Header-view();
    $('.main').html(view.$el);
    

    Потому что передача контейнера при инициализации это нестандартная фича. А $el/el есть в backbone всегда.

    Вот так выглядит инициализация модулей в Router:
    newMessage : function(){
                require([
                    'js/modules/NewMessage/NewMessageModel',
                    'js/modules/NewMessage/NewMessageView'
                ], function(NewMessagesModel, NewMessagesView){
                    var model = new NewMessagesModel(),
                        view = new NewMessagesView({model : model});
                    $('.main').html(view.$el);
                }.bind(this));
            }
    

    соотвественно все данные(типа ID модели) записываются из роутера, и он — звено, связующее компоненты между собой(модели-вьюхи). Всё что нужно сделать — после инициализации вьюхи вернуть this.
    А с подходом container.append(this.el) мы рано или поздно перейдём к $('.body, .ololo').append(this.el) что черевато.


    1. Nadoedalo
      18.05.2016 18:41

      Заголовок спойлера
      нда, минус 6 в репу и минус 2 в карму. Очень заслуженно и в тихую, без какой-либо критики.


      1. jivosite
        18.05.2016 19:29

        Есть логика в ваших словах, доработаю статью, чтобы этот момент больше не вызывал вопросов. Спасибо.


      1. RubaXa
        18.05.2016 19:32

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


        Ну и «Толик» всё же не справился с задачей до конца и использует глобальную переменную app:


        app.headerView  = new HeaderView(app, document.getElementById('head'));


        1. jivosite
          18.05.2016 19:40
          +1

          Ну модульный подход типа CommonJS, как require и import, автоматом решают эти задачи. Если использовать сборщик webpack, то вообще красота будет. Однако, спасибо за коммент!


        1. Nadoedalo
          18.05.2016 19:56

          А кто сказал что они глобальные? $ объявлен в require уровнем выше и уровнем ниже(в самой view). Просто не видел смысла приводить полный пример кода ради аргументирования своего мнения по поводу одной строки кода.
          Если нужен полный пример — пожалуйста:

          router.js
          define([
              'jquery',
              'underscore',
              'backbone',
              'require'
          ], function ($, _, Backbone, App, require) {
          "use strict";
          return Backbone.Router.extend({
          	routes: {
          		'': 'notFound',
          		'newMessage' : 'newMessage',
          		'*notFound': 'notFound'
          	},
          	notFound: function (route) {
          		this.navigate('newMessage', {trigger: true});
          	},
          	newMessage : function(){
          		require([
          			'js/modules/NewMessage/NewMessageModel',
          			'js/modules/NewMessage/NewMessageView'
          		], function(NewMessagesModel, NewMessagesView){
          			var model = new NewMessagesModel(),
          				view = new NewMessagesView({model : model});
          			$('.main').html(view.$el);
          		}.bind(this));
          	}
          })
          


  1. Terras
    18.05.2016 17:10
    +5

    В каждой умной книге пишут, что нужно свое решение создать из n-ого числа отдельных приложений, которые замыкают всю логику в себе (по типу ввод/вывод) и не влияют на окружающий код. Тут это показано на примере. Годнота. Разве, что «беженцы» с мегамозга не оценят.