Декомпозиция подсистем на модули Известны два типа моделей модульной декомпозиции: модель потока данных; модель объектов. В основе модели потока данных лежит разбиение по функциям. Модель объектов основана на слабо сцепленных сущностях, имеющих собственные наборы данных, состояния и наборы операций. Очевидно, что выбор типа декомпозиции должен определяться сложностью разбиваемой подсистемы.
Модульность
Свойства, обоснование
Модуль – фрагмент программного текста, являющийся строительным блоком для физической структуры системы. Как правило, модуль состоит из интерфейсной части и части – реализации. Модульность – свойство системы, которая может подвергаться декомпозиции на ряд внутренне связных и слабо зависящих друг от друга модулей. По определению Г.Майерса, модульность – свойство ПО, обеспечивающее интеллектуальную возможность создания сколь угодно сложной программы. Проиллюстрируем эту точку зрения.
Пусть - функция сложности решения проблемы -функция затрат времени на решение проблемы . Для двух проблем и из соотношения следует, что Этот вывод интуитивно ясен: решение сложной проблемы требует большего времени. Далее. Из практики решения проблем человеком следует:
Это соотношение и есть обоснование модульности. Оно приводит к заключению «разделяй и властвуй» - сложную проблему легче решить, разделив ее на управляемые части. Фактически, это аргумент в пользу модульности. Однако здесь отражена лишь часть реальности, ведь здесь не учитываются затраты на межмодульный интерфейс. Как показано на рис. 15, с увеличением количества модулей (и уменьшением их размера) эти затраты также растут.
Рис. 15.Затраты на модульность
Таким образом, существует оптимальное количество модулей Opt, которое приводит к минимальной стоимости разработки. Нет необходимого опыта для гарантированного предсказания Opt. Впрочем, разработчики знают, что оптимальный модуль должен удовлетворять двум критериям:
· снаружи он проще, чем внутри;
· его проще использовать, чем построить.
Информационная закрытость
Принцип информационной закрытости (автор – Д. Парнас, 1972) утверждает: содержание модулей должно быть скрыто друг от друга . Как показано на рис. 16, модуль должен определяться и проектироваться так, чтобы его содержимое (процедуры и данные) было недоступно тем модулям, которые не нуждаются в такой информации (клиентам).
Рис. 16.Информационная закрытость модуля
Информационная закрытость означает следующее:
- все модули независимы, обмениваются только информацией, необходимой для работы;
- доступ к операциям и структурам данных модуля ограничен.
Достоинства информационной закрытости:
· обеспечивается возможность разработки модулей различными, независимыми коллективами;
· обеспечивается легкая модификация системы (вероятность распространения ошибок очень мала, так как большинство данных и процедур скрыто от других частей системы).
Идеальный модуль играет роль «черного ящика», содержимое которого невидимо клиентам. Он прост в использовании – количество «ручек и органов управления» им невелико (аналогия с эксплуатацией телевизора). Его легко развивать и корректировать в процессе сопровождения программной системы. Для обеспечения таких возможностей система внутренних и внешних связей модуля должна отвечать особым требованиям. Обсудим характеристики внутренних и внешних связей модуля.
Связность модуля
Связность модуля (Cohesion) – это мера зависимости его частей. Связность – внутренняя характеристика модуля. Чем выше связность модуля, тем лучше результат проектирования,то есть тем «черней» его ящик (капсула, защитная оболочка модуля), тем меньше «ручек управления» на нем находится и тем проще эти «ручки». Для измерения связности используют понятие силы связности (СС). Существует 7 типов связности:
- Связность по совпадению(СС=0). В модуле отсутствуют явно выраженные внутренние связи.
- Логическая связность(СС=1). Части модуля объединены по принципу функционального подобия. Например, модуль состоит из разных подпрограмм обработки ошибок. При использовании такого модуля клиент выбирает только одну из подпрограмм. Недостатки: сложное сопряжение; большая вероятность внесения ошибок при изменении сопряжения ради одной из функций.
- Временная связность(СС=3). Части модуля не связанны, но необходимы в один и тот же период работы системы. Недостаток: сильная взаимная связь с другими модулями, отсюда сильная чувствительность к внесению изменений.
- Процедурная связность(СС=5). Части модуля связаны порядком выполняемых ими действий, реализующих некоторый сценарий поведения.
- Коммуникативная связность(СС=7). Части модуля связанны по данным (работают с одной и той же структурой данных).
- Информационная (последовательная) связность(СС=9). Выходные данные одной части используются как входные данные в другой части модуля.
- Функциональная связность(СС=10).части модуля вместе реализуют одну функцию.
Отметим, что типы связности 1, 2, 3 – результат неправильного планирования архитектуры, а тип связности 4 – результат небрежного планирования архитектуры приложения. Общая характеристика типов связности представлена в табл. 4.1.
Таблица 4.1.Характеристика связности модуля
Тип связности
| Сопровождаемость
| Роль модуля
| Функциональная
| Лучшая сопровождаемость
| «Черный ящик»
| Информационная
(последовательная)
| Не совсем «черный ящик»
| Коммуникативная
| «Серый ящик»
| Процедурная
| Худшая сопровождаемость
| «Белый» или «просвечивающий ящик»
| Временная
| Логическая
| «Белый ящик»
| По совпадению
|
Функциональная связность
Функционально связный модуль содержит элементы, участвующие в выполнении одной и только одной проблемной задачи. Примеры функционально связных модулей:
- Вычислять синус угла;
- Проверять орфографию;
- Читать запись файла;
- Вычислять координаты цели;
- Вычислять зарплату сотрудника;
- Определять место пассажира.
Каждый из этих модулей имеет единичное значение. Когда клиент вызывает модуль, выполнятся только одна работа, без привлечения внешних обработчиков. Например, модуль «Определять место пассажира» должен делать только это; он не должен распечатывать заголовки страницы.
Некоторые из функционально связных модулей очень просты (например, Вычислять синус угла или Читать запись файла), другие сложны (например, Вычислять координаты цели). Модуль Вычислять синус угла, очевидно, реализует единичную функцию, но как может модуль Вычислять зарплату сотрудника выполнять только одно действие? Ведь каждый знает, что приходится определять начисленную сумму, вычеты по рассрочкам, подоходный налог, социальный налог, алименты и т.д.! Дело в том, что, несмотря на сложность модуля и на то, что его обязанность исполняют несколько подфункций, если его действия можно представить как единую проблемную функцию (с точки зрения клиента), тогда считают, что модуль функционально связен.
Приложения, построенные из функционально связных модулей, легче всего сопровождать. Соблазнительно думать, что любой модуль можно рассматривать как одно-функциональный, но не надо заблуждаться. Существует много разновидностей модулей, которые выполняют для клиентов перечень различных работ, и этот перечень нельзя рассматривать как единую проблемную функцию. Критерий при определении уровня связности этих нефункциональных модулей – как связаны друг с другом различные действия, которые они исполняют.
Информационная связность
При информационной (последовательной) связности элементы – обработчики модуля образуют конвейер для обработки данных – результаты одного обработчика используются как исходные данные для следующего обработчика. Приведем пример:
Модуль Прием и проверка записи
прочитать запись из файла
проверить контрольные данные в записи
удалить контрольные поля в записи
вернуть обработанную запись
Конец модуля
В этом модуле 3 элемента. Результаты первого элемента (прочитать запись из файла) используются как входные данные для второго элемента (проверить контрольные данные в записи) и т.д. Сопровождать модули с информационной связностью почти так же легко, как и функционально связные модули. Правда, возможности повторного использования здесь ниже, чем в случае функциональной связности. Причина – совместное применение действий модуля с информационной связностью полезно далеко не всегда.
Коммуникативная связность
При коммуникативной связности элементы – обработчики модуля используют одни и те же данные, например внешние данные. Пример коммуникативно- связного модуля:
Модуль Отчет и средняя зарплата
используется Таблица зарплаты служащих
сгенерировать Отчет по зарплате
вычислить параметр Средняя зарплата
вернуть Отчет по зарплате. Средняя зарплата
Конец модуля
Здесь все элементы модуля работают со структурой «Таблица зарплаты служащих». С точки зрения клиента проблема применения коммуникативно связного модуля состоит в избыточности получаемых результатов. Например, клиенту требуется только отчет по зарплате, он не нуждается в значении средней зарплате. Такой клиент будет вынужден выполнять избыточную работу – выделение в полученных данных материала отчета. Почти всегда разбиение коммуникативно связного модуля на отдельные функционально связные модули улучшает сопровождаемость системы.
Попытаемся провести аналогию между информационной и коммуникативной связью.
Модули с коммуникативной и информационной связью подобны в том, что содержат элементы, связанные по данным. Их удобно использовать, потому что лишь не многие элементы в этих модулях связаны с внешней средой. Главное различие между ними – информационно связной модуль работает подобно сборочной линии; его обработчики действуют в определенном порядке; в коммуникативно связном модуле порядок выполнения действий безразличен. В нашем примере не имеет значения, когда генерирует отчет (до, после или одновременно с вычислением средней зарплаты).
Процедурная связность
При достижении процедурной связности мы попадаем в пограничную область между хорошей сопровождаемостью (для модулей с более высокими уровнями связности) и плохой сопровождаемостью (для модулей с более низкими уровнями связности). Процедурно связный модуль состоит из элементов, реализующих независимые действия, для которых задан порядок работы, то есть порядок передачи управления. Зависимости по данным между элементами нет. Например:
Модуль Вычисление средних значений
используется Таблица-А, Таблица-В
вычислить среднее по Таблице-А
вычислить среднее по Таблице-В
вернуть среднееТабл-А, среднееТабл-В
Конец модуля
Этот модуль вычисляет среднее значение для двух полностью несвязных таблиц Таблица-А и Таблица-В, каждая из которых имеет по 300 элементов.
Теперь представим себе программиста, которому поручили реализовать данный модуль. Соблазнившись возможностью минимизации кода (использовать один цикл в интересах двух обработчиков, ведь они находятся в нутрии единого модуля!), программист пишет:
Модуль Вычисление средних значений
используется Таблица-А, Таблица-В
суммаТабл-А := 0
суммаТабл-В := 0
для i := 1 до 300
суммаТабл-А := суммаТабл-А + Таблица-А(i)
суммаТабл-В := суммаТабл-В + Таблица-В(i)
конец для
среднееТабл-А := суммаТабл-А / 300
среднееТабл-В := суммаТабл-В / 300
вернуть среднееТабл-А, среднееТабл-В
Конец модуля
Для процедурной связности этот случай типичен – независимый (на уровне проблемы) код стал зависимым (на уровне реализации). Прошли годы, продукт сдали заказчику. И вдруг возникла задача сопровождения – модифицировать модуль под уменьшение размера таблицы В. Оцените, насколько удобно ее решать.
Временная связность
При связности по времени элементы-обработчики модуля привязаны к конкретному периоду времени (из жизни программной системы).
Классическим примером временной связности является модуль инициализации. Элементы данного модуля почти не связаны друг с другом (за исключением того, что должны выполняться в определенное время). Они все – часть программы запуска системы. Зато элементы более тесно взаимодействуют с другими модулями, что приводит к сложным внешним связям.
Модуль со связностью по времени испытывает те же трудности, что и процедурно связный модуль. Программист соблазняется возможностью совместного использования кода (действиями, которые связаны только по времени), модуль становиться трудно, использовать повторно.
Процедурно связные модули и модули с временной связностью очень похожи. Степень их непрозрачности изменяется от темного серого до светло-серого цвета, так как трудно объявить функцию такого модуля без перечисления ее внутренних деталей. Различие между ними подобно различию между информационной и коммуникативной связностью. Порядок выполнения действий более важен в процедурно связных модулях. Кроме того, процедурные модули имеют тенденцию к совместному использованию циклов и ветвлений, а модули с временной связностью чаще содержат более линейных код.
Логическая связность
Элементы логически связного модуля принадлежат к действиям одной категории, и из этой категории клиент выбирает выполняемое действие. Рассмотрим следующий пример:
Модуль Пересылка сообщения
переслать по электронной почте
переслать по факсу
послать в телеконференцию
переслать по ftp-протоколу
Конец модуля
Как видим, логически связный модуль – мешок доступных действий. Действия вынуждены совместно использовать один и тот же интерфейс модуля. В строке вызова модуля значение каждого параметра зависит от используемого действия. При вызове отдельных действий некоторые параметры должны иметь значение пробела, нулевые значения и т.д. (хотя клиент все же должен использовать их и знать их типы).
Действия в логически связном модуле попадают в одну категорию, хотя имеют не только сходства, но и различия. К сожалению, это заставляет программиста «завязывать код действий в узел», ориентируясь на то, что действия совместно используют общие строки кода. Поэтому логически связный модуль имеет:
• Уродливый внешний вид с различными параметрами, обеспечивающими, например, четыре вида доступа;
• Запутанную внутреннюю структуру с множеством переходов, похожую на волшебный лабиринт.
В итоге модуль становится сложным как для понимания, так и для сопровождения.
Связность по совпадению
Элементы связного по совпадению модуля вообще не имеют никаких отношений друг с другом:
Модуль Разные функции (какие-то параметры)
поздравить с Новым годом (…)
проверить исправность аппаратуры (…)
заполнить анкету героя (…)
измерить температуру (…)
вывести собаку на прогулку (…)
запастись продуктами (…)
Конец модуля
Связный по совпадению модуль похож на логически связный модуль. Его элементы-действия не связны ни потоком данных, ни потоком управления. Но в логически связном модуле действия, по крайней мере, относятся к одной категории; в связном по совпадению модуле даже это не так. Словом, связные по совпадению модули имеют все недостатки логически связных модулей и даже усиливают их. Применение таких модулей вселяет ужас, поскольку один параметр используется для разных целей.
Чтобы клиент мог воспользоваться модулем Разные функции, этот модуль (подобно всем связным по совпадению модулям) должен быть «белым ящиком», чья реализация полностью видима. Такие модули делают системы менее понятными и труднее сопровождаемыми, чем системы без модульности вообще!
К счастью, связность по совпадению встречается редко. Среди ее причин можно назвать: бездумный перевод существующего монолитного кода в модули; необоснованные изменения модулей с плохой (обычно временной) связностью, приводящие к добавлению флажков.
|