Научная статья на тему 'State Machine новый паттерн объектно-ориентированного проектирования'

State Machine новый паттерн объектно-ориентированного проектирования Текст научной статьи по специальности «Компьютерные и информационные науки»

CC BY
1437
117
i Надоели баннеры? Вы всегда можете отключить рекламу.

Аннотация научной статьи по компьютерным и информационным наукам, автор научной работы — Шамгунов Никита Назимович, Корнеев Георгий Александрович, Шалыто Анатолий Абрамович

В статье предлагается новый паттерн объектно-ориентированного проектирования, названный State Machine. Этот паттерн расширяет возможности паттерна State, предназначенного для реализации объектов, поведение которых зависит от их состояния. Также предложено использовать события для уведомления об изменении состояния. Это позволяет проектировать объекты такого рода из независимых друг от друга классов. Предлагаемый паттерн по сравнению с паттерном State лучше приспособлен для повторного использования входящих в него классов.

i Надоели баннеры? Вы всегда можете отключить рекламу.

Похожие темы научных работ по компьютерным и информационным наукам , автор научной работы — Шамгунов Никита Назимович, Корнеев Георгий Александрович, Шалыто Анатолий Абрамович

iНе можете найти то, что вам нужно? Попробуйте сервис подбора литературы.
i Надоели баннеры? Вы всегда можете отключить рекламу.

State Machine a new design pattern

This paper presents a new pattern for object-oriented design State Machine. This pattern extends capabilities of State design pattern. These patterns allow an object to alter its behavior when its internal state changes. Introduced event-driven approach allows to decrease coupling. Thus automaton could be constructed from independent state classes. The classes designed with State Machine pattern are more reusable than State pattern.

Текст научной работы на тему «State Machine новый паттерн объектно-ориентированного проектирования»

УДК 681.3.06

STATE MACHINE - НОВЫЙ ПАТТЕРН ОБЪЕКТНО - ОРИЕНТИРОВАННОГО

ПРОЕКТИРОВАНИЯ

Н. Н. Шамгунов,

аспирант

Г. А. Корнеев,

аспирант

А. А. Шалыто,

доктор техн. наук, профессор Санкт-Петербургский государственный университет информационных технологий, механики и оптики

В статье предлагается новый паттерн объектно-ориентированного проектирования, названный State Machine. Этот паттерн расширяет возможности паттерна State, предназначенного для реализации объектов, поведение которых зависит от их состояния. Также предложено использовать события для уведомления об изменении состояния. Это позволяет проектировать объекты такого рода из независимых друг от друга классов. Предлагаемый паттерн по сравнению с паттерном State лучше приспособлен для повторного использования входящих в него классов.

This paper presents a new pattern for object-oriented design — State Machine. This pattern extends capabilities of State design pattern. These patterns allow an object to alter its behavior when its internal state changes. Introduced event-driven approach allows to decrease coupling. Thus automaton could be constructed from independent state classes. The classes designed with State Machine pattern are more reusable than State pattern.

Введение

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

Различные модели детерминированных конечных автоматов, которые обычно называют «конечными автоматами» или просто «автоматами», были разработаны в середине 50-х годов прошлого века [2-5]. При этом если первые две из этих работ были посвящены синтезу схем (аппаратуры), то остальные носили более абстрактный характер. Считается, что итог этапу становления теории автоматов был подведен выпуском сборника статей [6], который, в частности, содержал работы [3, 5]. Интересно отметить, что перевод этого сборника на русский язык появился в СССР в том же году. Недетерминированные автоматы и их эквивалентность детерминированным автоматам были рассмотрены в работе [7].

В дальнейшем применительно к построению аппаратуры теория автоматов «распалась» на две вза-

имосвязанные теории: абстрактную и структурную [8], для первой из которых характерна последовательная обработка информации, а для второй -параллельная.

В программировании автоматы начали применяться после появления работы [9], в которой были введены регулярные выражения и приведено доказательство их эквивалентности конечным автоматам. При этом абстрактная теория автоматов используется в основном для обработки текстов [10-12], а структурная - для программного управления [13].

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

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

В объектно-ориентированном программировании под поведением объекта обычно понимается функциональность его методов (действий). Однако во многих приложениях такого определения понятия «поведение объекта» недостаточно, так как необходимо учитывать и его внутреннее состояние.

Наиболее известной реализацией объекта, изменяющего поведение в зависимости от состояния, является паттерн State [14]. В указанной работе данный паттерн недостаточно полно специфицирован, поэтому в разных источниках, например в работах [15, 16], он реализуется по-разному (в частности, громоздко и малопонятно). По этой причине многие программисты считают, что паттерн State не предназначен для реализации автоматов. Другой недостаток паттерна State состоит в том, что разделение реализации состояний по различным классам приводит и к распределению логики переходов по ним, что усложняет понимание программы. При этом не обеспечивается независимость классов, реализующих состояния, друг от друга. Таким образом, создание иерархии классов состояний и их повторное использование затруднено. Несмотря на эти недостатки, паттерн State достаточно широко применяется в программировании, например, при реализации синхронизации с источником данных в библиотеке Java Data Objects (JDO) [17].

Заметим, что проблемы с паттерном State возникают не только при его описании, но даже в определении, в том числе, из-за неправильного перевода. Так, в работе [18] приведено следующее определение: «шаблон State - состояние объекта контролирует его поведение», в то время как более правильным было бы следующее: «шаблон State - состояния объекта управляют его поведением». Эти определения имеют разный смысл, так как с английского языка слово control переводится как управление; в русском языке управление -это действие на объект, а контроль - мониторинг и проверка выполняемых действий на соответствие заданному поведению. Не случайно часто говорят «система управления и контроля».

