Шаблоны проектирования - Design Patterns

Шаблоны проектирования:
Элементы многоразового использования Объектно-ориентированный Программного обеспечения
Шаблоны дизайна cover.jpg
Автор«Банда четырех»:
СтранаСоединенные Штаты
ПредметШаблоны проектирования, программная инженерия, объектно-ориентированного программирования
ИздательЭддисон-Уэсли
Дата публикации
1994
Страницы395
ISBN0-201-63361-2
OCLC31171684
005.1/2 20
Класс LCQA76.64 .D47 1995 г.

Паттерны проектирования: элементы объектно-ориентированного программного обеспечения многократного использования (1994) - это программная инженерия книга с описанием шаблоны проектирования программного обеспечения. Книгу написал Эрих Гамма, Ричард Хелм, Ральф Джонсон, и Джон Влиссидес, с предисловием Грейди Буч. Книга разделена на две части: в первых двух главах рассматриваются возможности и подводные камни объектно-ориентированного программирования, а в остальных главах описываются 23 классических шаблоны проектирования программного обеспечения. В книгу включены примеры в C ++ и Болтовня.

Он оказал влияние на область разработки программного обеспечения и считается важным источником теории и практики объектно-ориентированного проектирования. Было продано более 500 000 копий на английском и 13 других языках. Авторов часто называют Банда из четырех (GoF).[1]

История

Книга началась в одного поля ягода (BoF) сессия в OOPSLA '90, «Путеводитель по архитектуре», изданный Брюсом Андерсоном, где Эрих Гамма и Ричард Хелм встретились и обнаружили общий интерес. Позже к ним присоединились Ральф Джонсон и Джон Влиссидес.[2] Первоначальной датой публикации книги было 21 октября 1994 года с авторским правом 1995 года, поэтому она часто цитируется с 1995 годом, несмотря на то, что была опубликована в 1994 году. Книга была впервые представлена ​​публике на собрании OOPSLA, состоявшемся в Портленде. , Орегон, в октябре 1994 г. В 2005 г. СИГПЛАН присудил авторам Премию за достижения в области языков программирования в этом году в знак признания влияния их работы «на практику программирования и дизайн языков программирования».[3] По состоянию на март 2012 года книга выходила в 40-й тираж.

Вступление

В главе 1 обсуждается объектно-ориентированный методы проектирования, основанные на опыте авторов, которые, по их мнению, приведут к хорошему объектно-ориентированному дизайну программного обеспечения, включая:

Авторы заявляют следующие преимущества: интерфейсы над реализацией:

  • клиенты остаются в неведении о конкретных типах объектов, которые они используют, пока объект придерживается интерфейса
  • клиенты остаются в неведении о классах, реализующих эти объекты; клиенты знают только об абстрактных классах, определяющих интерфейс

Использование интерфейса также приводит к динамическое связывание и полиморфизм, которые являются центральными особенностями объектно-ориентированного программирования.

Авторы ссылаются на наследование в качестве белая коробка повторное использование, withwhite-box относится к видимости, потому что внутренняя часть родительских классов часто видна подклассы. Напротив, авторы ссылаются на состав объекта (в котором объекты с четко определенными интерфейсами используются динамически во время выполнения объектами, получающими ссылки на другие объекты) как черный ящик повторное использование потому что в коде, использующем их, не должно быть видно внутренних деталей составных объектов.

Авторы подробно обсуждают противоречие между наследованием и инкапсуляцией и заявляют, что, по их опыту, дизайнеры злоупотребляют наследованием (Gang of Four 1995: 20). Опасность констатируется следующим образом:

"Поскольку наследование раскрывает подкласс Что касается деталей реализации его родителя, часто говорят, что «наследование нарушает инкапсуляцию» »(Gang of Four 1995: 19)

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

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

Для авторов «делегирование» - это крайняя форма композиции объектов, которую всегда можно использовать для замены наследования. Делегирование включает в себя два объекта: «отправитель» передает себя «делегату», чтобы позволить делегату обратиться к отправителю. Таким образом, связь между двумя частями системы устанавливается только во время выполнения, а не во время компиляции. В Перезвоните В статье есть дополнительная информация о делегировании.

