Index.ts – зло и польза

Привет всем! Меня зовут Михаил, я старший Frontend-разработчик в НЛМК, занимаюсь разработкой одной из внутренних информационных систем на React + Typescript.

Расскажу про самый короткий и наименее трудоемкий способ экспорта и импорта модулей, что частенько требуется для построения современных приложений. А именно опишу свой эксперимент с импортом и экспортом без использования файла Index.ts, затем — с его использованием. Для наглядности я создал небольшой проект с Webpack и Typescript в редакторе исходного кода Visual Studio Code (далее по тексту VS Code).

ba850842d0ae5ae767e82970d9c4ef8d.jpeg

Файл webpack.config.js содержит дефолтные настройки, никаких плагинов я не использую. В статье посмотрим на результаты сборки проекта с помощью Webpack в режиме »mode=development», в режиме »mode=production» и затем подведем итоги.

Структура проекта выглядит следующим образом:

8d0e492a2f5ce00f288f25b83104f46d.png

В app.ts происходит подключение модулей mod1, mod2, mod3. А модули mod1, mod2, mod3 внутри себя импортируют типы из папки types.ts.

Само приложение очень простое.

Объявляем две числовые константы. Вызываем функцию суммирования чисел из mod1, затем вызываем функцию из mod2, получаем строку с текстовым описанием проделанной операции и для вывода описания в консоль вызываем функцию из mod3.

Начнем рассмотрение без файлов Index.ts.

Содержимое файлов представлено ниже:

app.ts

8f7c883659fa7ec42b0ed5630818568f.png

mod1:

bdb69fb1d7867310aa99b6df434fa086.png

mod2:

945aa1cb3d5d6a9701dea953375ff41c.png

mod3:

f624da732c242d8fc58e7a46e6149512.png

Типы простейшие используются для демонстрации:

caption_types:

546c7951ac028d900cea9ec2b52cf4ca.png

resultMessage_types:

5b48f4777177fb623295a31e2941c9cd.png

sumElement_types:

4e0db15e1343b81feb44c4255841576a.png

Видим в app.ts три строки импорта до каждого файла, т.е.

  1. необходимо прописывать путь до файла;

  2. проникновение во внутреннюю реализацию, в структуру и именование файлов внутри общей папки parts;

  3. необходимость писать 3 отдельных импорта.

В файлах модулей mod1, mod2, mod3 импорт типов обладает всеми вышеописанными недостатками. Если бы происходил импорт более одного типа, то каждый тип импортировался бы отдельной строкой и напрямую зависел бы от расположения файла с типом внутри папки types. Т.е., если внутри папки types возникнет необходимость перенести файл с экспортируемым типом в другую папку или перенести экспортируемый тип в другой файл, во всех модулях, импортирующих этот тип, придется внести изменения.

Проведем сборку с помощью Webpack в режиме »mode=development», в режиме »mode=production».

  1. mode=development:

    c6a06771d1a885c46faacc4951663535.png

Сборка прошла за 3380 миллисекунд, результирующий файл main.js весит 6.16 килобайт.

  1. mode=production:

c0a70168f7dc8a2c8b0fe72406ae931d.png

Сборка прошла за 3622 миллисекунды, результирующий файл main.js весит 50 байт.

Запомним эти результаты и теперь добавим в проект файлы Index.ts.

Первым делом видим структуру проекта, которая явно увеличилась.

52820b52dc37f2e0de0cbc7f4494cfde.png

Посмотрим, что содержат файлы Index.ts в папках parts и types:

parts/Index.ts:

2268e6b226f25b35fe90fb8d8b7d4cfa.png

types/Index.ts:

d2679dcde36618b34868a7b510e5ec35.png

Благодаря файлам Index.ts в папках с модулями mod1, mod2, mod3 и в папках с типами caption, resultMessage, sumElement не нужно писать что-то типа export * from './mod1/mod1' — видим задвоение /mod1/mod1.

Посмотрим, как изменился наш файл app.ts:

3059680673a02c154a65b50a49880dec.png

В app.ts:

  1. нет необходимости прописывать путь вплоть до файла;

  2. мы ничего не знаем про внутреннюю реализацию и структуру файлов внутри общей папки parts;

  3. одна строка импорта для функций через именованный деструктурированный импорт.