Кроме работы [14], паттерн State, как отмечалось выше, описывается в работах [15, 16]. В них авторы рассматривают реализацию графов переходов автоматов с помощью этого паттерна. В работе [15] код реализации графа переходов с двумя вершинами занимает более десяти страниц текста. Такую же странную ситуацию можно видеть и в работе [16]. Автор этой книги легко справился с описанием и примерами к сорока шести паттернам,

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

В настоящей работе предлагается новый паттерн, объединяющий достоинства реализации автоматов в SWITCH-технологии [13] (централизация логики переходов) и паттерна State (локализация кода, зависящего от состояния в отдельных классах).

Новый паттерн назван State Machine. Обратим внимание, что в работе [19] уже был предложен паттерн с аналогичным названием, предназначенный для программирования параллельных систем реального времени на языке Ada 95. Тем не менее, авторы выбрали именно это название, как наиболее точно отражающее суть предлагаемого паттерна.

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

о друге. При этом реализация логики переходов может осуществляться различными способами, например, с использованием таблицы переходов или оператора выбора (оператор switch в языке C++).

Более двадцати методов реализации объектов, изменяющих поведение в зависимости от состояния, рассмотрены в работе [20]. Паттерн State Machine мог бы продолжить этот список. Наиболее близким к новому паттерну является объединение паттернов State и Observer, рассмотренное в работе [21]. Однако предложенный в этой работе подход достаточно сложен, так как добавляет новый уровень абстракции — класс ChangeManager. В паттерне State Machine используется более простая модель событий, не привлекающая относительно тяжелую реализацию паттерна Observer. В работе [22] предложена реализация паттерна State, позволяющая создавать иерархии классов состояний. Зависимость между классами состояний снижается за счет того, что переход в новое состояние осуществляется по его имени. Такая реализация, тем не менее, не снимает семантической зависимости между классами состояний.

Описание паттерна

В названиях разделов будем придерживаться соглашений, введенных в работе [14].

Назначение. Паттерн State Machine предназначен для создания объектов, поведение которых варьируется в зависимости от состояния. При этом у клиента создается впечатление, что изменился класс объекта. Таким образом, назначение предлагаемого паттерна фактически совпадает с таковым для паттерна State [14], однако ниже будет показано, что область применимости последнего более узка.

Отметим, что в определении имеются в виду так называемые управляющие, а не вычислительные состояния [23]. Их различие может быть проиллюстрировано на следующем примере. При создании вычислительной системы для банка имеет смысл выделить режимы нормальной работы и банкротства в разные управляющие состояния, так как в этих режимах поведение банка может существенно отличаться. В то же время, конкретные суммы денег в банковском балансе будут представлять вычислительное состояние.

Мотивация. Предположим, что требуется спроектировать класс Connection, представляющий сетевое соединение. Простейшее сетевое соединение имеет два управляющих состояния: соединено и разъединено. Переход между этими состояниями происходит или при возникновении ошибки или посредством вызовов методов установить соединение (connect) и разорвать соединение (disconnect). В состоянии соединено может производиться получение (метод receive) и отправка (метод send) данных по соединению. В случае возникновения ошибки при передаче данных генерируется исключительная ситуация(IOException) и сетевое соединение разъединяется. В состоянии разъединено прием и отправка данных невозможны. При попытке осуществить передачу данных в этом состоянии объект также генерирует исключительную ситуацию.

Таким образом, интерфейс, который необходимо реализовать в классе Connection, должен выглядеть следующим образом (здесь и далее примеры приводятся на языке Java):

package connection;

import java.io.IOException;

public interface IConnection {

void connect() throws IOException; void disconnectO throws IOException; int receiveO throws IOException; void send(int value) throws IOException;

}

Основная идея паттерна State Machine заключается в разделении классов, реализующих логику переходов (контекст), конкретных состояний и модели данных. Для осуществления взаимодействия конкретных состояний с контекстом используются события, представляющие собой объекты, передаваемые состояниями контексту. Отличие от паттерна State состоит в методе определения следующего состояния при осуществлении перехода. Если в паттерне State следующее состояние указывается текущим состоянием, то в предлагаемом паттерне это выполняется путем уведомления класса контекста о наступлении события. После этого, в зависимости от события и текущего состояния, контекст устанавливает следующее состояние в соответствии с графом переходов.

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

Отметим, что графы переходов, применяемые для описания логики переходов при проектировании с использованием паттерна State Machine, отличаются от графов переходов, рассмотренных в работах [11, 13, 24]. Применяемые графы переходов состоят только из состояний и переходов, помеченных событиями. Переход из текущего состояния S в следующее состояние S осуществляется по событию Е, если на графе переходов существует дуга из S в S\ помеченная событием Е. При этом из одного состояния не могут выходить две дуги, помеченные одним и тем же событием. Отметим, что на графе переходов не отображаются ни методы, входящие в интерфейс реализуемого объекта, ни условия порождения событий.

DISCONNECT

■ Рис. 1. Граф переходов для класса Connection

Граф переходов для описания поведения класса Connection приведен на рис. 1. В нем классы, реализующие состояния соединено и разъединено, называются, соответственно, Connectedstate и DisconnectedState, тогда как событие CONNECT обозначает установление связи, disconnect -обрыв связи, a error - ошибку передачи данных.

