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

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

Я хочу, чтобы мой код был чистым!
Я хочу, чтобы мой код был чистым!

Что такое "чистый код" и почему вас это должно беспокоить?

История возникновения и определение "чистого кода"

Обязательно стоит упомянуть одноименную книгу, написанную Робертом К. Мартином в 2008 году (Прим. переводчика:  издание на русском "Чистый код. Создание, анализ и рефакторинг."). Но и до выхода этой книги было много других книг и опытных разработчиков, говоривших о подобных концепциях.

Я вывел своего рода определение чистого кода, объединив мнения нескольких авторов и источников:

  • Чистый код важен. По крайней мере, так же важен, как и другие характеристики, такие как производительность, функциональность, не допускать багов.…

  • Легко читается любым разработчиком.

  • Легко модифицируется любым разработчиком.

  • Автор кода заботится о нем.

  • Код делает то, что от него ожидается. Код вас не обманет, никаких сюрпризов.

Почему вы должны писать чистый код

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

Программисты тратят больше времени (гораздо больше) на чтение кода, чем на его написание. Мы читаем легаси-код, код библиотек, код коллег по команде, код, написанный вами несколько месяцев назад (которого вы не помните), код, написанный кем-то, кто ушел из компании, код на Stack Overflow… Роберт Мартин резюмирует:

"На самом деле соотношение времени чтения и написания кода превышает 10:1" — Роберт С. Мартин, "Чистый код".

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

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

Некоторые принципы

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

Итак, давайте рассмотрим некоторые принципы работы с кодом.

Давайте поэкспериментируем с кодом!
Давайте поэкспериментируем с кодом!

Имена

При написании кода мы повсюду придумываем имена: переменные, функции, классы, пакеты, файлы… Серьезное отношение к именам — первый шаг к чистому коду.

Несколько советов для чистых имен:

  • Используйте имена, передающие намерения программиста:

Если вам позже встретится "s", вам будет трудно понять ее назначение
Если вам позже встретится "s", вам будет трудно понять ее назначение
  • Выбирайте произносимые имена:

Нелегко обсуждать с коллегой красную переменную без потери лица
Нелегко обсуждать с коллегой красную переменную без потери лица
  • Используйте имена, доступные для поиска:

Если вы попытаетесь найти переменную "r" с помощью инструментов поиска вашей IDE, возможно, вы найдете больше, чем хотите.
Если вы попытаетесь найти переменную "r" с помощью инструментов поиска вашей IDE, возможно, вы найдете больше, чем хотите.
  • Избегайте префиксов, суффиксов и аббревиатур

Что, черт возьми, такое "hp": гипотенуза (hypotenuse), верхняя точка (high point) …? У испаноговорящих будет еще больше вариантов.
Что, черт возьми, такое "hp": гипотенуза (hypotenuse), верхняя точка (high point) …? У испаноговорящих будет еще больше вариантов.

По именованию есть множество других рекомендаций. Вероятно, использование имен, передающих намерения, является наиболее важным, но еще более важно — серьезное отношение к именованию.

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

Функции

Среди всех идей о функциях я выделю три:

  • Первое правило простое и легко запоминающееся: функции должны делать что-то одно, делать это хорошо и делать только это. Избегайте побочных эффектов, разделяйте функции, если заметите, что они делают несколько вещей одновременно.

  • Функции должны быть небольшими. Хорошо, но насколько короткими? Как это измерить? Давайте договоримся, что в функциях будет не более двух уровней отступов. Если для вас это сложно, можно начать с более высокого предела (например, с трех уровней), но, пожалуйста, установите ограничение на уровни отступов, которые вы разрешаете.

  • По поводу аргументов… Чем меньше аргументов у функции, тем она чище. Почему? Аргументы требуют знания контекста. При каждом вызове читатель должен обладать контекстом, чтобы понять все аргументы. Больше аргументов — больше контекста, который вам нужно понять. Также возникают сложности с точки зрения тестирования. Больше аргументов — больше тестов для покрытия всех комбинаций аргументов.

У вас должна быть веская причина для функций с более чем двумя аргументами
У вас должна быть веская причина для функций с более чем двумя аргументами

Комментарии

Когда я учился программировать в 90-х, мои учителя обычно просили меня писать комментарии везде. Довольно часто приходилось слышать что-то вроде: "Если вы не будете комментировать код, я не приму ваш экзамен…". Целью этих комментариев было облегчить чтение нашего кода. Но мы преследуем эту же цель при написании чистого кода, и комментарии — не лучший способ ее достичь.

