Читати книгу - "Занурення в патерни проектування, Олександр Швець"
Шрифт:
Інтервал:
Добавити в закладку:
Отже, абстракція (або інтерфейс) — це уявний рівень керування чим-небудь, що не виконує роботу самостійно, а делегує її рівню реалізації (який зветься платформою).
Тільки не плутайте ці терміни з інтерфейсами або абстрактними класами вашої мови програмування — це не одне і те ж саме.
Якщо говорити про реальні програми, то абстракцією може виступати графічний інтерфейс програми (GUI), а реалізацією — низькорівневий код операційної системи (API), до якого графічний інтерфейс звертається, реагуючи на дії користувача.
Ви можете розвивати програму у двох різних напрямках:
мати кілька різних GUI (наприклад, для звичайних користувачів та адміністраторів). підтримувати багато видів API (наприклад, працювати під Windows, Linux і macOS).Така програма може виглядати як один великий клубок коду, в якому змішано умовні оператори рівнів GUI та API.
Коли зміни беруть проект в «осаду», вам легше відбиватися, якщо розділити монолітний код на частини.
Ви можете спробувати структурувати цей хаос, створивши для кожної з варіацій інтерфейсу-платформи свої підкласи. Але такий підхід призведе до зростання класів комбінацій, і з кожною новою платформою їх буде все більше й більше.
Ми можемо вирішити цю проблему, застосувавши Міст. Патерн пропонує розплутати цей код, розділивши його на дві частини:
Абстракцію: рівень графічного інтерфейсу програми. Реалізацію: рівень взаємодії з операційною системою.Один з варіантів крос-платформової архітектури.
Абстракція делегуватиме роботу одному з об’єктів реалізації. Причому, реалізації можна буде взаємозаміняти, але тільки за умови, що всі вони слідуватимуть єдиному інтерфейсу.
Таким чином, ви зможете змінювати графічний інтерфейс програми, не чіпаючи низькорівневий код роботи з операційною системою. І навпаки, ви зможете додавати підтримку нових операційних систем, створюючи нові підкласи реалізації, без необхідності правити код у класах графічного інтерфейсу.
СтруктураАбстракція містить керуючу логіку. Код абстракції делегує реальну роботу пов’язаному об’єктові реалізації.
Реалізація описує загальний інтерфейс для всіх реалізацій. Всі методи, які тут описані, будуть доступні з класу абстракції та його підкласів.
Інтерфейси абстракції та реалізації можуть або збігатися, або бути абсолютно різними. Проте, зазвичай в реалізації живуть базові операції, на яких будуються складні операції абстракції.
Конкретні реалізації містять платформо-залежний код.
Розширені абстракції містять різні варіації керуючої логіки. Як і батьківский клас, працює з реалізаціями тільки через загальний інтерфейс реалізацій.
Клієнт працює тільки з об’єктами абстракції. Не рахуючи початкового зв’язування абстракції з однією із реалізацій, клієнтський код не має прямого доступу до об’єктів реалізації.
ПсевдокодУ цьому прикладі Міст ділить монолітний код приладів та пультів на дві частини: прилади (виступають реалізацією) і пульти керування ними (виступають абстракцією).
Приклад поділу двох ієрархій класів — приладів та пультів керування.
Клас пульта має посилання на об’єкт приладу, яким він керує. Пульти працюють з приладами через загальний інтерфейс. Це дає можливість зв’язати пульти з різними приладами.
Пульти можна розвивати незалежно від приладів. Для цього достатньо створити новий підклас абстракції. Ви можете створити як простий пульт з двома кнопками, так і більш складний пульт з тач-інтерфейсом.
Клієнтському коду залишається вибрати версію абстракції та реалізації, з якими він хоче працювати, та зв’язати їх між собою.
// Клас пультів має посилання на пристрій, яким керує. Методи// цього класу делегують роботу методам пов'язаного пристрою.
class Remote is
protected field device: Device
constructor Remote(device: Device) is
this.device = device
method togglePower() is
if (device.isEnabled()) then
device.disable()
else
device.enable()
method volumeDown() is
device.setVolume(device.getVolume() - 10)
method volumeUp() is
device.setVolume(device.getVolume() + 10)
method channelDown() is
device.setChannel(device.getChannel() - 1)
method channelUp() is
device.setChannel(device.getChannel() + 1)
// Ви можете розширювати клас пультів, не чіпаючи код пристроїв.
class AdvancedRemote extends Remote is
method mute() is
device.setVolume(0)
// Всі пристрої мають спільний інтерфейс, тому з ними може
// працювати будь-який пульт.
interface Device is
method isEnabled()
method enable()
method disable()
method getVolume()
method setVolume(percent)
method getChannel()
method setChannel(channel)
// Разом з цим, кожен пристрій має особливу реалізацію.
class Tv implements Device is
// ...
class Radio implements Device is
// ...
// Десь у клієнтському програмному коді.
tv = new Tv()
remote = new Remote(tv)
remote.togglePower()
radio = new Radio()
remote = new AdvancedRemote(radio) Застосування
Якщо ви хочете розділити монолітний клас, який містить кілька різних реалізацій якої-небудь функціональності (наприклад, якщо клас може працювати з різними системами баз даних).
Чим більший клас, тим важче розібратись у його коді, і тим більше це розтягує час розробки. Крім того, зміни, що
!Увага!
Сайт зберігає кукі вашого браузера. Ви зможете в будь-який момент зробити закладку та продовжити читання книги «Занурення в патерни проектування, Олександр Швець», після закриття браузера.