Рассмотрим в качестве примера обработку разрыва соединения при ошибке передачи данных. При реализации с использованием паттерна state состояние Connectedstate укажет контексту, что следует перейти в состояние DisconnectedState. В случае же паттерна State Machine контекст будет уведомлен о наступлении события ERROR И ОСуЩвСТВИТ ПвреХОД В СОСТОЯНИв DisconnectedState. Таким образом, классы Connectedstate И DisconnectedStäte не «знают» о существовании друг друга.

Применимость. Паттерн State Machine может быть использован в следующих случаях.

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

2. Рефакторинг кода [25], зависящего от состояния объекта. Примером может служить рефакторинг кода, проверяющего права доступа к тем или иным функциям программного обеспечения в зависимости от текущего пользователя или приобретенной лицензии.

3. Повторное использование классов, входящих в паттерн, в том числе посредством создания иерархии классов состояний. Пример такого случая будет рассмотрен ниже.

4. Эмуляции абстрактных и структурных автоматов.

Таким образом, область применимости паттерна State Machine шире, чем у паттерна State.

Структура. На рис. 2 изображена структура паттерна State Machine.

Здесь IAutomatonlnterfасе - интерфейс реализуемого объекта, a operationl, operation2, ... - его методы. Этот интерфейс реализуется основным классом Context и классами состояний ConcreteStatel, ConcreteState2, ... . Для смены состояния объекта используются события eventl_l, event 2_1,..., event 2_1, event2_2, ..., являющиеся объектами класса Event. Класс Context содержит ссылки на все состояния объекта(ConcreteStatel и ConcreteState2), а также на текущее состояние (state). В свою оче-

«interface»

lEventSink

+castEvent(in event: Event)

Context

-state : I Automaton Interface

+ope ration 10 +operation2()

+...()

+castEvent(in event : Event)

<► <►

«interface»

lAutomatonlnterface

+operation1 Q +operation2() +-()___________

Event

ConcreteStatel

-event1_1 : Event -event1_2 : Event -... : Event ^-eventSink : lEventSink

■automaton : lAutomatonlnterface

+operation1Q

+operation2()

±J1___________

■o

/ / / / / / / \ \ \ \ \ \ \ \ \ \ \ \ \ \

ConcreteState2

-event2 1 : Event

-event2 2 : Event

-... : Event

-eventSink : iEventSink

-automaton : lAutomatonlnterface

+operation1Q

+operation2()

+...()

DataModel

О

-... : Event -eventSink : IEventSink

+operation1 +operation2 +...() I

Рис. 2. Структура паттерна Sate Machine

редь, классы состояний содержат ссылку на модель данных (dataModel) и интерфейс уведомления о событиях (eventSink). Для того чтобы не загромождать рисунок, на нем не отражены связи классов состояний и класса Event.

Классы Context, ConcreteStatel,

С о п с г е t е S t a t е 2 , ... реализуют интерфейс lAutomatonlnterface. Класс Context содержит переменные типа lAutomatonlnterface. Одна из них - текущее состояние автомата, а остальные хранят ссылки на классы состояний автомата. Отметим, что стрелки, соответствующие ссылкам на классы состояний, ведут к интерфейсу, а не к этим классам. Это следствие того, что все взаимодействие между контекстом и классами состояний производится через интерфейс автомата. Связи между контекстом и классами состояний отмечены стрелками с ромбом (используется агрегация).

Участники. Паттерн State Machine состоит из следующих частей.

1. Интерфейс автомата (lAutomatonlnterface) -реализуется контекстом и является единственным способом взаимодействия клиента с автоматом. Этот же интерфейс реализуется классами состояний.

2. Контекст (Context) - класс, в котором инкапсулирована логика переходов. Он реализует интерфейс автомата, хранит экземпляры модели данных и текущего состояния.

3. Классы состояний (ConcreteStatel, ConcreteState2, ...) - определяют поведение в конкретном состоянии. Реализуют интерфейс автомата.

4. События (eventl_l, eventl__2, ...) - инициируются состояниями и передаются контексту, который осуществляет переход в соответствии с текущим состоянием и событием.

5. Интерфейс уведомления о событиях (IEventSink) - реализуется контекстом и является единственным способом взаимодействия объектов состояний с контекстом.

6. Модель данных (DataModel) - класс предназначен для хранения и обмена данными между состояниями.

Отметим, что в предлагаемом паттерне интерфейс автомата реализуется как контекстом, так и классами состояний. Это позволяет добиться проверки целостности еще на этапе компиляции. В паттерне State такая проверка невозможна из-за различия интерфейсов контекста и классов состояний.

iНе можете найти то, что вам нужно? Попробуйте сервис подбора литературы.

Отношения. При инициализации контекст создает экземпляр модели данных и использует его при конструировании экземпляров состояний. Кроме модели данных в конструктор класса состояния также передается интерфейс уведомления

о событии (ссылка на контекст).

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

рировать событие - уведомить об этом контекст по интерфейсу уведомления о событиях.

Решение о смене состояния принимает контекст на основе события, пришедшего от конкретного объекта состояния.

Результаты. Сформулируем результаты, получаемые при использовании паттерна State Machine.

1. Также как и в паттерне State, поведение, зависящее от состояния, локализовано в отдельных классах состояний.

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

о разрыве соединения).

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

4. Логика переходов может быть реализована табличным способом, что повышает скорость смены состояний.

5. Паттерн State Machine предоставляет «чистый» (без лишних методов) интерфейс для пользователя. Для того чтобы клиенты не имели доступа К интерфейсу IEventSink, реализуемому классом контекста, следует использовать закрытое (private) наследование (например, в языке C++) или определить закрытый конструктор и статический метод, создающий экземпляр контекста, который возвращает интерфейс автомата. Соответствующий фрагмент кода имеет вид:

class Automaton {

private Automaton() {}

public static IAutomatalnterface CreateAutomaton() {

return new Automaton();

}

}

6. В отличие от паттерна State паттерн State Machine не содержит дублирующих интерфейсов для контекста и классов состояний.

7. Возможно повторное использование классов состояний, в том числе посредством создания их иерархии. В работе [ 14] сказано, что «поскольку зависящий от состояния код целиком находится в одном из подклассов класса state, то добавлять новые состояния и переходы можно просто путем порождения новых подклассов». На самом же деле, добавление нового состояния зачастую влечет за собой модификацию остальных классов состояний, так как иначе переход в данное состояние не может быть осуществлен. Таким образом, расширение автомата, построенного на основе паттерна State, является проблематичным. Более того, при реализации наследования в паттерне State также затруднено и расширение интерфейса автомата. Скорее всего, именно по этим причинам в работе [14] не описано наследование автоматов.

■ Рис. 3. Реализация методов, не зависящих от состояния

8. В работе [26] рассматривается задача реализации объектов, часть методов которых не зависит от состояния. Для решения этой задачи предложен паттерн Three Level FSM. Данная задача может быть также решена и при помощи паттерна State Machine, для чего следует разделить интерфейс реализуемого объекта и интерфейс автомата. При этом последний реализуется контекстом в соответствии с описываемым паттерном. Для реализации полного интерфейса объекта создается наследник контекста, в котором определяются методы, не зависящие от состояния (рис. 3).

Реализация. Рассмотрим возможные модификации паттерна State Machine.

1. Хранение модели данных. Контекст можно реализовать таким образом, чтобы он включал в себя модель данных, как предлагается в паттерне State. Тогда в конструктор объекта состояния передается только один параметр. Однако при таком подходе зависимость реализации состояний от контекста увеличивается, что усложняет повторное использование классов состояний.

2. Stateless и stateful классы состояний. В паттерне State Machine контекст и классы состояний реализуют интерфейс автомата. Это достигается за счет того, что указанные классы содержат ссылки на модель данных и интерфейс уведомления о событиях. Таким образом, классы состояний являются stateful (зависят от предыстории). Такой подход не всегда приемлем, поскольку в некоторых ситуациях критичен расход памяти. Паттерн State Machine можно видоизменить так, чтобы состояния были stateless (не зависят от предыстории). При этом для классов состояний придется определить новый интерфейс, отличающийся от интерфейса автомата тем, что в каждый метод добавлены параметры, через которые передаются ссылки на модель данных и интерфейс уведомления о событиях. Это приводит к фактическому дублированию интерфейсов, что затруд-

няет модификацию кода и его повторное использование, но экономит память.

3. Задание переходов. Переходы между состояниями задаются в контексте. Это можно сделать, например, при помощи конструкции switch, как предлагается в SWITCH-технологии [27]. Переходы также можно задать таблицей, отображающей пару <текущее состояние, событие> в <новое со-стояние>. Инфраструктуру для реализации табличного подхода можно реализовать в базовом для всех контекстов классе.

4. Протоколирование. Вынесение логики переходов в контекст позволяет осуществлять протоколирование переходов автоматов в терминах состояний и событий.

5. Создание модели данных. Экземпляр модели данных может либо создаваться конструктором контекста (как описано выше), либо передаваться в параметрах конструктора контекста.

Пример кода. Коды всех примеров, описанных в статье, доступны по адресу [28].

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

Интерфейсы и базовые классы, которые используются в данном примере, вынесены в пакет ru . ifmo . is . sm (sm - сокращение от State Machine). Диаграмма классов этого пакета приведена на рис. 4.

Опишем классы и интерфейсы, входящие в него.

1 . IEventSink - интерфейс уведомления о событии:

package ru.ifmo.is.sm;

public interface IEventSink {

public void castEvent(Event event);

}

2. Event - класс события. Используется для уведомления контекста из классов состояний:

package ru.ifmo.is.sm;

---------------------------LAi

StateBase

-automaton : Al -eventSink : EventSink

+StateBase(in automaton : Ai, in eventSink : ¡EventSink)

I «uses»

+castEvent(in event; Event)

#addEdge(in source : Al, in event: Event, in target: Al)

ш Рис. 4. Диаграмма классов пакета ru. i fmo . i s . sm

«interface» ru.ifmo. is.sm:: IEventSink

+castEvent(in event: Event)

A

AutomatonBase

Event

-name

--------------7^r

I

«uses» I

Al

public final class Event { private final String name;

public Event(String name) { if (name == null) (

throw new NullPointerException();

}

this.name = name;

}

public String getNameO { return name;

}

}

3. StateBase - базовый класс для состояний. Для проверки типов во время компиляции применяются параметры типа (generic, template), появившееся в Java 5.0. В конструкторе запоминается интерфейс для приема событий:

package ru.ifmo.is.sm;

public abstract class StateBase<AI> { protected final AI automaton; protected final IEventSink eventSink;

public StateBase(AI automaton, IEventSink eventSink) { if (automaton == null I I

eventSink == null) { throw new NullPointerException();

}

this.automaton = automaton; this.eventSink = eventSink;

}

protected void castEvent(Event event) { eventSink.castEvent(event);

}

}

4. AutomatonBase - базовый класс для всех автоматов. Он предоставляет возможность наследнику регистрировать переходы, используя метод addEdge. Дополнительно класс AutomatonBase реализует интерфейс уведомления о событии:

package ru.ifmo.is.sm; import java.util.