"Комментарии должны компенсировать нашу неудачу в выражении своих мыслей в коде. Комментарии — всегда признак неудачи." — Роберт С. Мартин, "Чистый код"

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

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

Основная мысль — стараться избегать комментариев для пояснения кода. Например:

  • Избегайте комментариев для переменной. Вместо этого выберите хорошее имя для этой переменной, и вам не понадобится комментарий.

  • Избегайте комментариев для функций. Вместо этого напишите функцию так, чтобы она делала только одну вещь, у нее было небольшое количество аргументов и понятные имена самой функции и аргументов, и вам не понадобятся комментарии.

Давайте рассмотрим пример:

1. У нас есть нетривиальный участок кода. Мы чувствуем, что в будущем будет трудно его понять:

Что здесь происходит? Вам понятно? Круто, но для остальных может быть все не так просто.
Что здесь происходит? Вам понятно? Круто, но для остальных может быть все не так просто.

2. Давайте добавим комментарий, чтобы стало понятнее ("Проверяем year - високосный год или нет"):

Я добавил комментарий. Теперь я могу спать спокойно. Но лучшее ли это решение?
Я добавил комментарий. Теперь я могу спать спокойно. Но лучшее ли это решение?

3. Попробуем другой вариант: извлечем этот кусок кода в метод с говорящим названием:

Теперь все чисто!
Теперь все чисто!

Подумайте, что будущему читателю кода будет интересно, когда он встретит этот if? Ему нужно понять, что этот if проверяет, является ли год високосным (leap year). Но, вероятнее всего, его не будет волновать, как выполняется эта проверка. Если все-таки это будет интересно, то он может перейти к реализации этого метода. Убрав комментарий, мы невольно также разделили разные уровни абстракции в коде.

В общем, избегайте комментариев для пояснения кода. А, так как вы используете систему управления версиями кода вроде GIT, то избегайте и закомментированного кода (удалите его!), информации об авторах частей кода и т. п.

Комментарии не запрещены, есть ситуации, когда комментарии вполне уместны:

  • Информация об авторских правах.

  • TODO-комментарии.

  • Пояснения важности чего-либо или объяснение конкретного решения в коде.

  • Комментарии публичных API обязательны (JavaDocs, …), но избегайте их в непубличном коде. Не заставляйте команду комментировать все функции всех ваших классов!

Дополнительные принципы

Как говорил ранее, я хотел привести примеры лишь нескольких принципов, чтобы вы получили основы теории "чистого кода". Итак, это всего лишь некоторые из базовых идей о чистом коде. Если они показались вам интересными и вам захотелось узнать больше, поищите дополнительные ресурсы в Интернете или прочитайте книгу Мартина.

Как сделать код чистым? Рефакторинг!

Понятные имена, маленькие функции, отсутствие комментариев для пояснения кода… все ясно. Но как это сделать? Как написать код, следуя этим концепциям?

Наша работа трудна сама по себе. Написать работающий код уже может быть достаточно сложной задачей. И это не беспокоясь о том, чтобы он был чистый.

Написать код уже достаточно сложно, и без раздумий о его чистоте
Написать код уже достаточно сложно, и без раздумий о его чистоте

Таким образом, ключевым здесь может стать Рефакторинг. Хороший подход — первоначальное написание кода без беспокойства о его чистоте, с последующей очисткой уже работающего кода с помощью рефакторинга.

Рефакторинг 

Рефакторинг кода — это процесс реструктуризации существующего кода без изменения его внешнего поведения. Это означает, что код до и после рефакторинга должен работать одинаково.

Примеры того, что не является рефакторингом:

  • Изменение алгоритма.

  • Замена одного типа цикла другим.

  • Повышение производительности фрагмента кода.

Примеры рефакторинга:

  • Извлечение фрагмента кода в функцию.

  • Переименование.

  • Извлечение нескольких функций в новый класс.

  • Создание константы для хранения жестко заданного значения в коде.

Безопасный рефакторинг

Возможно, вы думаете: "Я не хочу ломать код, он работает нормально!". Да, конечно. Написать работающий код довольно сложно, мы не хотим ломать его при рефакторинге. И это частая причина, из-за которой люди спорят, чтобы не вносить изменения в плохой код. Но не волнуйтесь, есть безопасный способ.

При рефакторинге нас спасут две вещи:

  • Тестирование. У вас должны быть хорошие автоматизированные тесты по многим причинам. И очевидно, что они помогут провести рефакторинг, ничего не сломав. После каждого рефакторинга вы можете проверить, все ли тесты по-прежнему зеленые. В этом посте я не буду писать о тестировании, возможно, в следующем. Если вы не знакомы с тестированием, вам стоит изучить этот вопрос. В сети есть много информации.

