Адаптер — это шаблон структурного проектирования, который позволяет объектам с несовместимыми интерфейсами взаимодействовать друг с другом.
Также известен как “Обертка”.
Проблема
Чтобы продемонстрировать этот шаблон, я буду использовать упрощенный пример игровой механики, в которой есть интерфейс IEnemy, но один из врагов отличается от других и не имеет реализации метода атаки. Вместо этого, этот конкретный враг (SpecialEnemy) накладывает заклинания.
Решение
Вы можете создать адаптер. Это специальный объект, который преобразует интерфейс одного объекта, чтобы другой объект мог его понять. Адаптер оборачивает один из объектов, чтобы скрыть сложность преобразования, происходящего за кулисами. Обернутый объект даже не знает об адаптере. Например, вы можете обернуть объект, который работает в метрах и километрах, с помощью адаптера, который преобразует все данные в британские единицы, такие как футы и мили. Адаптеры могут не только преобразовывать данные в различные форматы, но также могут помогать объектам с разными интерфейсами взаимодействовать друг с другом. Вот как это работает:
Адаптер получает интерфейс, совместимый с одним из существующих объектов.
Используя этот интерфейс, существующий объект может безопасно вызывать методы адаптера.
При получении вызова адаптер передает запрос второму объекту, но в формате и порядке, которые ожидает второй объект.
Иногда даже можно создать двусторонний адаптер, который может конвертировать вызовы в обоих направлениях. Вернемся к нашему игровому приложению. Чтобы решить дилемму несовместимых врагов, вы можете создать адаптер EnemyAdapter для каждого класса особенных врагов, с которым ваш код работает напрямую. Затем вы настраиваете свой код для связи с SpecialEnemy только через эти адаптеры. Когда адаптер получает вызов, он переводит входящие атаки в кастование спеллов.
Структура
Существует несколько видов адаптеров
Object adapter
В этой реализации используется принцип композиции объектов: адаптер реализует интерфейс одного объекта и обертывает другой. Его можно реализовать на всех популярных языках программирования.
Class adapter
В этой реализации используется наследование: адаптер наследует интерфейсы от обоих объектов одновременно. Обратите внимание, что этот подход может быть реализован только в языках программирования, поддерживающих множественное наследование, таких как C ++.
Применимость
Используйте класс Adapter, если вы хотите использовать какой-либо существующий класс, но его интерфейс несовместим с остальной частью вашего кода.
Шаблон адаптера позволяет вам создать класс среднего уровня, который служит транслятором между вашим кодом и унаследованным классом, сторонним классом или любым другим классом со странным интерфейсом.
Используйте шаблон, если вы хотите повторно использовать несколько существующих подклассов, в которых отсутствуют некоторые общие функции и нельзя добавить в суперкласс.
Вы можете расширить каждый подкласс и добавить недостающие функции в новые дочерние классы. Однако вам придется продублировать код во всех этих новых классах, что очень плохо "пахнет".
Гораздо более элегантным решением было бы поместить недостающие функции в класс адаптера. Затем вы должны обернуть объекты с недостающими функциями внутри адаптера, динамически получая необходимые функции. Чтобы это работало, целевые классы должны иметь общий интерфейс, а поле адаптера должно следовать за этим интерфейсом. Этот подход очень похож на шаблон Decorator.
Реализация
Добавляем специальных врагов.
Создадим класс SpecialEnemy, использующий метод CastSpell.
class SpecialEnemy {
public String castSpell() {
return "using spell";
}
}
Я буду использовать строку возврата, чтобы упростить процесс.
Затем, создадим интерфейс IEnemy для управления поведением врагов, в данном случае это только один метод для атаки
interface IEnemy {
String attack();
}
Теперь нам нужно реализовать этот интерфейс в классе EnemyAdapter, который соединит обе части вместе.
class EnemyAdapter implements IEnemy {
SpecialEnemy e = new SpecialEnemy();
public string attack() {
return e.CastSpell();
}
}
Нам нужно добавить ссылку на SpecialEnemy, чтобы иметь доступ к методу CastSpell. Таким образом, мы можем использовать метод атаки без реализации его в SpecialEnemy. Или мы можем создать конструктор и передать туда экземпляр SpecialEnemy.
class EnemyAdapter implements IEnemy {
SpecialEnemy e;
EnemyAdapter(SpecialEnemy se) {
this.e = se;
}
public string attack() {
return e.CastSpell();
}
}
Последний шаг - вызвать метод Enemy Attack в клиентском классе.
class Main {
public static void main(String[] args) {
IEnemy enemy = new EnemyAdapter();
System.out.println(enemy.Attack());
}
}
В качестве вывода мы получаем строку «использование заклинания». Этот пример был слишком упрощен, но он демонстрирует идею, лежащую в основе этого простого шаблона. Надеюсь, он вам понравился и был полезен.
Комментарии (8)
Matroskin24
14.09.2021 07:03Адаптер и Обёртка (Декоратор) - это разные паттерны
playermet
14.09.2021 12:53Слово wrapper в английском языке применяется и для адаптера, и для декоратора.
Matroskin24
15.09.2021 18:43-1Применяется где? Гугл переводчик среди всех возможных вариантов перевода слова "wrapper" не показывает "адаптер", а само слово "адаптер" переводит как "adapter". Но, это всё - мелочи, главное, что в книге "Приемы объектно-ориентированного проектирования. Паттерны проектирования", авторы которой известны как "Банда четырёх", адаптер и декоратор (он же - обёртка) - это разные паттерны
playermet
16.09.2021 10:12+1Применяется где?
Людьми, в общении вживую и в интернете.
Гугл переводчик среди всех возможных вариантов перевода слова "wrapper" не показывает "адаптер"
Он и не должен его показывать, потому что это переводчик, а не справочник по терминам программирования.
Но, это всё - мелочи, главное, что в книге "Приемы объектно-ориентированного проектирования. Паттерны проектирования", авторы которой известны как "Банда четырёх", адаптер и декоратор (он же - обёртка) - это разные паттерны
Открываем GoF в оригинале, листаем до страницы адаптера, начинаем читать:
Matroskin24
16.09.2021 11:12Супер! А теперь открываем паттерн "Декоратор" и видим
Означает ли это, что адаптер и обёртка - один и тот же паттерн? Нет не означает. В статье речь идёт как раз об адаптере а не об обёртке. Смешивать их - неправильно
playermet
16.09.2021 18:31Означает ли это, что адаптер и обёртка - один и тот же паттерн?
Нет, это означает, цитирую свое первое сообщение, "Слово wrapper в английском языке применяется и для адаптера, и для декоратора.". Никакого смешивания в статье нет, они просто цитируют GoF.
А в некоторых источниках wrapper это вообще паттерн, объединяющий под собой adapter и decorator как частные случаи. И такое направление мыслей очень легко понять. Когда мы помещаем один класс в другой, и используем первый через вызовы второго, получается мы его оборачиваем. А значит то что снаружи - обертка. Я например неоднократно сталкивался с тем что wrapper'ами называют плюсовые классы вокруг какого-то элемента winapi, например сокетов или окон. Причем сделанные не с целью подогнать интерфейс, как в случае адаптера, а только чтобы упростить их использование. А паттерны только затем и записаны, чтобы систематизировать то как программисты часто называют некоторые приемы.
LeshaRB
А причем здесь дизайн игр?
dmlogv
Сложности перевода ведь. «Дизайн» — понятие растяжимое