public abstract class AutomatonBase<AI> implements IEventSink { protected AI state;

private final Map<AI, Map<Event, AI»

edges = new HashMap<AI, Map<Event, AI» () ;

protected void addEdge(AI source, Event event, AI target) {

Map<Event, AI> row = edges.get(source); if (null == row) (

row = new IdentityHashMap<Event, AI>(); edges.put(source, row);

}

row.put(event, target);

}

public void castEvent(Event event) { try {

state = edges.get(state).get(event);

} catch (NullPointerException e) { throw new IllegalStateException(

"Edge is not defined");

}

}

}

Классы, созданные в соответствии с паттерном State Machine, образуют пакет connection. Диаграмма классов этого пакета приведена на рис. 5.

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

После определения интерфейса для клиента и модели данных необходимо перейти к определению управляющих состояний автомата. Для данного примера реализуем классы Connectedstate иDisconnectedState. В состоянии Connectedstate могут произойти события ERROR И DISCONNECT, а в состоянии DisconnectedState - события CONNECT И ERROR(см. рис. 1 ).

Обратим внимание, что на рис. 1 присутствуют дуги, помеченные одинаковыми событиями. В данном примере объекты событий создаются в классе состояния, из которого исходит переход. Например, событие error на переходе из состояния Connectedstate В состояние DisconnectedState не совпадает с аналогичным событием на петле из состояния DisconnectedState. При другой реализации ДЛЯ события ERROR МОГ бы быть СОЗДЭН только один объект.

iНе можете найти то, что вам нужно? Попробуйте сервис подбора литературы.

Приведем код классов состояний.

Класс Connectedstate:

№~2004*^^"

ШФОРМАЩОШО-УПР^ШШЮищЁс^Юташ”^^ 19

Connection «uses» \ /

«interface» connection:: IConnection

+connect() +connect()

+disconnect() +disconnect()

+receive(): int +receive(): int

+send(in data : int) +send(in data: int)

+createAutomaton(in socket: Socket я k

' / / / /

ConnectedState

+DISCONNECT : Event -socket: Socket

+connect()

+disconnect()

+receive(): int

+send(in data : int)

Socket

+connect() +disconnect() +receive(): int +send(in data : int)

DisconnectedState

+CONNECT : Event -socket: Socket

+connect() +disconnect() +receive(): int i-send(in data : int)

л -

■ Рис. 5. Диаграмма классов пакета connection

package connection;

import ru.ifmo.is.sm.*; import java.io.IOException;

public class ConnectedState <AI extends IConnection> extends StateBase<AI> implements IConnection { public static final Event DISCONNECT = new Event("DISCONNECT"); public static final Event ERROR = new Event("ERROR");

protected final Socket socket;

public ConnectedState(AI automaton,

IEventSink eventSink, Socket socket) { super(automaton, eventSink); this.socket = socket;

}

public void connect() throws IOException {}

public void disconnect() throws IOException { try {

socket.disconnect();

} finally {

eventSink.castEvent(DISCONNECT);

}

}

public int receive() throws IOException { try {

return socket.receive();

} catch (IOException e) { eventSink.castEvent(ERROR);

throw e;

}

}

public void send(int value) throws IOException {

try {

socket.send(value);

} catch (IOException e) { eventSink.castEvent(ERROR); throw e;

}

}

}

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

Класс DisconnectedState: package connection;

import j ava.io.IOException; import ru.ifmo.is.sm.*;

public class DisconnectedState <AI extends IConnection> extends StateBase<AI> implements IConnection {

public static final Event CONNECT = new Event("CONNECT");

public static final Event ERROR = new Event("ERROR");

protected final Socket socket;

public DisconnectedState(AI automaton, IEventSink eventSink, Socket socket) { super(automaton, eventSink); this.socket = socket;

}

public void connect() throws IOException { try {

socket.connect() ;

} catch (IOException e) { eventSink.castEvent(ERROR); throw e;

}

eventSink.castEvent(CONNECT);

}

public void disconnect() throws IOException {

}

public int receive() throws IOException { throw new IOException(

"Connection is closed (receive)");

}

public void send(int value) throws IOException { throw new IOException(

"Connection is closed (send)");

}

}

Остается описать класс Connection - контекст. В этом классе реализована логика переходов, в соответствии с графом переходов на рис. 1. Заметим, что последние четыре метода этого класса - делегирование интерфейса текущему состоянию:

package connection;

import java.io.IOException;

import ru.ifmo.is.sm.AutomatonBase;

public class Connection extends

AutomatonBase<IConnection> implements IConnection {

private Connection() {

Socket socket = new Socket();

// Создание объектов состояний IConnection connected = new

ConnectedState<Connection>(this, this, socket);

IConnection disconnected = new

DisconnectedState<Connection>(this, this, socket);

// Логика переходов addEdge(connected,

ConnectedState.DISCONNECT, disconnected); addEdge(connected, ConnectedState.ERROR, disconnected); addEdge(disconnected,

DisconnectedState.CONNECT, connected); addEdge(disconnected,

DisconnectedState.ERROR, disconnected);

// Начальное состояние state = disconnected;

}

// Создание экземпляра автомата

public static IConnection createAutomaton()

{

return new Connection();

}

// Делегирование методов интерфейса public void connect() throws IOException { state.connect (); } public void disconnect() throws IOException { state.disconnect(); } public int receive() throws IOException { return state.receive(); } public void send(int value) throws IOException { state.send(value); }

iНе можете найти то, что вам нужно? Попробуйте сервис подбора литературы.

}

Обратим внимание, что в классах состояний определена только логика генерации событий, а логика переходов сконцентрирована в классе контекста.

Повторное использование классов состояний

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

