Интерфейсы vs. классы
Обсуждая с различными людьми — в большинстве своём опытными разработчиками — классический труд «Приёмы объектно-ориентированного проектирования. Паттерны проектирования» Гаммы, Хелма и др., я с изумлением встретил полное непонимание одного из базовых подходов ООП — различия классов и интерфейсов.
Авторам книги этот вопрос кажется настолько прозрачным, что они посвящают ему едва две страницы, предполагая, что читателям всё это и так должно быть очевидно. И, действительно, это не вызвало у меня никаких вопросов — это казалось настолько само собой разумеющимся, что, когда я некоторое время назад встретил у сразу нескольких программистов откровенное непонимание концепции, я даже не смог найти слов, чтобы объяснить её суть.
Поэтому я попытался систематизировать своё понимание вопроса в этой заметке.
Главное отличие класса от интерфейса — в том, что класс состоит из интерфейса и реализации.
Любой класс всегда неявно объявляет свой интерфейс — то, что доступно при использовании класса извне. Если у нас есть класс Ключ и у него публичный метод Открыть, который вызывает приватные методы Вставить, Повернуть и Вынуть, то интерфейс класса Ключ состоит из метода Открыть. Когда мы унаследуем какой-то класс от класса Ключ, он унаследует этот интерфейс.
Кроме этого интерфейса, у класса есть также реализация — методы Вставить, Повернуть, Вынуть и их вызов в методе Открыть. Наследники Ключа наследуют вместе с интерфейсом и реализацию.
И вот здесь таятся проблемы. Предположим, у нас есть некая модель, которая предполагает использование ключа для открытия двери. Она знает интерфейс Ключа и поэтому вызывает метод Открыть.
Но, предположим, некоторые двери открываются не таким вот поворотным ключом, а магнитной карточкой — которая ведь тоже по своей сути ключ! Интерфейс этой карточки никак принципиально не отличается от интерфейса обычного ключа — можно Открыть ключом, а можно Открыть карточкой.
И мы хотим сделать класс Магнитную Карточку, который тоже будет содержать интерфейс Ключа. Для этого мы унаследуем Магнитную Карточку от Ключа. Но вместе с интерфейсом унаследовалась и реализация, которая в методе Открыть вызывает методы Вставить, Повернуть и Вынуть — а это совершенно не подходит для Магнитной Карточки.
Нам придётся самое меньшее перегружать в Магнитной Карточке реализацию метода Открыть, используя уже последовательность Вставить, Провести и Вынуть. Это уже плохо, потому что мы не знаем детали реализации класса Ключ — вдруг мы упустили какое-то очень важное изменение данных, которое должно было быть сделано — и было сделано в методе Ключ:: Открыть? Нам придётся лезть во внутреннюю реализацию Ключа и смотреть, что и как — даже если у нас есть такая техническая возможность (open source навсегда и так далее), это грубое нарушение инкапсуляции, которое ни к чему хорошему не приведёт.
Именно так и пишут Гамма и др.: наследование является нарушением инкапсуляции.
Можете попробовать самостоятельно поразмышлять над такими вопросами:
— Что делать с тем фактом, что Ключ вставляется просто в скважину, а Магнитная Карточка — обязательно сверху (не посередине и не снизу)?
— Что делать, когда нам понадобиться сделать Бесконтактную Карточку, которую надо не вставлять, а подносить?
Правильный подход к данному вопросу — отделить интерфейс от реализации. Во многих современных языках для этого предусмотрен специальный синтаксис. В отсталых языках можно использовать хаки, например, в Цэ-два-креста — чисто абстрактные классы и множественное наследование (извините за обсценную лексику — из песни слова не выкинешь).
Мы должны опираться на интерфейсы, а не классы.
Объявим интерфейс Ключ, содержащий метод Открыть.
Объявим класс Поворотный Ключ, реализующий интерфейс Ключ при помощи своих методов Вставить, Повернуть и Вынуть.
Объявим класс Магнитная Карточка, тоже реализующий интерфейс Ключ, но уже по-своему — и без каких-либо неприятных пересечений с реализацией Поворотного Ключа. Этого помогло нам достичь отделение интерфейса от реализации.
Всегда помните об общем принципе: нам нужно использовать не класс, а интерфейс. Нам не важно, что это за штука — поворотный ключ или магнитная карточка, — нам важно, что им можно открыть дверь. То есть, вместо того, чтобы задумываться о природе объекта, мы задумываемся о способах его использования.
Вам может показаться странным, но это именно то, что отличает человека от животного — использование интерфейсов вместо классов. Вы наверняка помните классический опыт с обезьяной, которую приучили гасить огонь водой из ведёрка; а потом поставили ведёрко на плот посреди бассейна, но обезьяна всё равно бегала по мостику на плот и черпала воду из ведёрка, вместо того, чтобы черпать воду прямо из бассейна. То есть обезьянка использовала класс Вода-в-Ведёрке вместо интерфейса Вода (и даже больше, скажу по секрету: вместо интерфейса Средство-для-Тушения).
Когда мы мыслим классами — уподобляемся животным. Люди мыслят (и программируют) интерфейсами.
Использование интерфейсов даёт большие возможности. Например, класс может реализовывать несколько интерфейсов: класс Ключ-от-Домофона может содержать интерфейсы Ключ и Брелок.
Что касается наследования классов, которое, как вы помните, нарушает инкапсуляцию — часто вместо наследования лучше использовать делегирование и композицию. Не забыли Бесконтактную Карточку? Так хочется сделать её родственной Магнитной Карточке! (Например, чтобы знать, что их обе можно положить в Отделение-для-Карточек в Бумажнике.) Однако у них, кажется, нет ничего общего: интерфейс Ключ их роднит в той же мере, что и Поворотный Ключ с Ключом-от-Домофона.
Решение? Делаем класс (а может, и интерфейс — подумайте, что здесь подойдёт лучше) Карточка, реализующий интерфейс Ключ за счёт делегирования Магнитной Карточке либо же Бесконтактной Карточке. А чтобы узнать, что такое делегирование и композиция, а так же при чём тут абстрактная фабрика — обратитесь к книгам, посвящённым паттернам проектирования.
У использования интерфейсов вместо классов есть ещё много преимуществ. Вы сами сможете увидеть их на практике. Оставайтесь людьми!
P.S. К моему удивлению, я не смог найти на «Хабрахабре» ни блога, посвящённого ООП, ни блога, посвящённого программированию в целом. Если они есть — укажите, пожалуйста, а пока за неимением лучших вариантов размещаю в персональном блоге.
Определение множественного наследования: какие утверждения верны
Множественное наследование — одна из самых интересных и спорных возможностей объектно-ориентированного программирования. Оно позволяет классу наследовать свойства нескольких родительских классов, что может быть полезно в определенных ситуациях. Однако, множественное наследование может приводить к некоторым проблемам и неоднозначностям, что делает его использование изначально ограниченным.
Правильное использование множественного наследования предполагает определенные ограничения и соблюдение некоторых правил. Во-первых, классы-родители не должны иметь конфликтующих методов или атрибутов. В случае, если такие конфликты возникают, необходимо явно указывать, какую реализацию следует использовать в классе-наследнике. Во-вторых, множественное наследование может усложнить понимание и поддержку кода, поэтому его использование требует особого внимания и документирования.
В ходе использования множественного наследования возникает некоторое количество вопросов и утверждений, которые необходимо проанализировать и проверить на истинность. Например, можно ли утверждать, что множественное наследование всегда полезно и удобно? Наследуются ли от родительских классов только методы и атрибуты, или еще что-то? Какие есть альтернативы множественному наследованию? На эти и другие вопросы мы постараемся ответить в данной статье.
Утверждение 1: Множественное наследование является механизмом, позволяющим классу наследовать свойства и методы от нескольких родительских классов.
Множественное наследование — это механизм объектно-ориентированного программирования, который позволяет классу наследовать свойства и методы от нескольких родительских классов. Это означает, что класс может получить функциональность от нескольких разных классов.
Когда класс наследует от нескольких родительских классов, он наследует все свойства и методы, объявленные в каждом из этих классов. Это позволяет классам унаследовать и использовать код, который определен в разных местах.
Преимущество множественного наследования заключается в том, что оно позволяет создавать гибкие и модульные структуры кода. Классы могут наследовать от разных классов и расширять их функциональность, создавая новый класс, который сочетает в себе свойства и методы из разных источников.
Однако, множественное наследование может привести к проблемам, например, при возникновении конфликтов имен методов или свойств, определенных в разных родительских классах. В таких случаях необходимо явно указать, из какого родительского класса должен быть взят метод или свойство.
В целом, множественное наследование предоставляет мощный инструмент для создания гибкого и модульного кода, но его использование требует внимания к деталям и бережного проектирования классов и их иерархии.
Утверждение 2: Одним из основных преимуществ множественного наследования является возможность повторного использования кода и уменьшение дублирования функционала.
Множественное наследование является мощным средством организации кода и является одним из ключевых преимуществ, которые он предоставляет. Оно позволяет классу наследовать функциональность от нескольких родительских классов, что позволяет повторно использовать уже существующий код и уменьшает дублирование функционала.
Когда класс наследует от нескольких классов, он получает доступ к методам и атрибутам каждого из них. Это позволяет использовать код, написанный в родительских классах, без необходимости его дублирования в каждом классе. Наследование от нескольких классов позволяет создать класс, который объединяет функциональность из разных источников и предоставляет его в одном месте.
Такой подход повышает читаемость кода, упрощает его сопровождение и ускоряет разработку, поскольку повторное использование кода позволяет снизить объем кода, который необходимо написать. Он также позволяет легко добавлять и изменять функциональность, поскольку изменения в родительских классах автоматически применяются ко всем классам, которые их наследуют.
Однако необходимо быть аккуратным при использовании множественного наследования, поскольку оно может привести к проблемам с коллизиями имен и сложному пониманию иерархии классов. Неправильное использование или непродуманное использование множественного наследования может вызвать запутанность кода и усложнение отладки.
В целом, использование множественного наследования позволяет повторно использовать код и уменьшить дублирование функционала. Оно является мощным инструментом разработки и должно использоваться внимательно и обдуманно для достижения наилучших результатов.
Утверждение 3: В некоторых языках программирования множественное наследование может привести к проблемам с недостатком ясности в иерархии классов.
Множественное наследование — это возможность для класса наследоваться от нескольких классов одновременно. Хотя это может быть полезным инструментом, некоторые языки программирования, такие как C++, могут столкнуться с проблемами, связанными с недостатком ясности в иерархии классов.
Когда класс наследуется от нескольких классов, возникает проблема, когда эти классы имеют одинаковые методы или свойства. Например, представим ситуацию, когда класс «A» и класс «B» оба имеют метод «doSomething()». Если класс «C» наследуется как от «A», так и от «B», возникает вопрос о том, какой метод «doSomething()» должен быть вызван. Это может вызвать путаницу и привести к ошибкам в коде.
Чтобы решить эту проблему, некоторые языки программирования предлагают механизм разрешения конфликтов, когда возникает ситуация множественного наследования. Например, в C++ разработчик может явно указать, какой метод или свойство должно быть использовано при конфликте. Однако в таких случаях разработчик должен быть внимателен и аккуратен при определении иерархии классов, чтобы избежать неоднозначности.
Другой проблемой, связанной с недостатком ясности в иерархии классов при множественном наследовании, может быть нарушение принципов проектирования, таких как принцип подстановки Лисков. Этот принцип утверждает, что объекты должны быть заменяемы друг за друга без изменения корректности программы. Однако, если класс наследуется от нескольких классов с разными реализациями одного и того же метода, замена одного класса другим может привести к непредсказуемым результатам.
В целом, множественное наследование может быть полезным инструментом, но разработчикам следует быть осторожными и аккуратными при его использовании. Недостаток ясности в иерархии классов может быть проблемой, если неоправданно множественное наследование приводит к конфликтам и путанице в коде.
Утверждение 4: При множественном наследовании возможно возникновение конфликтов имен методов или свойств, которые нужно разрешить вручную.
При использовании множественного наследования в классе возникает возможность унаследовать методы и свойства от нескольких родительских классов. В некоторых случаях это может привести к конфликтам имен, когда классы-родители имеют методы или свойства с одинаковыми именами. В таких ситуациях необходимо разрешить конфликты имен методов или свойств вручную.
Существуют несколько способов разрешения конфликтов имен:
- Изменение имени метода или свойства в дочернем классе. Можно переименовать метод или свойство, чтобы избежать конфликта с методом или свойством родительского класса. Например, если у родительского класса есть метод getName() , а у другого родительского класса тоже есть метод с таким же именем, можно переименовать один из методов в дочернем классе.
- Использование метода или свойства из одного из родительских классов с помощью явного указания имени класса. В случае конфликта имен можно использовать родительский класс, у которого нужный метод или свойство, с помощью указания имени класса при вызове метода или обращении к свойству.
- Использование механизмов языка программирования для разрешения конфликтов имен. Некоторые языки программирования предлагают специальные механизмы для разрешения конфликтов имен при множественном наследовании, такие как «виртуальное наследование» или «разрешение конфликтов имен с помощью приоритета».
Разрешение конфликтов имен при множественном наследовании требует внимательности и аккуратности при проектировании классов. Важно правильно выбирать способ разрешения конфликта, чтобы сохранить семантику и функциональность класса.
Утверждение 5: В некоторых языках программирования, таких как C++, используется виртуальное наследование, которое позволяет решить проблемы с повторным наследованием.
Виртуальное наследование является механизмом, используемым в некоторых языках программирования, чтобы обойти проблемы, связанные с повторным наследованием. Оно позволяет решить проблему «алмазного наследования», когда класс наследуется от двух классов, которые имеют общего предка.
В C++ виртуальное наследование позволяет избежать создания дублирующихся экземпляров общего предка при множественном наследовании.
Ключевым словом virtual указывается виртуальное наследование.
Например, если класс A является базовым классом для классов B и C, а класс D наследуется от B и C, то при использовании виртуального наследования класс D будет иметь только один экземпляр класса A, который будет общим для классов B и C. Это помогает избежать проблем с дублированием данных и вызовом неоднозначных функций, которые могут возникать при использовании обычного множественного наследования.
Виртуальное наследование также обеспечивает упорядоченное разрешение коллизий, когда комбинируются методы и члены данных от нескольких базовых классов.
Класс-наследник может точно указать, какой именно метод или данные нужно использовать из каждого базового класса. Это делается с помощью указания полного имени базового класса перед именем метода или данных в коде.
Однако, необходимо быть внимательным при использовании виртуального наследования, так как это может усложнить код и привести к нарушению принципов проектирования. Использование виртуального наследования следует оценивать с учетом конкретной задачи и потребностей проекта.
Утверждение 6: Множественное наследование может быть полезным для создания более гибкой и масштабируемой системы классов.
Множественное наследование – это возможность создания класса, который наследует свойства и методы от двух или более родительских классов. Такой подход может быть очень полезным для создания более гибкой и масштабируемой системы классов.
Позволяет использовать повторное использование кода.
Одно из основных преимуществ множественного наследования — возможность повторного использования кода. Если классы A и B имеют ряд общих свойств и методов, используя множественное наследование, можно создать новый класс C, который наследует эти свойства и методы и дополнительно может иметь свои собственные.
Обеспечивает более гибкую архитектуру.
Множественное наследование позволяет создавать более гибкую архитектуру классов. Возможность наследовать свойства и методы от различных родительских классов позволяет создавать более специализированные классы, которые лучше отражают реальную структуру предметной области. Это позволяет более точно моделировать отношения между объектами и улучшает структуру программы.
Однако, следует отметить, что множественное наследование может быть сложным в понимании и использовании. При неумелом использовании, может возникнуть проблема «алмазного наследования», когда класс C наследует одни и те же свойства и методы от двух родительских классов, приводя к неоднозначности вызова методов и ухудшению производительности. Поэтому, необходимо внимательно планировать и проектировать структуру классов при использовании множественного наследования.
Утверждение 7: Для более безопасного и понятного программирования рекомендуется использовать композицию или интерфейсы вместо множественного наследования
Множественное наследование в языках программирования может стать источником сложностей и проблем, особенно при неосторожном использовании. Поэтому для достижения более безопасного и понятного программирования рекомендуется использовать другие подходы, такие как композиция и интерфейсы.
Композиция – это механизм, который позволяет объединить несколько классов в один, используя отношение «содержит». Класс, который содержит другие классы, называется контейнером, а вложенные классы – компонентами. При использовании композиции, классы контейнера не наследуются напрямую от классов компонентов, а имеют экземпляры этих классов как свои поля.
Использование композиции вместо множественного наследования позволяет избежать проблем, связанных с наследованием от нескольких классов. В частности, композиция позволяет избежать проблем, связанных с алмазным наследованием, когда класс наследует одновременно от двух классов, которые имеют общего предка.
Интерфейсы – это контракты, которые определяют, какие методы должны быть реализованы классом. Они определяют только сигнатуры методов, но не их реализацию. Классы могут реализовывать несколько интерфейсов, что позволяет им обладать набором различных функциональностей.
Использование интерфейсов также способствует более безопасному программированию. Интерфейсы позволяют обеспечить модульность и гибкость кода, поскольку классы, реализующие интерфейсы, могут быть заменены другими классами, которые также реализуют эти интерфейсы.
В итоге, для достижения более безопасного и понятного программирования рекомендуется использовать композицию или интерфейсы вместо множественного наследования. Это позволяет избежать проблем, связанных с наследованием от нескольких классов, и обеспечивает более гибкую и понятную структуру кода.
Вопрос-ответ
Можно ли в языке программирования Python использовать множественное наследование?
Да, в Python можно использовать множественное наследование. В этом языке программирования класс может наследовать свойства и методы от нескольких базовых классов.
Какие преимущества есть у множественного наследования?
Множественное наследование позволяет использовать функциональность нескольких классов в одном производном классе. Это позволяет упростить код, избежать дублирования и повторного использования кода, а также создавать более гибкую иерархию классов.
Как решается проблема алмаза при множественном наследовании?
Проблема алмаза возникает, когда один класс наследует свойства и методы от двух других классов, которые сами являются потомками одного и того же базового класса. В Python проблема алмаза решается таким образом, что методы базового класса вызываются только один раз, чтобы избежать неоднозначности.
Верно ли, что множественное наследование усложняет чтение и понимание кода?
Множественное наследование может усложнить чтение и понимание кода, особенно если не учитывать его особенности и не следовать принципам хорошего программирования. Однако, если правильно использовать множественное наследование, разделять иерархию классов на отдельные части и разбивать её на более простые компоненты, то код может быть понятным и поддерживаемым.
Может ли множественное наследование привести к конфликтам имен?
Да, множественное наследование может привести к конфликтам имен, когда один класс наследует методы с одинаковыми именами из разных базовых классов. В таких случаях необходимо явно указывать, какой метод следует вызывать или какой метод нужно переопределить. Это позволяет разрешить конфликт имен и выбрать нужную функциональность.
ООП в картинках
ООП (Объектно-Ориентированное Программирование) стало неотъемлемой частью разработки многих современных проектов, но, не смотря на популярность, эта парадигма является далеко не единственной. Если вы уже умеете работать с другими парадигмами и хотели бы ознакомиться с оккультизмом ООП, то впереди вас ждет немного лонгрид и два мегабайта картинок и анимаций. В качестве примеров будут выступать трансформеры.
Прежде всего стоит ответить, зачем? Объектно-ориентированная идеология разрабатывалась как попытка связать поведение сущности с её данными и спроецировать объекты реального мира и бизнес-процессов в программный код. Задумывалось, что такой код проще читать и понимать человеком, т. к. людям свойственно воспринимать окружающий мир как множество взаимодействующих между собой объектов, поддающихся определенной классификации. Удалось ли идеологам достичь цели, однозначно ответить сложно, но де-факто мы имеем массу проектов, в которых с программиста будут требовать ООП.
Не следует думать, что ООП каким-то чудным образом ускорит написание программ, и ожидать ситуацию, когда жители Вилларибо уже выкатили ООП-проект в работу, а жители Виллабаджо все еще отмывают жирный спагетти-код. В большинстве случаев это не так, и время экономится не на стадии разработки, а на этапах поддержки (расширение, модификация, отладка и тестирование), то бишь в долгосрочной перспективе. Если вам требуется написать одноразовый скрипт, который не нуждается в последующей поддержке, то и ООП в этой задаче, вероятнее всего, не пригодится. Однако, значительную часть жизненного цикла большинства современных проектов составляют именно поддержка и расширение. Само по себе наличие ООП не делает вашу архитектуру безупречной, и может наоборот привести к излишним усложнениям.
Иногда можно столкнуться с критикой в адрес быстродействия ООП-программ. Это правда, незначительный оверхед присутствует, но настолько незначительный, что в большинстве случаев им можно пренебречь в пользу преимуществ. Тем не менее, в узких местах, где в одном потоке должны создаваться или обрабатываться миллионы объектов в секунду, стоит как минимум пересмотреть необходимость ООП, ибо даже минимальный оверхед в таких количествах может ощутимо повлиять на производительность. Профилирование поможет вам зафиксировать разницу и принять решение. В остальных же случаях, скажем, где львиная доля быстродействия упирается в IO, отказ от объектов будет преждевременной оптимизацией.
В силу своей природы, объектно-ориентированное программирование лучше всего объяснять на примерах. Как и обещал, нашими пациентами будут трансформеры. Я не трансформеролог, и комиксов не читал, посему в примерах буду руководствоваться википедией и фантазией.
Классы и объекты
Сразу лирическое отступление: объектно-ориентированный подход возможен и без классов, но мы будем рассматривать, извиняюсь за каламбур, классическую схему, где классы — наше всё.
Самое простое объяснение: класс — это чертеж трансформера, а экземпляры этого класса — конкретные трансформеры, например, Оптимус Прайм или Олег. И хотя они и собраны по одному чертежу, умеют одинаково ходить, трансформироваться и стрелять, они оба обладают собственным уникальным состоянием. Состояние — это ряд меняющихся свойств. Поэтому у двух разных объектов одного класса мы можем наблюдать разное имя, возраст, местоположение, уровень заряда, количество боеприпасов и т. д. Само наличие этих свойств и их типы описываются в классе.
Таким образом, класс — это описание того, какими свойствами и поведением будет обладать объект. А объект — это экземпляр с собственным состоянием этих свойств.
Мы говорим «свойства и поведение», но звучит это как-то абстрактно и непонятно. Привычнее для программиста будет звучать так: «переменные и функции». На самом деле «свойства» — это такие же обычные переменные, просто они являются атрибутами какого-то объекта (их называют полями объекта). Аналогично «поведение» — это функции объекта (их называют методами), которые тоже являются атрибутами объекта. Разница между методом объекта и обычной функцией лишь в том, что метод имеет доступ к собственному состоянию через поля.
Итого, имеем методы и свойства, которые являются атрибутами. Как работать с атрибутами? В большинстве ЯП оператор обращения к атрибуту — это точка (кроме PHP и Perl). Выглядит это примерно вот так (псевдокод):
// объявление класса с помощью ключевого слова class class Transformer() < // объявление поля x int x // объявление метода конструктора (сюда нам чуть ниже передадут 0) function constructor(int x)< // инициализация поля x // (переданный конструктору 0 превращается в свойство объекта) this.x = x >// объявление метода run function run() < // обращение к собственному атрибуту через this this.x += 1 >> // а теперь клиентский код: // создаем новый экземпляр трансформера с начальной позицией 0 optimus = new Transformer(0) optimus.run() // приказываем Оптимусу бежать print optimus.x // выведет 1 optimus.run() // приказывает Оптимусу еще раз бежать print optimus.x // выведет 2
В картинках я буду использовать такие обозначения:
Я не стал использовать UML-диаграммы, посчитав их недостаточно наглядными, хоть и более гибкими.
Что мы видим из кода?
1. this — это специальная локальная переменная (внутри методов), которая позволяет объекту обращаться из своих методов к собственным атрибутам. Обращаю внимание, что только к собственным, то бишь, когда трансформер вызывает свой метод, либо меняет собственное состояние. Если снаружи обращение будет выглядеть так: optimus.x, то изнутри, если Оптимус захочет сам обратиться к своему полю x, в его методе обращение будет звучать так: this.x, то есть «я (Оптимус) обращаюсь к своему атрибуту x«. В большинстве языков эта переменная называется this, но встречаются и исключения (например, self)
2. constructor — это специальный метод, который автоматически вызывается при создании объекта. Конструктор может принимать любые аргументы, как и любой другой метод. В каждом языке конструктор обозначается своим именем. Где-то это специально зарезервированные имена типа __construct или __init__, а где-то имя конструктора должно совпадать с именем класса. Назначение конструкторов — произвести первоначальную инициализацию объекта, заполнить нужные поля.
3. new — это ключевое слово, которое необходимо использовать для создания нового экземпляра какого-либо класса. В этот момент создается объект и вызывается конструктор. В нашем примере, конструктору передается 0 в качестве стартовой позиции трансформера (это и есть вышеупомянутая инициализация). Ключевое слово new в некоторых языках отсутствует, и конструктор вызывается автоматически при попытке вызвать класс как функцию, например так: Transformer().
4. Методы constructor и run работают с внутренним состоянием, а во всем остальном не отличаются от обычных функций. Даже синтаксис объявления совпадает.
5. Классы могут обладать методами, которым не нужно состояние и, как следствие, создание объекта. В этом случае метод делают статическим.
SRP
(Single Responsibility Principle / Принцип единственной ответственности / Первый принцип SOLID). С ним вы, наверняка, уже знакомы из других парадигм: «одна функция должна выполнять только одно законченное действие». Этот принцип справедлив и для классов: «Один класс должен отвечать за какую-то одну задачу». К сожалению с классами сложнее определить грань, которую нужно пересечь, чтобы принцип нарушался.
Существуют попытки формализовать данный принцип с помощью описания назначения класса одним предложением без союзов, но это очень спорная методика, поэтому доверьтесь своей интуиции и не бросайтесь в крайности. Не нужно делать из класса швейцарский нож, но и плодить миллион классов с одним методом внутри — тоже глупо.
Ассоциация
Традиционно в полях объекта могут храниться не только обычные переменные стандартных типов, но и другие объекты. А эти объекты могут в свою очередь хранить какие-то другие объекты и так далее, образуя дерево (иногда граф) объектов. Это отношение называется ассоциацией.
Предположим, что наш трансформер оборудован пушкой. Хотя нет, лучше двумя пушками. В каждой руке. Пушки одинаковые (принадлежат к одному классу, или, если будет угодно, выполненные по одному чертежу), обе одинаково умеют стрелять и перезаряжаться, но в каждой есть свое хранилище боеприпасов (собственное состояние). Как теперь это описать в ООП? С помощью ассоциации:
class Gun() < // объявляем класс Пушка int ammo_count // объявляем количество боеприпасов function constructor()< // конструктор this.reload() // вызываем собственный метод "перезарядить" >function fire() < // объявляем метод пушки "стрелять" this.ammo_count -= 1 // расходуем боеприпас из собственного магазина >function reload() < // объявляем метод "перезарядить" this.ammo_count = 10 // забиваем собственный магазин боеприпасами >> class Transformer() < // объявляем класс Трансформер Gun gun_left // объявляем поле "левая пушка" типа Пушка Gun gun_right // объявляем поле "правая пушка" тоже типа Пушка /* теперь конструктор Трансформера принимает в качестве аргументов две уже конкретные созданные пушки, которые передаются извне */ function constructor(Gun gun_left, Gun gun_right)< this.gun_left = gun_left // устанавливаем левую пушку на борт this.gun_right = gun_right // устанавливаем правую пушку на борт >// объявляем метод Трансформер "стрелять", который сначала стреляет. function fire() < // левой пушкой, вызывая ее метод "стрелять" this.gun_left.fire() // а затем правой пушкой, вызывая такой же метод "стрелять" this.gun_right.fire() >> gun1 = new Gun() // создаем первую пушку gun2 = new Gun() // создаем вторую пушку optimus = new Transformer(gun1, gun2) // создаем трансформера, передавая ему обе пушки
this.gun_left.fire() и this.gun_right.fire() — это обращения к дочерним объектам, которые происходят так же через точки. По первой точке мы обращаемся к атрибуту себя (this.gun_right), получая объект пушки, а по второй точке обращаемся к методу объекта пушки (this.gun_right.fire()).
Итог: робота сделали, табельное оружие выдали, теперь разберемся, что тут происходит. В данном коде один объект стал составной частью другого объекта. Это и есть ассоциация. Она в свою очередь бывает двух видов:
1. Композиция — случай, когда на фабрике трансформеров, собирая Оптимуса, обе пушки ему намертво приколачивают к рукам гвоздями, и после смерти Оптимуса, пушки умирают вместе с ним. Другими словами, жизненный цикл дочернего объекта совпадает с жизненным циклом родительского.
2. Агрегация — случай, когда пушка выдается как пистолет в руку, и после смерти Оптимуса этот пистолет может подобрать его боевой товарищ Олег, а затем взять в свою руку, либо сдать в ломбард. То бишь жизненный цикл дочернего объекта не зависит от жизненного цикла родительского, и может использоваться другими объектами.
Ортодоксальная ООП-церковь проповедует нам фундаментальную троицу — инкапсуляцию, полиморфизм и наследование, на которых зиждется весь объектно-ориентированный подход. Разберем их по порядку.
Наследование
Наследование — это механизм системы, который позволяет, как бы парадоксально это не звучало, наследовать одними классами свойства и поведение других классов для дальнейшего расширения или модификации.
Что если, мы не хотим штамповать одинаковых трансформеров, а хотим сделать общий каркас, но с разным обвесом? ООП позволяет нам такую шалость путем разделения логики на сходства и различия с последующим выносом сходств в родительский класс, а различий в классы-потомки. Как это выглядит?
Оптимус Прайм и Мегатрон — оба трансформеры, но один является автоботом, а второй десептиконом. Допустим, что различия между автоботами и десептиконами будут заключаться только в том, что автоботы трансформируются в автомобили, а десептиконы — в авиацию. Все остальные свойства и поведение не будут иметь никакой разницы. В таком случае можно спроектировать систему наследования так: общие черты (бег, стрельба) будут описаны в базовом классе «Трансформер», а различия (трансформация) в двух дочерних классах «Автобот» и «Десептикон».
class Transformer() < // базовый класс function run()< // код, отвечающий за бег >function fire() < // код, отвечающий за стрельбу >> class Autobot(Transformer) < // дочерний класс, наследование от Transformer function transform()< // код, отвечающий за трансформацию в автомобиль >> class Decepticon(Transformer) < // дочерний класс, наследование от Transformer function transform()< // код, отвечающий за трансформацию в самолет >> optimus = new Autobot() megatron = new Decepticon()
Сей пример наглядно иллюстрирует, как наследование становится одним из способов дедуплицировать код (DRY-принцип) с помощью родительского класса, и одновременно предоставляет возможности для мутации в классах-потомках.
Перегрузка
Если же в классе-потомке переопределить уже существующий метод в классе-родителе, то сработает перегрузка. Это позволяет не дополнять поведение родительского класса, а модифицировать. В момент вызова метода или обращения к полю объекта, поиск атрибута происходит от потомка к самому корню — родителю. То есть, если у автобота вызвать метод fire(), сначала поиск метода производится в классе-потомке — Autobot, а поскольку его там нет, поиск поднимается на ступень выше — в класс Transformer, где и будет обнаружен и вызван. Следует отметить, что модификация нарушает LSP из набора принципов SOLID, но мы рассматриваем только техническую возможность.
Неуместное применение
Любопытно, что чрезмерно глубокая иерархия наследования может привести к обратному эффекту — усложнению при попытке разобраться, кто от кого наследуется, и какой метод в каком случае вызывается. К тому же, не все архитектурные требования можно реализовать с помощью наследования. Поэтому применять наследование следует без фанатизма. Существуют рекомендации, призывающие предпочитать композицию наследованию там, где это уместно. Любая критика наследования, которую я встречал, подкрепляется неудачными примерами, когда наследование используется в качестве золотого молотка. Но это совершенно не означает, что наследование в принципе всегда вредит. Мой нарколог говорил, что первый шаг — это признать, что у тебя зависимость от наследования.
Как при описании отношений двух сущностей определить, когда уместно наследование, а когда — композиция? Можно воспользоваться популярной шпаргалкой: спросите себя, сущность А является сущностью Б? Если да, то скорее всего, тут подойдет наследование. Если же сущность А является частью сущности Б, то наш выбор — композиция.
Применительно к нашей ситуации это будет звучать так:
- Автобот является Трансформером? Да, значит выбираем наследование.
- Пушка является частью Трансформера? Да, значит — композиция.
Наследование статично
Еще одно важное отличие наследования от композиции в том, что наследование имеет статическую природу и устанавливает отношения классов только на этапе интерпретации/компиляции. Композиция же, как мы видели в примерах, позволяет менять отношение сущностей на лету прямо в рантайме — иногда это очень важно, поэтому об этом нужно помнить при выборе отношений (если конечно нет желания использовать метапрограммирование).
Множественное наследование
Мы рассмотрели ситуацию, когда два класса унаследованы от общего потомка. Но в некоторых языках можно сделать и наоборот — унаследовать один класс от двух и более родителей, объединив их свойства и поведение. Возможность наследоваться от нескольких классов вместо одного — это множественное наследование.
Вообще, в кругах иллюминатов бытует мнение, что множественное наследование — это грех, оно несет за собой ромбовидную проблему и неразбериху с конструкторами. Кроме того, задачи, которые решаются множественным наследованием, можно решать другими механизмами, например, механизмом интерфейсов (о котором мы тоже поговорим). Но справедливости ради, следует отметить, что множественное наследование удобно использовать для реализации примесей.
Абстрактные классы
Кроме обычных классов в некоторых языках существуют абстрактные классы. От обычных классов они отличаются тем, что нельзя создать объект такого класса. Зачем же нужен такой класс, спросит читатель? Он нужен для того, чтобы от него могли наследоваться потомки — обычные классы, объекты которых уже можно создавать.
Абстрактный класс наряду с обычными методами содержит в себе абстрактные методы без имплементации (с сигнатурой, но без кода), которые обязан имплементировать программист, задумавший создать класс-потомок. Абстрактные классы не обязательны, но они помогают установить контракт, обязующий имплементировать определенный набор методов, дабы уберечь программиста с плохой памятью от ошибки имплементации.
Полиморфизм
Полиморфизм — свойство системы, позволяющее иметь множество реализаций одного интерфейса. Ничего непонятно. Обратимся к трансформерам.
Положим, у нас есть три трансформера: Оптимус, Мегатрон и Олег. Трансформеры боевые, стало быть обладают методом attack(). Игрок, нажимая у себя на джойстике кнопку «воевать», сообщает игре, чтобы та вызвала метод attack() у трансформера, за которого играет игрок. Но поскольку трансформеры разные, а игра интересная, каждый из них будет атаковать каким-то своим способом. Скажем, Оптимус — объект класса Автобот, а Автоботы снабжаются пушками с плутониевыми боеголовками (да не прогневаются фанаты трансформеров). Мегатрон — Десептикон, и стреляет из плазменной пушки. Олег — басист, и он обзывается. А в чем польза?
Польза полиморфизма в данном примере заключается в том, что код игры ничего не знает о реализации его просьбы, кто как должен атаковать, его задача просто вызвать метод attack(), сигнатура которого одинакова для всех классов персонажей. Это позволяет добавлять новые классы персонажей, или менять методы существующих, не меняя код игры. Это удобно.
Инкапсуляция
Инкапсуляция — это контроль доступа к полям и методам объекта. Под контролем доступа подразумевается не только можно/неможно, но и различные валидации, подгрузки, вычисления и прочее динамическое поведение.
Во многих языках частью инкапсуляции является сокрытие данных. Для этого существуют модификаторы доступа (опишем те, которые есть почти во всех ООП языках):
- publiс — к атрибуту может получить доступ любой желающий
- private — к атрибуту могут обращаться только методы данного класса
- protected — то же, что и private, только доступ получают и наследники класса в том числе
class Transformer() < public function constructor()< >protected function setup() < >private function dance() < >>
Как правильно выбрать модификатор доступа? В простейшем случае так: если метод должен быть доступен внешнему коду, выбираем public. В противном случае — private. Если есть наследование, то может потребоваться protected в случае, когда метод не должен вызываться снаружи, но должен вызываться потомками.
Аксессоры (геттеры и сеттеры)
Геттеры и сеттеры — это методы, задача которых контролировать доступ к полям. Геттер считывает и возвращают значение поля, а сеттер — наоборот, принимает в качестве аргумента значение и записывает в поле. Это дает возможность снабдить такие методы дополнительными обработками. Например, сеттер при записи значения в поле объекта, может проверить тип, или входит ли значение в диапазон допустимых (валидация). В геттер же можно добавить, ленивую инициализацию или кэширование, если актуальное значение на самом деле лежит в базе данных. Применений можно придумать множество.
В некоторых языках есть синтаксический сахар, позволяющий такие аксессоры маскировать под свойства, что делает доступ прозрачным для внешнего кода, который и не подозревает, что работает не с полем, а с методом, у которого под капотом выполняется SQL-запрос или чтение из файла. Так достигается абстракция и прозрачность.
Интерфейсы
Задача интерфейса — снизить уровень зависимости сущностей друг от друга, добавив больше абстракции.
Не во всех языках присутствует этот механизм, но в ООП языках со статической типизацией без них было бы совсем худо. Выше мы рассматривали абстрактные классы, затрагивая тему контрактов, обязующих имплементировать какие-то абстрактные методы. Так вот интерфейс очень смахивает на абстрактный класс, но является не классом, а просто пустышкой с перечислением абстрактных методов (без имплементации). Другими словами, интерфейс имеет декларативную природу, то есть, чистый контракт без капельки кода.
Обычно в языках, в которых есть интерфейсы, нет множественного наследования классов, но есть множественное наследование интерфейсов. Это позволяет классу перечислить интерфейсы, которые он обязуется имплементировать.
Классы с интерфейсами состоят в отношении «многие ко многим»: один класс может имплементировать множество интерфейсов, и каждый интерфейс, в свою очередь, может имплементироваться многими классами.
У интерфейса двустороннее применение:
- По одну сторону интерфейса — классы, имплементирующие данный интерфейс.
- По другую сторону — потребители, которые используют этот интерфейс в качестве описания типа данных, с которым они (потребители) работают.
Представим, что каркас трансформера оборудован тремя слотами: слот для оружия, для генератора энергии и для какого-нибудь сканера. Эти слоты обладают определенными интерфейсами: в каждый слот можно установить только подходящее оборудование. В слот для оружия можно установить ракетную установку или лазерную пушку, в слот для генератора энергии — ядерный реактор или РИТЭГ (радиоизотопный термоэлектрический генератор), а в слот для сканера — радар или лидар. Суть в том, что каждый слот имеет универсальный интерфейс подключения, а уже конкретные устройства должны соответствовать этому интерфейсу. К примеру, на материнских платах используется несколько типов слотов: слот для процессора позволяет подключать различные процессоры, подходящие под данный сокет, а слот SATA — любой SSD или HDD накопитель или даже CD/DVD.
Обращаю внимание, что получившаяся система слотов у трансформеров — это пример использования композиции. Если же оборудование в слотах будет сменным в ходе жизни трансформера, то тогда это уже агрегация. Для наглядности, мы будем называть интерфейсы, как принято в некоторых языках, добавляя заглавную «И» перед именем: IWeapon, IEnergyGenerator, IScanner.
// описания интерфейсов: interface IWeapon < function fire() <>// декларация метода без имплементации. Ниже аналогично > interface IEnergyGenerator < // тут уже два метода, которые должны будут реализовать классы: function generate_energy() <>// первый function load_fuel() <> // второй > interface IScanner < function scan() <>> // классы, реализующие интерфейсы: class RocketLauncher() : IWeapon < function fire()< // имплементация запуска ракеты >> class LaserGun() : IWeapon < function fire()< // имплементация выстрела лазером >> class NuclearReactor() : IEnergyGenerator < function generate_energy()< // имплементация генерации энергии ядерным реактором >function load_fuel() < // имплементация загрузки урановых стержней >> class RITEG() : IEnergyGenerator < function generate_energy()< // имплементация генерации энергии РИТЭГ >function load_fuel() < // имплементация загрузки РИТЭГ-пеллет >> class Radar() : IScanner < function scan()< // имплементация использования радиолокации >> class Lidar() : IScanner < function scan()< // имплементация использования оптической локации >> // класс - потребитель: class Transformer() < // привет, композиция: IWeapon slot_weapon // Интерфейсы указаны в качестве типов данных. IEnergyGenerator slot_energy_generator // Они могут принимать любые объекты, IScanner slot_scanner // которые имплементируют указанный интерфейс /* в параметрах методов интерфейс тоже указан как тип данных, метод может принимать объект любого класса, имплементирующий данный интерфейс: */ function install_weapon(IWeapon weapon)< this.slot_weapon = weapon >function install_energy_generator(IEnergyGenerator energy_generator) < this.slot_energy_generator = energy_generator >function install_scanner(IScanner scanner) < this.slot_scanner = scanner >> // фабрика трансформеров class TransformerFactory() < function build_some_transformer() < transformer = new Transformer() laser_gun = new LaserGun() nuclear_reactor = new NuclearReactor() radar = new Radar() transformer.install_weapon(laser_gun) transformer.install_energy_generator(nuclear_reactor) transformer.install_scanner(radar) return transformer >> // использование transformer_factory = new TransformerFactory() oleg = transformer_factory.build_some_transformer()
К сожалению, в картинку не влезла фабрика, но она все равно необязательна, трансформера можно собрать и во дворе.
Обозначенный на картинке слой абстракции в виде интерфейсов между слоем имплементации и слоем-потребителем дает возможность абстрагировать одних от других. Вы можете это наблюдать, посмотрев на каждый слой в отдельности: в слое имплементации (слева) нет ни слова про класс Transformer, а в слое-потребителе (справа) нет ни слова про конкретные имплементации (там нет слов Radar, RocketLauncher, NuclearReactor и т. д.)
В таком коде мы можем создавать новые комплектующие к трансформерам, не затрагивая чертежи самих трансформеров. В то же время и наоборот, мы можем создавать новых трансформеров, комбинируя уже существующие комплектующие, либо добавлять новые комплектующие, не меняя существующих.
Утиная типизация
Явление, которое мы наблюдаем в получившейся архитектуре, называется утиной типизацией: если что-то крякает как утка, плавает как утка, и выглядит как утка, то, скорее всего — это утка.
Переводя это на язык трансформеров, звучать будет так: если что-то стреляет как пушка, и перезаряжается как пушка, скорее всего, это пушка. Если устройство генерирует энергию, скорее всего, это генератор энергии.
В отличие от иерархической типизации наследования, при утиной типизации трансформеру пофиг, какого класса пушку ему дали, и пушка ли это вообще. Главное, что эта штуковина умеет стрелять! Это не достоинство утиной типизации, а скорее компромисс. Может быть и обратная ситуация, как на этой картинке ниже:
ISP
(Interface Segregation Principle / Принцип разделения интерфейса / Четвертый принцип SOLID) призывает не создавать жирные универсальные интерфейсы. Вместо этого интерфейсы нужно разделять на более мелкие и специализированные, это поможет гибче их комбинировать в имплементирующих классах, не заставляя имплементировать лишние методы.
Абстракция
В ООП все крутится вокруг абстракции. Существуют фанатики, утверждающие, что абстракция должна быть частью ООП-троицы (инкапсуляция, полиморфизм, наследование). А мой инспектор по УДО говорил обратное: абстракция присуща для любого программирования, а не только для ООП, поэтому она должна стоять отдельно. С другой стороны, то же самое можно сказать и про остальные принципы, но из песни слов не выкинешь. Так или иначе, абстракция нужна, и особенно в ООП.
Уровень абстракции
Тут нельзя не процитировать одну известную шутку:
— любую архитектурную проблему можно решить добавлением дополнительного слоя абстракции, кроме проблемы большого количества абстракций.
В нашем примере с интерфейсами мы внедрили слой абстракции между трансформерами и комплектующими, сделав архитектуру более гибкой. Но какой ценой? Нам пришлось усложнить архитектуру. Мой психотерапевт говорил, что умение балансировать между простотой архитектуры и гибкостью приложения — это искусство. Выбирая золотую середину, следует опираться не только на собственный опыт и интуицию, но и на контекст текущего проекта. Поскольку будущее человек видеть пока не научился, нужно аналитически прикинуть, какой уровень абстракции и с какой долей вероятности может пригодиться в данном проекте, сколько времени потребуется на проработку гибкой архитектуры, и окупится ли затраченное время в будущем.
Неверный выбор уровня абстракции ведет к одной из двух проблем:
- если абстракции недостаточно, дальнейшие расширения проекта будут упираться в архитектурные ограничения, которые ведут либо к рефакторингу и смене архитектуры, либо к обилию костылей (оба варианта обычно несут за собой боль и финансовые потери)
- если уровень абстракции слишком высок, это приведет к оверинжинирингу в виде чересчур сложной архитектуры, которую трудно поддерживать, и излишней гибкости, которая никогда в этом проекте не пригодится. В этой ситуации любые простейшие изменения в проекте будут сопровождаться дополнительной работой для удовлетворения требований архитектуры (это тоже порой несет определенную боль и финансовые потери)
Еще важно понимать, что уровень абстракции определяется не для всего проекта в целом, а отдельно для разных компонентов. В каких-то местах системы абстракции может быть недостаточно, а где-то наоборот — перебор. Однако, неверный выбор уровня абстракции можно исправить своевременным рефакторингом. Ключевое слово — своевременным. Запоздалый рефакторинг провести проблематично, когда на данном уровне абстракции реализовано уже множество механизмов. Проводить обряд рефакторинга в запущенных системах может сопрягаться с острой болью в труднодоступных местах программиста. Это примерно как поменять фундамент в доме — дешевле построить рядом дом с нуля.
Давайте рассмотрим определение уровня абстракции из возможных вариантов на примере гипотетической игры «трансформеры-онлайн». Уровни абстракции в данном случае будут выступать как слои, каждый последующий рассматриваемый слой будет ложиться поверх предыдущего, забирая из него часть функционала в себя.
Первый слой. В игре есть один класс трансформера, все свойства и поведение описаны в нем. Это совсем деревянный уровень абстракции, подходит для казуальной игры, которая не предполагает никакой особой гибкости.
Второй уровень. В игре есть базовый трансформер с основными способностями и классы трансформеров со своей специализацией (типа разведчик, штурмовик, саппорт), которая описывается дополнительными методами. Тем самым игроку предоставляется возможность выбора, а разработчикам упрощается добавление новых классов.
Третий уровень. Помимо классификации трансформеров вводится агрегация с помощью системы слотов и компонентов (как в нашем примере с реакторами, пушками и радарами). Теперь часть поведения будет определяться тем, какой стаф игрок установил в своего трансформера. Это дает игроку еще больше возможностей для кастомизации игровой механики персонажа, а разработчикам дает возможность добавлять эти самые модули расширения, что в свою очередь упрощает работу гейм-дизайнерам по выпуску нового контента.
Четвертый уровень. В компоненты можно тоже включить собственную агрегацию, предоставляющую возможность выбора материалов и деталей, из которого собираются эти компоненты. Такой подход даст игроку возможность не только набивать трансформеров нужными комплектующими, но и самостоятельно производить эти комплектующие из различных деталек. Признаться, такой уровень абстракции я в играх никогда не встречал, и не без резона! Ведь это сопровождается значительным усложнением архитектуры, а регулировка баланса в таких играх превращается в ад. Но не исключаю, что такие игры существуют.
Как видим, каждый описанный слой, в принципе, имеет право на жизнь. Все зависит от того, какую именно гибкость мы хотим заложить в проект. Если в техническом задании ничего об этом не сказано, или автор проекта сам не знает, что может потребовать бизнес, можно посмотреть на похожие проекты в этой сфере и ориентироваться на них.
Паттерны проектирования
Десятилетия разработки привели к тому, что сформировался список наиболее часто применяемых архитектурных решений, которые со временем были классифицированы сообществом, и стали называться паттернами проектирования. Именно поэтому, когда я прочитал впервые про паттерны, я с удивлением обнаружил, что оказывается, многие из них я уже использую на практике, просто не знал, что у этих решений есть название.
Паттерны проектирования, как и абстракция, свойственны не только ООП разработке, но и другим парадигмам. Вообще, тема паттернов выходит за рамки данной статьи, но здесь хотелось бы предостеречь молодого разработчика, который только намерен познакомиться с паттернами. Это ловушка! Сейчас объясню, почему.
Предназначение паттернов — помощь в решении архитектурных проблем, которые либо уже обнаружились, либо, вероятнее всего, обнаружатся в ходе развития проекта. Так вот, у новичка, который прочитал про паттерны, может появиться непреодолимый соблазн использовать паттерны не для решения проблем, а для их порождения. А поскольку разработчик в своих желаниях необуздан, он может начать не решать задачу при помощи паттернов, а подстраивать любые задачи под решения с помощью паттернов.
Еще одна ценность от паттернов — формализации терминологии. Гораздо проще коллеге сказать, что в этом месте используется «цепочка обязанностей», чем полчаса рисовать поведение и отношения объектов на бумажке.
Заключение
В условиях современных требований наличие в вашем коде слова class не делает из вас ООП-программиста. Ибо если вы не используете описанные в статье механизмы (полиморфизм, композицию, наследование и т. д.), а вместо этого применяете классы лишь для группировки функций и данных, то это не ООП. То же самое можно решить какими-нибудь неймспейсами и структурами данных. Не путайте, иначе на собеседовании будет стыдно.
Хочется закончить свою песнь важными словами. Любые описанные механизмы, принципы и паттерны, как и ООП в целом не стоит применять там, где это бессмысленно или может навредить. Это ведет к появлению статей со странными заголовками типа «Наследование — причина преждевременного старения» или «Синглтон может приводить к онкологическим заболеваниям».
Я серьезно. Если рассмотреть случай с синглтоном, то его повсеместное применение без знания дела, стало причиной серьезных архитектурных проблем во многих проектах. И любители забивать гвозди микроскопом любезно его нарекли антипаттерном. Будьте благоразумны.
К сожалению, в проектировании не существует однозначных рецептов на все случаи жизни, где что применять уместно, а где неуместно. Это будет постепенно укладываться в голове с опытом.
Какие утверждения верны относительно определения множественного наследования
[. ] [. ] заместитель Генерального директора по вопросам коммуникации и информации; он подчеркнул тот важный вклад, который средства информации с их способностью формировать восприятие реальности, могут внести в дело развития диалога, основанного на взаимном уважении, который должен охватывать широкий круг участников и способствовать установлению таким образом множественного диалога.
unesdoc.unesco.org
The discussions of the Round Table were introduced by Mr Abdul Waheed Khan, Assistant Director-General for Communication and Information, highlighting the essential contribution that media, with its power to shape the perception of reality, can make to foster dialogue founded on mutual respect that must involve a great variety of actors, thus creating a “multilogue”.
unesdoc.unesco.org
Для вопросов множественного выбора, комментарий отображается [. ]
только для варианта, выбранного учеником.
moodle.itcm.ru
For Multiple Choice questions, feedback is displayed only [. ]
for the answer the student selected.
moodle.itcm.ru
Однако даже при склонении этих двух слов очевидно, что окончания множественного числа почти полностью вытеснили окончания двойственного числа.
But even in these two words, the student will remark that the inflection of the plural has almost entirely superseded that of the dual.
Мы молчали и не устраивали никаких акций протеста, боясь, как бы правительство не обвинило нас в
[. ] [. ] связях с Игорем Гиоргадзе.145 В день выборов представитель Крайсис Груп наблюдал в Марнеули и Болниси многочисленные нарушения, в том числе завышение данных по явке избирателей, случаи подделки членами участковых комиссий подписей в списках избирателей, вмешательства представителей ОНД, непоследовательного осуществления чернильной маркировки, голосования без удостоверяющих личность документов, а также попытки множественного голосования.
crisisgroup.org
We remained silent and did not launch any protest actions, as we were afraid that the government might link us to Igor Giorgadze.145 On election day, Crisis Group observed numerous violations in Marneuli and Bolnisi, including inflated turnouts, PEC members forging signatures in the voter registry, interference by UNM representatives, inconsistent application of inking, lack of identity verification and attempts at multiple voting.
crisisgroup.org
При испрашивании множественного приоритета эти коды ИНИД могут появляться несколько раз, [. ]
по меньшей мере, для данного серийного номера и даты.
If multiple priorities are invoked, these INID codes can have multiple occurrences of at [. ]
least the serial number and the date.
Конвенции Совета Европы, содержащие положения относительно снятия оговорок, в целом
следуют этой формуле; см.
[. ] пункт 2 статьи 8 Конвенции о сокращении случаев множественного гражданства и о воинских обязательствах в случаях множественного гражданства 1963 года; пункт 2 статьи 13 Европейского [. ]
соглашения о передаче
просьб относительно судебной помощи 1977 года или пункт 3 статьи 29 Европейской конвенции о гражданстве 1997 года.
daccess-ods.un.org
Council of Europe conventions containing clauses on the withdrawal of reservations generally follow this
formula: cf. the 1963
[. ] Convention on the Reduction of Cases of Multiple Nationality and on Military Obligations in Cases of Multiple Nationality, art. 8, para. 2; the 1977 European Agreement on the Transmission [. ]
of Applications
for Legal Aid, art. 13, para. 2; or the 1997 European Convention on Nationality, art. 29, para. 3.
daccess-ods.un.org
Метод включает этапы функционального анализа, структурного синтеза, и поиск множественного числа оптимальных решений и совмещает лексикографический критерий преимущества (L-критерий) для отбора электронных компонентов на этапе функционального анализа и безусловный критерий преимущества (оптимум по Парето, ?-критерий) на этапе поиска множественного числа оптимальных решений, которое рассматривается в литературе, как альтернативные методы поиска оптимальных решений.
computingonline.net
It is based on a method of morphological analysis and synthesis and includes phases of functional analysis, structural synthesis, and search for a set of optimal solutions. The proposed method combines lexicographical criterion of preference (L- criterion) at a stage of functional analysis and unconditional criterion of preference (Pareto optimality) during the search phase, which are considered in literature as alternative methods of search for optimal solutions.