Авторы также обсуждают так называемые параметризованные типы, которые также известны как дженерики (Ада, Эйфель, Ява, C #, VB.NET и Delphi) или шаблоны (C ++). Они позволяют определять любой тип без указания всех других используемых им типов - неуказанные типы предоставляются как «параметры» в момент использования.

Авторы признают, что делегирование и параметризация очень эффективны, но добавляют предупреждение:

«Динамическое программное обеспечение с высокой степенью параметризации труднее понять и создать, чем более статичное». (Банда четырех 1995: 21)

Далее авторы различают 'Агрегация ', где один объект' имеет 'или' является частью 'другого объекта (подразумевая, что совокупный объект и его владелец имеют одинаковые сроки жизни) и знакомство, где один объект просто «знает» другой объект. Иногда знакомство называют отношениями «ассоциации» или «использования». Объекты знакомств могут запрашивать операции друг у друга, но не несут ответственности друг за друга. Знакомство - более слабые отношения, чем агрегирование, и предполагает многое. более слабая связь между объектами, что часто бывает желательно для обеспечения максимальной ремонтопригодности проекта.

Авторы используют термин «инструментарий» там, где другие сегодня могут использовать «библиотеку классов», как в C # или Java. На их языке, инструментальные средства являются объектно-ориентированным эквивалентом библиотек подпрограмм, тогда как 'рамки 'представляет собой набор взаимодействующих классов, которые составляют повторно используемый дизайн для определенного класса программного обеспечения. Они заявляют, что Приложения сложно спроектировать, наборы инструментов сложнее, и рамки труднее всего проектировать.

Пример использования

Глава 2 представляет собой пошаговое тематическое исследование «конструкции»Что-видишь-то-то-получаешь '(или' WYSIWYG ') редактор документов под названием Lexi. "(стр. 33)

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

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

Ниже перечислены семь проблем (включая их ограничения) и их решения (включая указанные шаблоны):

Структура документа

Документ представляет собой «набор основных графических элементов», таких как символы, линии, другие формы и т. Д., Которые «фиксируют общее информационное содержание документа» (стр. 35). Структура документа содержит набор этих элементов, и каждый элемент, в свою очередь, может быть подструктурой других элементов.

Проблемы и ограничения

  1. Текст и графику следует обрабатывать одинаково (то есть графика не является производным экземпляром текста, и наоборот).
  2. Реализация должна одинаково относиться к сложным и простым структурам. Он не должен знать разницу между ними.
  3. Конкретные производные абстрактных элементов должны иметь специализированные аналитические элементы.

Решение и образец

А рекурсивная композиция представляет собой иерархическую структуру элементов, которая строит «все более сложные элементы из более простых» (стр. 36). Каждый узел в структуре знает о своих дочерних элементах и ​​своих родителях. Если операция должна выполняться для всей структуры, каждый узел вызывает операцию для своих дочерних элементов (рекурсивно).

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

Форматирование

Форматирование отличается от структуры. Форматирование - это метод построения конкретного экземпляра физической структуры документа. Это включает разбиение текста на строки, использование дефисов, регулировку ширины полей и т. Д.

Проблемы и ограничения

  1. Баланс между качеством (форматированием), скоростью и объемом памяти
  2. Сохраняйте независимость (несвязанность) форматирования от структуры документа.

Решение и образец

А Композитор class будет инкапсулировать алгоритм, используемый для форматирования композиции. Композитор - это подкласс примитивного объекта структуры документа. Compositor имеет связанный экземпляр объекта Composition. Когда Compositor запускает Составить (), он выполняет итерацию по каждому элементу связанной с ним композиции и изменяет структуру, вставляя при необходимости объекты Row и Column.

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

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

Украшение пользовательского интерфейса

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

Проблемы и ограничения

  1. Обозначьте страницу текста рамкой вокруг области редактирования
  2. Полосы прокрутки, которые позволяют пользователю просматривать различные части страницы
  3. Объекты пользовательского интерфейса не должны знать об украшениях
  4. Избегайте «взрывного роста классов», который может быть вызван выделением подклассов для «всех возможных комбинаций украшений» и элементов (стр. 44)

Решение и образец

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

Это Шаблон декоратора, который добавляет обязанности к объекту без изменения самого объекта.

Поддержка нескольких стандартов внешнего вида

Смотреть и чувствовать относится к Платформа -специфические стандарты пользовательского интерфейса. Эти стандарты «определяют правила внешнего вида приложений и их реакции на пользователя» (стр. 47).

Проблемы и ограничения

  1. Редактор должен реализовывать стандарты нескольких платформ, чтобы портативный
  2. Легко адаптироваться к новым и возникающим стандартам
  3. Разрешить изменение внешнего вида во время выполнения (т. Е .: Нет жесткое кодирование )
  4. Имейте набор абстрактных подклассов элементов для каждой категории элементов (ScrollBar, Buttons и т. Д.)
  5. Имейте набор конкретных подклассов для каждого абстрактного подкласса, который может иметь различный стандарт внешнего вида. (ScrollBar, имеющий MotifScrollBar и PresentationScrollBar для оформления Motif и Presentation)

Решение и образец

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

Это Абстрактная фабрика. Обычная фабрика создает конкретные объекты одного типа. Абстрактная фабрика создает конкретные объекты разных типов в зависимости от конкретной реализации самой фабрики. Его способность фокусироваться не только на конкретных объектах, но и на целых семьи конкретных объектов "отличает его от других моделей создания, которые включают только один вид объекта продукта" (стр. 51).

Поддержка многооконных систем

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

Проблемы и ограничения

  1. Редактор документов должен работать на многих существующих «важных и в значительной степени несовместимых оконных системах» (стр. 52)
  2. Абстрактную фабрику использовать нельзя. Из-за различных стандартов не будет общего абстрактного класса для каждого типа виджета.
  3. Не создавайте новую нестандартную оконную систему

Решение и образец

Возможно разработать «наши собственные абстрактные и конкретные классы продуктов», потому что «все оконные системы в основном делают одно и то же» (стр. 52). Каждая оконная система обеспечивает операции по рисованию примитивных форм, уменьшению / уменьшению значков, изменению размеров и обновлению содержимого окна.

Абстрактная база Окно класс может быть унаследован от различных типов существующих окон, таких как приложение, иконизированный, диалог. Эти классы будут содержать операции, связанные с окнами, такие как изменение формы, графическое обновление и т. Д. Каждое окно содержит элементы, чьи Рисовать() функции вызываются Окнособственные функции, связанные с отрисовкой.

Чтобы избежать необходимости создавать подклассы Window для каждой возможной платформы, будет использоваться интерфейс. В Окно класс будет реализовывать Окно выполнение (WindowImp) абстрактный класс. Этот класс, в свою очередь, будет производным от нескольких реализаций, зависящих от платформы, каждая из которых выполняет операции, зависящие от платформы. Следовательно, только один набор Окно классы нужны для каждого типа Окно, и только один набор WindowImp классы необходимы для каждой платформы (а не Декартово произведение всех доступных типов и платформ). Кроме того, добавление нового типа окна не требует каких-либо изменений в реализации платформы и наоборот.

Это Образец моста. Окно и WindowImp разные, но родственные. Окно занимается управлением окнами в программе, и WindowImp имеет дело с окнами на платформе. Один из них может измениться, не изменяя другой. Паттерн Мост позволяет этим двум «отдельным иерархиям классов работать вместе, даже если они развиваются независимо» (стр. 54).

Пользовательские операции

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

Проблемы и ограничения

  1. Доступ к операциям должен осуществляться через разные входы, такие как параметр меню и сочетание клавиш для одной и той же команды.
  2. У каждой опции есть интерфейс, который можно изменять.
  3. Операции реализованы в нескольких разных классах
  4. Во избежание связывания между классами реализации и пользовательского интерфейса не должно быть много зависимостей.
  5. Команды отмены и повтора должны поддерживаться в большинстве операций изменения документа, с без произвольного ограничения по количеству уровней отмены
  6. Функции нежизнеспособны, так как их нелегко отменить / повторить, их нелегко связать с состоянием и их сложно расширить или повторно использовать.
  7. К меню следует относиться как к иерархической составной структуре. Следовательно, меню - это пункт меню, который содержит пункты меню, которые могут содержать другие пункты меню и т. Д.

Решение и образец

Каждый пункт меню создается не со списком параметров, а с Команда объект.

Команда - это абстрактный объект, который имеет только один абстрактный Выполнять() метод. Производные объекты расширяют Выполнять() метод соответствующим образом (т.е. PasteCommand.Execute () будет использовать буфер обмена содержимого). Эти объекты могут использоваться виджетами или кнопками так же легко, как и элементами меню.

Для поддержки отмены и повтора Команда также дается Unexecute () и Реверсивный (). В производных классах первый содержит код, который отменяет эту команду, а второй возвращает логическое значение, которое определяет, нельзя ли выполнить команду. Реверсивный () позволяет не отменять некоторые команды, например команду «Сохранить».

Все выполнено Команды хранятся в списке с методом сохранения маркера «настоящего» сразу после последней выполненной команды. Запрос на отмену вызовет Command.Unexecute () непосредственно перед словом «присутствует», затем переместите «настоящее» на одну команду назад. И наоборот, a Повторить запрос позвонит Command.Execute () после «присутствует» и переместить вперед «настоящее».

Этот Команда подход - это реализация Шаблон команды. Он инкапсулирует запросы в объекты и использует общий интерфейс для доступа к этим запросам. Таким образом, клиент может обрабатывать разные запросы, а команды могут быть разбросаны по всему приложению.

Проверка орфографии и расстановка переносов

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

Проблемы и ограничения

  1. Разрешить несколько способов проверки орфографии и определения мест для расстановки переносов
  2. Возможность расширения для будущего анализа (например, подсчет слов, проверка грамматики)
  3. Уметь перебирать содержимое текста без доступа к фактической структуре текста (например, массив, связанный список, строка)
  4. Разрешить любой способ обхода документа (от начала до конца, от конца к началу, в алфавитном порядке и т. Д.)

Решение и образец

Удаление целочисленного индекса из базового элемента позволяет реализовать другой интерфейс итерации. Это потребует дополнительных методов для обхода и поиска объекта. Эти методы помещены в аннотацию Итератор интерфейс. Затем каждый элемент реализует вывод Итератор, в зависимости от того, как этот элемент хранит свой список (ArrayIterator, LinkListIterator, так далее.).

Функции обхода и поиска помещены в абстрактный интерфейс Iterator. Будущие итераторы могут быть получены на основе типа списка, который они будут повторять, например, массивы или связанные списки. Таким образом, независимо от того, какой метод индексации использует любая реализация элемента, у нее будет соответствующий итератор.

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

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

Вместо этого общий Проверь меня() Метод встроен в абстрактный класс элемента. Каждому итератору дается ссылка на определенный алгоритм (например, проверка орфографии, проверка грамматики и т. Д.). Когда этот итератор выполняет итерацию по своей коллекции, он вызывает каждый элемент Проверь меня, проходя указанный алгоритм. Проверь меня затем передает ссылку на свой элемент обратно в указанный алгоритм для анализа.

Таким образом, для выполнения проверки орфографии интерфейсному итератору будет дана ссылка на Проверка орфографии объект. Затем итератор будет обращаться к каждому элементу, выполняя его Проверь меня() метод с Проверка орфографии параметр. Каждый Проверь меня затем позвонил бы Проверка орфографии, передавая ссылку на соответствующий элемент.

Таким образом, любой алгоритм может использоваться с любым методом обхода без жесткой привязки одного кода к другому. Например, Find может использоваться как «найти следующий» или «найти предыдущий», в зависимости от того, использовался ли итератор «вперед» или итератор «назад».

Кроме того, сами алгоритмы могут отвечать за работу с разными элементами. Например, Проверка орфографии алгоритм проигнорирует Графический элемент, вместо того, чтобы программировать каждый Графический-производный элемент, чтобы не отправлять себя в Проверка орфографии.

Выкройки по типу

Творческий

Шаблоны создания - это те, которые создают объекты, а не создают экземпляры объектов напрямую. Это дает программе больше гибкости в выборе объектов, которые необходимо создать для данного случая.

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

Структурные

Они касаются композиции классов и объектов. Они используют наследование для создания интерфейсов и определения способов компоновки объектов для получения новых функций.

  • Адаптер позволяет классам с несовместимыми интерфейсами работать вместе, оборачивая свой собственный интерфейс вокруг интерфейса уже существующего класса.
  • Мост отделяет абстракцию от ее реализации, так что они могут различаться независимо.
  • Композитный составляет ноль или более похожих объектов, так что ими можно управлять как одним объектом.
  • Декоратор динамически добавляет / переопределяет поведение в существующем методе объекта.
  • Фасад предоставляет упрощенный интерфейс для большого объема кода.
  • Наилегчайший вес снижает стоимость создания и манипулирования большим количеством похожих объектов.
  • Прокси предоставляет заполнитель для другого объекта для управления доступом, снижения затрат и упрощения.

Поведенческий

Большинство этих шаблонов проектирования специально связаны с коммуникацией между объекты.

  • Цепочка ответственности делегирует команды цепочке обрабатывающих объектов.
  • Команда создает объекты, которые инкапсулируют действия и параметры.
  • Устный переводчик реализует специализированный язык.
  • Итератор последовательно обращается к элементам объекта, не раскрывая его базовое представление.
  • Посредник позволяет Слабая связь между классами, будучи единственным классом, который подробно знает свои методы.
  • Memento предоставляет возможность восстановить объект в его предыдущее состояние (отменить).
  • Наблюдатель - это шаблон публикации / подписки, который позволяет нескольким объектам-наблюдателям видеть событие.
  • Состояние позволяет объекту изменять свое поведение при изменении его внутреннего состояния.
  • Стратегия позволяет выбирать один из семейств алгоритмов на лету во время выполнения.
  • Шаблонный метод определяет каркас алгоритма как абстрактный класс, позволяя его подклассам обеспечивать конкретное поведение.
  • Посетитель отделяет алгоритм от структуры объекта, перемещая иерархию методов в один объект.

Критика

Критика была направлена ​​на концепцию шаблоны проектирования программного обеспечения как правило, и в Шаблоны проектирования конкретно. Основная критика Шаблоны проектирования заключается в том, что его шаблоны - это просто обходные пути для недостающих функций в C ++, заменяющие элегантные абстрактные функции длинными конкретными шаблонами, по сути становясь «человеческим компилятором» или «генерируя вручную расширения некоторых макросов».[4] Питер Норвиг показывает, что 16 из 23 паттернов в Шаблоны проектирования упрощены или устранены (через прямую языковую поддержку) в Лисп или же Дилан.[5] Соответствующие наблюдения были сделаны Ханнеманом и Kiczales кто реализовал несколько из 23 шаблонов проектирования, используя аспектно-ориентированный язык программирования (AspectJ ) и показал, что зависимости на уровне кода были удалены из реализаций 17 из 23 шаблонов проектирования и что аспектно-ориентированное программирование может упростить реализацию шаблонов проектирования.[6]

Пол Грэм написал:[4]

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

Также была юмористическая критика, такая как показательный процесс на OOPSLA '99 3 ноября 1999 г.[7][8][а] и пародия на формат, автор Джим Коплиен, озаглавленный "Канзас-Сити кондиционер ".

В интервью InformIT в 2009 году Эрих Гамма заявил, что авторы книги обсуждали в 2005 году, как они реорганизовали бы книгу, и пришли к выводу, что они бы перекатегоризовали некоторые шаблоны и добавили несколько дополнительных. Гамма хотела убрать шаблон синглтона, но авторы не достигли консенсуса по этому поводу.[9]

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

Примечания

  1. ^ Председательствующий магистрат Нил Харрисон, главный прокурор Кент Бек, Адвокат защиты Мартин Фаулер, Суд Балифф Брайан Фут; Ричард Хелм представил признание, а остальные предстали перед судом.

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

  1. ^ Банда из четырех, Вики по созданию контента для людей, проекты и шаблоны в разработке программного обеспечения.
  2. ^ Ричард Хелм
  3. ^ SIGPLAN FY '05 Годовой отчет
  4. ^ а б Грэм, Пол (2002). Месть ботаников. Получено 2012-08-11.
  5. ^ Норвиг, Питер (1998). Шаблоны проектирования на динамических языках.
  6. ^ Ханнеманн, Ян (2002). Реализация шаблона проектирования в Java и AspectJ.
  7. ^ Обвинительный акт
  8. ^ Показательный процесс над бандой четырех, Брайан Фут
  9. ^ Гамма, Эрих; Хелм, Ричард; Джонсон, Ральф (2009-10-22). «Паттерны дизайна 15 лет спустя: интервью с Эрихом Гаммой, Ричардом Хелмом и Ральфом Джонсоном». InformIT (Опрос). Беседовал Ларри О'Брайен. В архиве с оригинала на 20.02.2019. Получено 2019-09-01.