Расширение интерфейса автомата. Проиллюстрируем возможность расширения интерфейса автомата на примере добавления возможности возврата данных в объект соединения для их последующего считывания. Введем интерфейс IPushBackConnection, расширяющий интерфейс IConnection методом pushBack. Таким образом, расширенный интерфейс выглядит следующим образом:

package push_back_connection;

import connection.IConnection;

import java.io.IOException;

public interface IPushBackConnection extends IConnection { void pushBack(int value) throws IOException;

}

Отметим, что код для этого примера помещен в пакет push_back_connection, диаграмма классов которого представлена на рис. 6.

При вызове метода pushBack ( int value) значение, переданное в параметре этого метода, заносится в стек. При последующем вызове метода receive возвращается элемент из вершины стека, а если стек пуст, то значение извлекается из объекта socket.

Заметим, что в рассматриваемом случае количество управляющих состояний автомата не изменится, равно как и граф переходов автомата, но контекст и классы состояний должны реализовывать более широкий интерфейс IPushBackConnection. Назовем контекст нового автомата BushBackConnection, а новые состояния - PushBackConnectedState и PushBackDisconnectedState.

Приведем реализацию класса PushBack ConnectedState (класс PushBackDisconnected State реализуется аналогично). При этом класс

«interface» connection:.IConnection

ConnectedState

+DISCONNECT : Event -socket: Socket

+connect() +disconnect() +receive() : int +send(in data : int)

+connect() +disconnect() y| +receive() : int +send(in data : int)

TV

«interface»

push_back_connection:IPushBackConnection

+pushBack(in value : int)

DisconnectedState

+CONNECT : Event -socket: Socket

+connect() +disconnect() +receive() : int +send(in data : int)

\ \ \ \ 1

PushBackConnectedState

-stack

+pushBack(in value : int)

+connect()

+disconnect()

+receive() : int

+send(in data : int)

PushBackConnection

-state : IConnection

+pushBack(in value : int)

+connect()

+disconnect()

+receive() : int +send(in data : int)

+createAutomaton(in socket : Socket)

N \

PushBackDisconnectedState

+pushBack(in value : int)

+connect()

+disconnect()

+receive() : int

+send(in data : int)

■ Рис. 6. Диаграмма классов пакета push_back_connection

PushBackConnectedState расширяет класс ConnectedState,наследуя его логику:

package push_back_connection;

import connection.*;

import ru.ifmo.is.sm.IEventSink;

import java.util.Stack;

import j ava.io.IOException;

public class PushBackConnectedState <AI extends IPushBackConnection> extends ConnectedState<AI> implements IPushBackConnection { Stack<Integer> stack

= new Stack<Integer>();

public PushBackConnectedState(AI automaton, IEventSink eventSink, Socket socket) {

super(automaton, eventSink, socket);

}

public int receive() throws IOException { if (stack.empty()) {

return super.receive();

}

return stack.pop().intValue();

}

public void pushBack(int value) { stack.push(new Integer(value));

}

}

Отметим,что в классе PushBackConnectedState параметр типа специализируется еще более, чем в классе ConnectedState. Это требуется для того, чтобы поле automaton, определенное в классе StateBase, имело правильный тип - IPushBack Connection. Таким образом, интерфейс автома-

та «протягивается« сквозь всю иерархию классов состояний через параметр типа.

Созданные классы состояний используются для реализации класса контекста PushBackConnection:

package push_back_connection;

import connection.Socket;

import ru.ifmo.is.sm.AutomatonBase;

import j ava.io.IOException;

public class PushBackConnection extends

AutomatonBase<IPushBackConnection> implements IPushBackConnection { private PushBackConnection() {

Socket socket = new Socket ();

// Создание объектов состояний IPushBackConnection connected = new PushBackConnectedState <PushBackConnection>(this, this, socket);

IPushBackConnection disconnected = new PushBackDisconnectedState <PushBackConnection>(this, this, socket);

// Логика переходов addEdge(connected,

PushBackConnectedState. DISCONNECT, disconnected); addEdge(connected,

PushBackConnectedState.ERROR, disconnected); addEdge(disconnected,

PushBackDisconnectedState.CONNECT, connected);

// Начальное состояние state = disconnected;

}

// Создание экземпляра автомата public static IPushBackConnection créateAutomaton() {

return new PushBackConnection();

}

// Делегирование методов интерфейса public void connect() throws IOException { state.connect(); } public void disconnect() throws IOException { state.disconnect(); } public int receive() throws IOException { return state.receive(); } public void send(int value) throws IOException { state.send(value); } public void pushBack(int value) throws IOException { state.pushBack(value) ; }

}

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

Расширение логики введением новых состояний. Расширение логики поведения будем рассматривать на примере сетевого соединения, которое в случае возникновения ошибки при передаче данных закрывает канал и бросает исключение. При очередной попытке передачи данных производится попытка восстановить соединение и передать данные.

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

Таким образом, требуется реализовать класс ResumableConnection. Для этого необходимо дополнительно реализовать класс ErrorState, определяющий поведение в состоянии ошибка. Для состояний соединено и разъединено используются классы ConnectedState и DisconnectedState. Граф переходов ДЛЯ класса ResumableConnection изображен на рис. 7.

Класс ErrorState реализуется следующим образом:

package resumable_connection;

import connection.* ;

import ru.ifmo.is.sm.* ;

import j ava.io.IOException;

iНе можете найти то, что вам нужно? Попробуйте сервис подбора литературы.

public class ErrorState