Все тесты зеленые. Вроде ничего не сломали.
Все тесты зеленые. Вроде ничего не сломали.
  • Инструменты рефакторинга. В современных IDE есть инструменты, которые автоматически выполняют некоторые из наиболее распространённых рефакторингов. При их использовании снижается вероятность того, что мы что-то сломаем при внесении изменений в код. Я расскажу о некоторых из них далее в этой статье.

Когда следует проводить рефакторинг кода?

Постоянно. Я имею в виду, что у вас должен быть следующий цикл разработки:

  • Написание кода.

  • Написание тестов.

  • Рефакторинг.

Хотя не обязательно именно в таком порядке. При разработке вы можете использовать TDD, и в этом случае вы будете писать тесты перед кодом. Но в любом случае, вы должны проводить рефакторинг каждый раз, когда напишете кусок рабочего кода. Другими словами, вы должны проводить рефакторинг в конце каждого цикла. И эти циклы должны быть небольшими.

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

Приложение. Инструменты рефакторинга

Для демонстрации инструментов рефакторинга я выбрал JetBrains IntelliJ. Но вы найдете подобные инструменты и в других IDE. А если не найдете, возможно, вам следует попробовать другую IDE.

Переименование (rename)

Самым простым рефакторингом является переименование (Rename). Например, есть сущность с именем, которое вам не нравится, и вы хотите его изменить. Конечно, можно отредактировать его вручную... но это будет непросто, так как эта сущность может использоваться во многих местах.

Например, я хочу переименовать класс Input в WordFrequency.

Поскольку это класс, есть много потенциальных мест, где я должен изменить его имя Input. Вручную я должен найти их все. Но у нас есть инструмент рефакторинга:

Инструмент переименования с различными параметрами для безопасного переименования
Инструмент переименования с различными параметрами для безопасного переименования

Этот инструмент переименует класс, а также всё остальное, что нам нужно: разные варианты использования, имя файла, тесты ... даже имена переменных, которые могут быть связаны с нашим классом:

IDE сообразило, что есть несколько переменных, которые я хотел бы изменить. Спасибо IDE!
IDE сообразило, что есть несколько переменных, которые я хотел бы изменить. Спасибо IDE!

Извлечение метода (Extract method)

Этот рефакторинг я использую чаще всего. Давайте рассмотрим пример:

  1. У меня есть фрагмент кода, являющийся частью очень большой функции, которую я хочу отрефакторить. Я думаю, что часть этих строк делает какую-то законченную операцию и поэтому ее можно вынести в приватную функцию.

Я хочу извлечь создание wordFrequencyList в метод
Я хочу извлечь создание wordFrequencyList в метод

2. Я использую рефакторинг Extract Method. У этого инструмента также есть ряд параметров:

Инструмент извлечения метода даже осмеливается предложить подходящее имя
Инструмент извлечения метода даже осмеливается предложить подходящее имя

3. Теперь код перемещен в метод, а вместо него идет вызов этого метода. Инструмент позаботился о создании метода, перемещении туда кода и изменении всех мест, где был такой же код, на этот один вызов метода (он ищет дубликаты кода).

Стало чище, не так ли?
Стало чище, не так ли?

Другие инструменты

Существует множество инструментов рефакторинга для популярных действий, которые выполняются при реорганизации кода. Говоря об IntelliJ Idea, вы можете взглянуть на ее документацию по рефакторингу, или поискать в документации вашей любимой IDE. Я рекомендую вам изучить все доступные инструменты рефакторинга вашей IDE и поэкспериментировать с ними, чтобы понять, как они работают и насколько они будут вам полезны.

Множество инструментов рефакторинга!
Множество инструментов рефакторинга!

На самом деле, в моей IDE я могу выделить кусок кода и в меню "Refactor This" увидеть все доступные рефакторинги, которые можно применить к этому конкретному случаю:

Скажи мне, добрая IDE, что я могу с этим сделать?
Скажи мне, добрая IDE, что я могу с этим сделать?

Финальный аргумент

Есть финальный аргумент в пользу поддержания чистоты вашего кода, если я до сих пор вас не убедил:

Пишите код так, как будто сопровождать его будет склонный к насилию психопат, который знает, где вы живёте — Джон Ф. Вудс (John F. Woods).