Теперь импорты типов в mod1, mod2, mod3 напрямую не зависят от расположения файла с типом внутри папки types и имени самого файла, экспортирующего данный тип.

Для примера переименуем файл с типом CaptionItem из »caption_types» в »renamed_caption_types». Единственной точкой внесения изменений будет файл Index.ts внутри самой папки caption, содержащей переименованный файл.

adf67940c4278e559332aa09a18faf37.png

Таким образом, файлы индекса служат «шлюзом» для всего импорта, связанного с папкой, в которой они находятся. Это своего рода «оглавление» экспортов папки.

Экспорт через файл Index.ts защищает от экспорта типов или переменных с одинаковыми названиями, но возможно с разным содержимым.

Например, в папке types/caption_types находится тип CaptionItem, который равен number.

41ac8312d80b9dacada104772b6cb7d5.png

Добавим в файл resultMessage_types тип с аналогичным названием CaptionItem, но приравняем его string.

62809557c30040625feb85499ad4ac93.png

После это VS Code выдаст подсказку, что экспортируемый ранее модуль уже экспортирует значение с аналогичным названием.

20aaa39fe0dfdfce0d0ea6583730394f.png

Импорт типа CaptionItem происходит в модуле mod2, удалим импорт и посмотрим какие варианты для импорта предложит VS Code.

1b1d51d7c1744a4c5d4f6f52baf8af58.png

VS Code выдает подсказку для импорта, и разработчик может принять неверное решение — импортировать тип из файла resultMessage_types, а потом гадать, что же происходит. Особенно актуально это на больших проектах. Файл Index.ts позволит избежать этих проблем.

Проведем сборку с помощью Webpack в режиме »mode=development» и в режиме »mode=production» с файлами Index.ts.

  1. mode=development:

8e474900bc4b5340d7778eff4280b796.png

Сборка прошла за 3519 миллисекунд, результирующий файл main.js весит 9.02 килобайта.

Без Index.ts файлов сборка в режиме »development» прошла за 3380 миллисекунд и сформировала результирующий файл main.js размером 6.16 килобайт, значит в режиме разработки файлы индекса увеличивают время сборки и попадают в результирующий бандл main.js. Это плохие новости.

Попробуем собрать в режиме «production».

  1. mode=production:

    d1abc0cacd2656c28e1379310fdb3654.png

Сборка прошла за 3915 миллисекунд, результирующий файл main.js весит 50 байт.

Без Index.ts файлов сборка в режиме »production» заняла 3622 миллисекунды, а результирующий бандл весил 50 байт, значит добавление файлов индекса увеличивает время сборки, но в результате обработки файлы индекса не попадают в результирующий бандл main.js., а значит не происходит увеличение бандла, который в итоге будет использоваться в «production».

Подведем итоги. Выделим + и — использования файлов Index.ts.

Минусы:

  1. Приходится создавать лишние файлы;

  2. Файл проходит через все стадии компиляции;

  3. Увеличивается время сборки;

  4. При сборке в режиме »mode=development» включаются в бандл, увеличивая его размер.

Плюсы:

  1. Пропадает необходимости прописывать путь вплоть до файла;

  2. При импорте из Index.ts мы не завязаны на внутреннюю реализацию и структуру файлов внутри общей папки, из которой происходит экспорт. Это дает нам больше свободы, гибкости в случаях, когда необходимо переименовать, перенести папку или файл, содержащий экспортируемое значение;

  3. При экспорте через файл Index.ts мы защищены от экспорта типов или переменных с одинаковыми названиями. И это может быть банальным задвоением, копированием, а ведь содержимое может быть разным…;

  4. Значительно уменьшается количество строк импорта;

  5. Облегчаем поиск файлов для импорта и делаем их более однозначными.

  6. При сборке в режиме »mode=production» файлы Index.ts не включаются в бандл, не увеличивают размер бандла.

Итого:

На мой взгляд, умеренное использование файла Index.ts в разработке несомненно дает исключительно положительный эффект, если его использовать как «оглавление» для большой директории, содержащей типы, компоненты, модули приложения.

Самое главное, что при сборке для »production» наличие данных файлов не влияет на размер бандла.

В режиме »development» увеличение размера бандла и времени пересборки не столь критично, сколько польза от использования данного файла.

Спасибо за внимание! Пишите комментарии, буду рад узнать Ваше мнение по данному вопросу!

© Habrahabr.ru