<AI extends IConnection> extends StateBase<AI> implements IConnection { public static final Event CONNECT = new Event("CONNECT"); public static final Event

DISCONNECT = new Event("DISCONNECT");

protected final Socket socket;

public ErrorState(AI automata, IEventSink eventSink, Socket socket) { super(automata, eventSink); this.socket = socket;

}

public void connect() throws IOException { socket.connect();

castEvent(CONNECT);

}

public void disconnect() throws IOException {

castEvent(DISCONNECT);

}

public int receive() throws IOException { connect();

return automaton.receive();

}

public void send(int value) throws IOException { connect();

automaton.send(value);

}

}

Класс ResumableConnection реализуется так же:

package resumable_connection;

import connection.* ;

import ru.ifmo.is.sm.AutomatonBase;

import j ava.io.IOException;

public class ResumableConnection extends Au toma tonBa s e<IConneс tion> implements IConnection {

private ResumableConnection() {

Socket socket = new Socket();

// Создание объектов состояний IConnection connected = new

OaxiectedState<FtesurtBbleQxinecticn> (this, this, socket);

IConnection disconnected = new

EiscarectedState<ResuTBblQ3xinecticrï> (this, this, socket);

IConnection error = new

ErrorState<ResumableConnection>(this, this, socket);

// Логика переходов addEdge(connected,

ConnectedState.DISCONNECT, disconnected); addEdge(connected, ConnectedState.ERROR, error); addEdge(disconnected,

DisconnectedState.CONNECT, connected); addEdge(disconnected,

DisconnectedState.ERROR , error);

addEdge(error, Errors tate.CONNECT, connected);

^/программные и аппаратные средства f

addEdge(error, ErrorState.DISCONNECT, disconnected);

// Начальное состояние state = disconnected;

}

// Создание экземпляра автомата public static IConnection createAutomaton() {

return new ResumableConnection();

}

// Делегирование методов интерфейса public void connect() throws IOException { state.connect(); } public void disconnect() throws IOException { state.disconnect(); } public int receive() throws IOException { return state.receive(); } public void send(int value) throws IOException { state.send(value); }

}

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

Выводы

Паттерн State Machine является усовершенствованием паттерна State. Он заимствует основную идею паттерна State - локализацию поведения, зависящего от состояния, в различных классах.

Новый паттерн устраняет ряд недостатков, присущих паттерну State.

1. Паттерн State Machine позволяет разрабатывать отдельные классы независимыми друг от друга. Поэтому один и тот же класс состояния можно использовать в нескольких автоматах, каж-

Литература /

1. McCuIloch W., Pitts W. A logical calculus of ideas immanent in nervous activity // Bull. Math. Biophysics. 1943,5. - P. 115-133.

2. Гаврилов М. А. Теория релейно-контактных схем.-М.: Изд-воАН СССР, 1950.-230 с.

3. Huffman D. A. The synthesis of sequential switching circuits // J. Franclin Inst. 1954. Vol. 257, N 3, 4. - P. 161-190.

4. Mealy G. A method for synthesizing sequential circuits // Bell system technical journal. 1955. Vol. 34. N 5. - P. 1045-1079.

5. Moor© E, Gedanken experiments on sequential machines // Ed. С. E. Shannon, J. McCarthy. Princeton Univ. Press, 1956. - P. 129-153.

6. Automata Studies//Ed.C. E.Shannon, J. McCarthy. Princeton

