Делегирование (объектно-ориентированное программирование) - Delegation (object-oriented programming)

В объектно-ориентированного программирования, делегация относится к оценке члена (свойства или метода) одного объекта (получателя) в контексте другого исходного объекта (отправителя). Делегирование можно сделать явно, передав объект-отправитель объекту-получателю, что можно сделать в любом объектно-ориентированный язык; или неявно, правилами поиска членов на языке, который требует языковой поддержки для функции. Неявное делегирование - это фундаментальный метод повторного использования поведения в программирование на основе прототипов, соответствующий наследование в программирование на основе классов. Наиболее известными языками, поддерживающими делегирование на языковом уровне, являются: Себя, который включает в себя понятие делегирования через понятие изменяемых родительских слотов, которые используются при поиске метода при самостоятельных вызовах, и JavaScript; видеть Делегирование JavaScript.

Период, термин делегация также свободно используется для различных других отношений между объектами; видеть делегирование (программирование) для большего. Часто путаемые понятия просто используют другой объект, более точно называемый консультация или же агрегирование; и оценка члена на одном объекте путем оценки соответствующего члена на другом объекте, особенно в контексте принимающего объекта, который более точно упоминается как пересылка (когда объект-оболочка не передает себя завернутому объекту).[1][2][а] В схема делегирования это шаблон разработки программного обеспечения для реализации делегирования, хотя этот термин также широко используется для обозначения консультации или пересылки.

Обзор

Это чувство делегация как функция языка программирования, использующая правила поиска метода для отправки так называемых самозвонки был определен Либерман в своей статье 1986 г. «Использование прототипов объектов для реализации общего поведения в объектно-ориентированных системах».

Делегирование зависит от динамическое связывание, поскольку это требует, чтобы вызов данного метода мог вызывать разные сегменты кода во время выполнения.[нужна цитата ]. Он используется повсюду macOS (и его предшественник Следующий шаг ) как средство настройки поведения компонентов программы.[3] Он позволяет реализовать такие реализации, как использование одного класса, предоставляемого ОС, для управления окнами, поскольку класс принимает делегат, зависящий от программы, и может при необходимости переопределять поведение по умолчанию. Например, когда пользователь щелкает поле закрытия, диспетчер окон отправляет делегату вызов windowShouldClose:, и делегат может отложить закрытие окна, если есть несохраненные данные, представленные содержимым окна.

Делегирование можно охарактеризовать (и отличить от пересылка ) в качестве позднее связывание себя:[4]

... сообщения, отправленные себя (или же это) в родительской переменной "вернется" к объекту, который изначально получил сообщение.

Это себя в определении метода в принимающем объекте нет статически привязан к этому объекту во время определения (например, во время компиляции или когда функция прикреплена к объекту), но скорее во время оценка время, это связано с оригинал объект.

Утверждалось, что в некоторых случаях делегирование может быть предпочтительнее наследование сделать программный код более читаемым и понятным.[5] Несмотря на то, что явное делегирование довольно широко распространено, относительно немногие основные языки программирования реализуют делегирование в качестве модели, альтернативной наследованию. Точная связь между делегированием и наследованием сложна; некоторые авторы считают их эквивалентными, либо одно - частным случаем другого.[6]

Языковая поддержка делегирования

В языках, поддерживающих делегирование через правила поиска методов, диспетчеризация методов определяется так, как они определены для виртуальных методов в наследовании: это всегда наиболее конкретный метод, который выбирается во время поиска метода. Следовательно, это оригинал объект-получатель, который является началом поиска метода, даже если он передал управление какому-то другому объекту (через ссылку делегирования, а не ссылку на объект).

Преимущество делегирования в том, что оно может происходить во время выполнения и влиять только на подмножество сущностей определенного типа и даже может быть удалено во время выполнения. Наследование, напротив, обычно нацелено на тип, а не на экземпляры, и ограничивается временем компиляции. С другой стороны, наследование может быть статически проверено типом, в то время как делегирование обычно невозможно без дженериков (хотя ограниченная версия делегирования может быть статически типобезопасной.[7]). Делегирование можно назвать «наследованием во время выполнения для определенных объектов».

Вот псевдокод пример в C # /Ява как язык:

учебный класс А {  пустота фу() {    // «это» также известно под названиями «текущий», «я» и «я» на других языках    это.бар();  }  пустота бар() {    Распечатать("бар");  }};учебный класс B {  частный делегировать А а; // ссылка на делегирование  общественный B(А а) {    это.а = а;  }  пустота фу() {    а.фу(); // вызываем foo () в a-экземпляре  }  пустота бар() {    Распечатать("b.bar");  }};а = новый А();б = новый B(а); // устанавливаем делегирование между двумя объектами

Вызов b.foo () приведет к б. бар печатается, так как это относится к оригинал объект приемника, б, в контексте а. Возникающая в результате неоднозначность это упоминается как объект шизофрении.

Перевод неявного это в явный параметр, вызов (в B, с а делегат) a.foo () переводится на A.foo (б), используя тип а для разрешения метода, но делегирующий объект б для это аргумент.

При использовании наследования аналогичный код (с использованием заглавных букв, чтобы подчеркнуть, что разрешение основано на классах, а не на объектах):

учебный класс А {  пустота фу() {    это.бар();  }  пустота бар() {    Распечатать("Бар");  }};учебный класс B расширяет А {  общественный B() {}  пустота фу() {    супер.фу(); // вызов foo () суперкласса (A)  }  пустота бар() {    Распечатать(«Б.бар»);  }};б = новый B();

Вызов b.foo () приведет к B.bar. В этом случае, это однозначно: есть единый объект, б, и this.bar () разрешается в метод подкласса.

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

Двойное наследование

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

учебный класс C расширяет А {  ссылка на делегирование D d;}

Это требует дополнительных правил для поиска методов, поскольку теперь потенциально существует два метода, которые можно обозначить как наиболее специфичные (из-за двух путей поиска).

Связанные области

Делегирование можно описать как механизм низкого уровня для совместного использования кода и данных между объектами. Таким образом, он закладывает основу для других языковых конструкций. Примечательно ролевое программирование языки использовали делегирование, но особенно старые фактически использовали агрегирование утверждая, что использует делегирование. Это не следует рассматривать как обман, это просто множественное определение того, что означает делегирование (как описано выше).

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

Делегация также была предложена для вынесения рекомендаций в аспектно-ориентированное программирование Эрнстом и Лоренцем в 2003 году.

Смотрите также

Различать:

Примечания

  1. ^ Beck 1997 использует термины «простое делегирование» для случаев, когда объект-получатель не имеет доступа к объекту-отправителю, и «самостоятельное делегирование», когда объект-получатель действительно имеет доступ к объекту-отправителю; на современном языке это «пересылка» и «делегирование», используемые в этой статье.

Рекомендации

  1. ^ Gamma et al. 1995 г., «Делегация», стр. 20–21.
  2. ^ Бек 1997, «Делегация», стр. 64–69.
  3. ^ Apple (20 августа 2009 г.). «Руководство по основам какао: делегаты и источники данных». Подключение разработчика Apple. Получено 2009-09-11.
  4. ^ «Пересекающиеся классы и прототипы». Перспективы системной информатики: 5-я Международная конференция памяти Андрея Ершова, PSI 2003, Академгородок, Новосибирск, Россия, 9-12 июля 2003 г., Исправленные статьи. п.38.
  5. ^ [1]Трюгве Реенскауг, Факультет информатики Университета Осло, «Дело в пользу читаемого кода» (2007)
  6. ^ Штейн, Линн Андреа. Делегирование - это наследование. OOPSLA '87 Материалы конференции по системам, языкам и приложениям объектно-ориентированного программирования. С. 138–146. Дои:10.1145/38807.38820.
  7. ^ Гюнтер Книзель (1999-11-19). «Типобезопасное делегирование для адаптации компонентов времени выполнения». ECOOP '99 - объектно-ориентированное программирование. Конспект лекций по информатике. 1628. Springer. С. 351–366. CiteSeerX  10.1.1.33.7584. Дои:10.1007/3-540-48743-3_16. ISBN  978-3-540-66156-6. Архивировано из оригинал на 1999-11-19. Получено 2015-03-04. В этом документе предлагается наследование на основе объектов (также известное как делегирование) в качестве дополнения к композиции объектов, основанной исключительно на пересылке. В нем представлена ​​безопасная для типов интеграция делегирования в объектную модель на основе классов и показано, как оно преодолевает проблемы, с которыми сталкивается взаимодействие компонентов на основе пересылки, как оно поддерживает независимую расширяемость компонентов и непредвиденную динамическую адаптацию компонентов.

внешняя ссылка