Приглашаем всех на открытое занятие «Элементы формальной логики. Базовые структуры данных в языке Java». На нем познакомимся с основами алгоритмов и булевой алгебры. В процессе мы изучим базовые структуры данных языка Java: массивы, списки и словари. Регистрация по ссылке.

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


  1. JordanCpp
    16.08.2022 18:29
    +13

    Мало, очень мало, почти нет примеров. Могли бы рассмотреть большие классы и на примере показать разделение на отдельные чистые, компактные сущности. И примеры тестов к ним. Я все больше замечаю, что таким статьям постоянно не хватает кода.

    Только прошу вас, не пишите примеры в стиле наследуем пингвина от слона и добавляем интерфейс "плавать":)

    Хотелось бы видеть более практичные кейсы.


    1. EevanW
      17.08.2022 11:59

      Подробнее чем Мартин и Макконнелл всё-равно не напишешь, их труды фундаментальны.
      Это скорее статья-напоминание.
      Ну и ссылку на доку для джавистов в статье вижу.


  1. Nialpe
    16.08.2022 18:51
    +1

    Я бы словосочетание "любым разработчиком" заменил на "разработчиком с определенным уровнем". Сталкивался с ситуацией когда адепт желания писать код понятный "любому разработчику" велосипедил и не использовал языковые возможности (принятой на проекте java), потому, что "а вдруг тот, кто будет потом не знает".


    1. ivegner
      16.08.2022 20:05
      +1

      Или ещё лучше: игнорировать существовать паттернов программирования. А то вдруг другие программисты их не знают.


      1. JordanCpp
        17.08.2022 10:38
        +1

        Согласен, но часто разработчики злоупотребляют и паттеранми и абстракциями. К примеру https://habr.com/ru/company/abbyy/blog/173885/

        Всегда нужно думать головой и искать простые решения если они приемлемы.


  1. Hrodvitnir
    17.08.2022 06:00
    +4

    Это просто статья, где много пересказа Роберта Мартина и чуток Мартина Фаулера с его рефакторингом. Хоть бы от себя добавили чего:)


    1. zloddey
      17.08.2022 07:06
      +1

      Ну как же, добавили описание Idea!

      По сути согласен: перепевка Мартина, и не более.


      1. Hrodvitnir
        17.08.2022 18:23
        +1

        Лучше, это перевод перепевки Мартина


  1. Emelian
    17.08.2022 07:51
    +1

    На самом деле здесь все из разряда «Капитан Очевидность». Не будем решать задачу «как сделать код понятный всем», хотя бы только для себя, спустя некоторое время. А сделать это достаточно нетривиально, если нет «семи пядей во лбу».

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

    Если приходится иметь дело с чужим кодом, то прежде чем начать серьезно работать с ним, я его «причесываю». Иначе, просто не могу адекватно воспринимать. Если интересно, можете посмотреть мой C++ код в моих последних двух статьях:

    habr.com/ru/post/566864
    habr.com/ru/post/466713

    По-хорошему, чтобы понять чужой код, нужно чтобы программист описал его в своей статье либо статьях. Да и мало просто хорошего стиля, должна быть еще «правильная» парадигма программирования. А тут трудно вести речь «вообще», даже по частным направлениям, скажем, программирование GUI на C++ / WTL уже может вызвать кучу споров, начиная с того, зачем вообще нужно иметь дело с этими инструментами?

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


  1. Denis_Andreevich
    17.08.2022 08:17

    Есть у кого то на примете какой-нибудь проект с открытым исходным кодом, популярный (библиотека, приложение), в котором бы было видно применение чистого кода?

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


  1. JordanCpp
    17.08.2022 10:08
    -1

    Могу привести пример из своего проекта. Не стал все пихать в один класс.

    Пример вывода линии. Делал по примеру SDL.

    https://github.com/JordanCpp/Lt/blob/master/Examples/Graphics/05_DrawLine/main.cpp

    #include <Lt/Graphics/Render.hpp>
    #include <Lt/Graphics/FpsLimiter.hpp>
    #include <Lt/Core/Console.hpp>
    #include <Lt/Graphics/FpsCounter.hpp>
    
    //Для удобства вывода ошибки
    void ShowError(const Lt::Core::ErrorHandler& errorHandler)
    {
    	Lt::Core::Console console;
    
    	console.Write(errorHandler.Message());
    	console.Show();
    }
    
    int main()
    {
      //Создаем обработчик ошибок
    	Lt::Core::ErrorHandler errorHandler;
    
      //Создаем окно
    	Lt::Graphics::Window window(&errorHandler, Lt::Graphics::Point2u(0, 0), Lt::Graphics::Point2u(800, 600), "Window!");
    
    	if (errorHandler.Error())
    	{
    		ShowError(errorHandler);
    
    		return 0;
    	}
      //Создаем рендер
    	Lt::Graphics::Render render(&errorHandler, &window);
    
    	if (errorHandler.Error())
    	{
    		ShowError(errorHandler);
    
    		return 0;
    	}
    
    	Lt::Events::Event report;
    //Создаем счетчик fps
    	Lt::Graphics::FpsCounter fpsCounter;
      //Создаем IntegerToString
    	Lt::Core::IntegerToString integerToString;
    
    	while (window.GetEvent(report))
    	{
    		render.Color(Lt::Graphics::Color(195, 195, 195));
    		render.Clear();
    
    		if (report.Type == Lt::Events::IsQuit)
    		{
    			window.StopEvent();
    		}
    
    		render.Color(Lt::Graphics::Color(237, 28, 36));
    		render.Line(Lt::Graphics::Point2u(0, 0), render.Size());
    
    		render.Present();
    
    		if (fpsCounter.Calc())
    		{
    			if (integerToString.Convert(fpsCounter.Fps()))
    			{
    				window.Title(integerToString.Result());
    			}
    
    			fpsCounter.Clear();
    		}
    	}
    
    	return 0;
    }

    В данном примере каждый класс выполняет свою небольшую задачу. Что позволило написать тесты для них.

    https://github.com/JordanCpp/Lt/tree/master/Tests/Graphics

    Так же в классы вынесены такой функционал как, создание скриншотов и ограничитель fps

    https://github.com/JordanCpp/Lt/blob/master/Tests/Graphics/Screenshoter.cpp

    К примеру конвертация пикселей, тоже имеет свой отдельный класс.

    https://github.com/JordanCpp/Lt/blob/master/Tests/Graphics/PixelConverter.cpp

    Такой способ мне помог написать тесты и не писать жирные классы содержащие джунгли и гориллу.:)


    1. JordanCpp
      17.08.2022 10:20

      Еще как пример. Не стал пихать в класс окна обработчик сообщений. Он реализован отдельно, и так же к нему написан тест.

      https://github.com/JordanCpp/Lt/blob/master/include/Lt/Events/Eventer.hpp

      Сам тест.

      https://github.com/JordanCpp/Lt/blob/master/Tests/Events/Eventer.cpp

      Сам я такую моду взял из C#, так как является основным языком для работы. С++ навещаю как любовницу:)


  1. SadOcean
    17.08.2022 10:54
    +4

    функции должны делать что-то одно

    Вот эта рекомендация часто встречается, но плохо поясняется.
    Функция должна делать одно действие на одном уровне асбтракции, используя методы из уровня абстракции ниже.


  1. oleg_shamshura
    17.08.2022 17:24
    +1

    Много лет назад ходила шутка, как отличить опытного программиста от нуба, начитавшегося умных книжек: по именам цикловых переменных. Опытный, который за свою карьеру написал их несметное количество раз, использует i, j, k. Неофит будет старательно выписывать indexOfStringArray. К великому сожалению, появление IDE с автоподсказками уничтожило этот гениальный критерий :)


  1. pin2t
    18.08.2022 09:27
    +3

    Совет по именам он прямо очень вредный. Никаких

    int fileSizeInBytes

    и уж тем более

    Date generationTimestamp

    быть не должно, несоответствие между Date и Timestamp приводит к ещё бОльшему количеству WTF. fileSizeInBytes просто долго читается и отвлекает от сути, забиывает голову ненужными нюансами. Эти переменные должна называться size и generation, то что это Date видно из типа, никакой это не Timestamp внезапно

    date.happensInLeapYear

    должно быть

    date.year.isLeap

    Как правильно уже написали выше про i,j,k в циклах полностью норм, и остальные локальные переменные должны быть одним словом, одним.

    List<WordFrequency> wordFrequencyList = getWordFrequencyList(inputStr)

    ещё один ужасный пример, на кой черт постоянно переписывать WordFrequency и List когда они уже есть в типе

    List<WordFrequency> frequencies = parse(input)

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

    Про "функция должна делать что-то одно" уже много раз говорили по-моему, что Мартин сам неоднократно переобувался что же он имел в виду под этим, очередное мутное и непонятное "правило"

    Ну единственное с чем можно согласиться так это с комментариями. Если программист не смог написать достаточно понятный код, не следует ожидать от него что он напишет понятный комментарий :-)


  1. TonyKentnarEarth
    18.08.2022 16:09

    Чем меньше аргументов у функции, тем она чище

    сперва не понял, как такое может быть:)

    потом дошло что чистая в плане clean, а не pure