Univ. Press, 1956.-P. 400 (Автоматы // Под ред. К. Э. Ше н но -на,Дж. МакКарти. М.: Изд-воиностр. пит., 1956.-451 с.).

7. Rubin М., Scott D. Finite automata and their decision problem //

IBM J. Research and Development. 1959. Vol. 3. N 2. - P. 115-125 (Кибернетическийсборник. Вып.4. М.: Изд-во иностр. лит., 1962).

8. Пл ушков В. М. Синтез цифровых автоматов. - М.: Изд-во фиэ.-мат. пит., 1962. -476с.

9. Kleene S. С. Representation of events in nerve nets and finite automata// Ed. G. E. Shannon, J. McCarthy. Princeton Univ. Press, 1956. - P. 3-41

10. Thompson K. Regular expression search algorithm // Communications of the ACM. 1966.-Vol. 11. N6.-P. 419-422.

11. Abo A,, Sethi R., UHman J. Compilers: principles, techniques and tools. MA; Addison-Wesley, 1985. - 500 p. (Axo А., Сети-

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

2. В паттерне State не описано, каким образом обеспечивается чистота интерфейса, предназначенного для клиента. Предлагаемый паттерн устраняет эту проблему.

3. В паттерне State логика переходов распределена по классам состояний, что порождает зависимости между классами и сложность восприятия логики переходов в целом. В паттерне State Machine логика переходов реализуется в контексте. Это позволяет разделить логику переходов и поведение в конкретном состоянии.

4. Использование паттерна State Machine не приводит к дублированию интерфейсов.

Тем не менее, паттерн State Machine не устраняет такой недостаток паттерна State, как необходимость производить делегирование интерфейса автомата текущему объекту состояния вручную. Это ограничение можно обойти, используя автоматическую генерацию кода контекста или языки программирования, поддерживающие динамическое делегирование. Автоматическая генерация кода обычно используется в CASE-средствах. Второй подход можно реализовать, например, на языке Self [29], в котором описанное выше делегирование можно выполнить при помощи динамического наследования. Это обеспечивается тем, что язык позволяет непосредственно изменять класс объекта во время выполнения, а объекты в этом языке могут делегировать операции другим объектам.

Р., Ульман Дж. Компиляторы: принципы, технологии и инструменты. - М.: Вильямс, 2001. -768 с.).

12. Hopcroft J., Motwani R., Uilman J. Introduction to automata theory, languages and computation. MA: Addison-Wesley, 2001. - 521 p. (Хопкрофт Д., Мотвани P., Ульман Дж. Введение в теорию автоматов, языков и вычислений. -М.: Випьямс, 2001. - 528 с.).

13. Шалыто А, А. SWITCH-технология. Алгоритмизация и программирование задач логического управления. СПб.: Наука, 1998.-628 с.

14. Gamma Е., Helm R., Johnson R., Vlissides J. Design Patterns. MA: Addison-Wesley Professional. 2001. - 395 (Гамма Э.,Хелм Р, Джонсон Р., Влассидес Дж. Приемы объектно-ориентированного проектирования. Паттерны проектирования. СПб.: Питер, 2001.-368 с.).

15. Steling S., Maassen О. Applied Java patterns. Pearson higher education. 2001, P. 608 (Стептинг С., Массен О. Применение шаблонов Java. Бибпиотека профессионала. М.: Вильямс, 2002. -564 с.).

16. Grand М. Patterns in Java: A catalog of reusable design patterns illustrated with UML. Wiley, 2002. - 544 p. (Гран-д М. Шаблоны проектирования в Java. М.: Новое знание, 2004. -560 с.).

17. Java Data Objects (JDO). (http://java.sun.com/products/jdo/ index.jsp).

18. Ellens A. Principles of object-oriented software development MA.: Addison-Wesley, 2000. - 502 p. (Элиенс А. Принципы

объектно-ориентированной разработки программ. М.; Вильямс, 2002.-496 с.).

19. Sandiftn 8. The state-machine pattern // Proceedings of the conference on TRI-Ada ’96 (http://java.sun.com/products/jdo/ index.jsp).

20. Adamczyk P. The anthology of the finite state machine design patterns. (http://jerry.cs.uiuc.edu/-plop/plop2003/Papers/ Adamczyk-State - Machine. pdf)

21. Odrowski J., Sogaard P. Pattern Integration - variations of state // Proceedings of PLoP96. (http://www.cs.wustl.edu/ '-schmidt/PLoP-96/odrowski.ps.gz).

22. Sane A., Campbell R.. Object-oriented state machines: subclassing, composition, delegation, andgenericity//OOPSLA ’95. (http://choices.cs.uiuc.edu/sane/home.html).

23. Шалыто A. A.} Туккель H. И. Оттьюрингова программирования к автоматному. // Мир ПК. 2002. N° 2. - С. 144-149. (http://isJfmo.ru, раздел «Статьи»),

24. Harel D. Statecharts: A visual formalism for complex systems // Sci. Comput. Program. 1987. Vol.8. - P. 231 -274

25. Fawler M. Refactoring, improving the design of existing code. MA; Addison-Wesley. - 1999. -431 p. (Фаулер М. Рефакторинг. Улучшение существующего кода. - М.: Символ-плюс, 2003. — 432 с.).

26. Martín R. Three Level FSM // PLoPD, 1995. (http:// cpptips.hyperformix.com/cpptips/fsm5).

27. Шальгго А. А*, Туккель H. И. SWITCH-технология - автоматный подход к созданию программного обеспечения «реактивных» систем // Программирование. 2001. № 5. С. 45-62. (http://is.ifmo.ru, раздел «Статьи»).

28. Раздел «Статьи» сайта кафедры «Технологии программирования» Санкт-Петербургского государственного университета информационных технологий, механики и оптики (http://is.ifmo.ru/articles).

29. The Self Language (http://research.sun.com/self/ language.html).

ИНФОРМАЦИОННО-УПРАВЛЯЮЩИЕ СИСТЕМЫ

Научно-практический журнал

Подписной индекс по каталогу «Роспечать»: «Газеты и журналы» - № 15385, «Издания органов НТИ» - № 69291

Периодичность - каждые два месяца. Тираж - 10ОО экз. Распространяется только по подписке в России и странах СНГ. Возможна подписка через редакцию по заявке (по почте, телефону, факсу или e-mail), по которой высылаем счет. Высылаем по Вашей просьбе (бесплатно) образец журнала для подписки. Стоимость годовой подписки (6 номеров) - 1800 руб. ( включая НДС 1 0 %), с добавлением стоимости доставки - 90 рублей по России и 300 рублей в страны СНГ. Подписчики информируются о новых книгах издательства «Политехника» и получают скидки на публикацию рекламы. При повторной подписке скидка 10 %.

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

Цветные полосы Черно-белые полосы Скидки при единовременной оплате

1 -я стор. обложки 15000 1 полоса A4 4000 2-х публикаций 10 %

2-я стор. обложки и каждая стр. вкладки 12000 1/2 полосы 2500 3-х публикаций 15 %

3-я стор. обложки 10000 1 /2 полосы 1125 4-х и более 20 %

iНе можете найти то, что вам нужно? Попробуйте сервис подбора литературы.

4-я стор. обложки 12000 1 /8 ПОЛОСЬ! 800

Примечание: при размещении цветного рекламного модуля не менее 1/2 страницы сопутствующая статья (1-2 страницы) печатается бесплатно.

Требования к рекламным модулям. Принимаются оригиналы фотографий высокого качества и контрастности. Рекламные модули в файловом виде на компакт-дисках или присланные по e-mail в заархивирован-ном виде (RAR, ZIP) с разбивкой на дискеты предоставляются только в форматах TIFF, JPEG, BMP (с разрешением не меньше 300 dpi), выполненные в программах Adobe Photoshop 5.0, Corel Draw 9.0, 10.0.

i Надоели баннеры? Вы всегда можете отключить рекламу.