Е. К. Пугачев Объектно-ориентированное программирование Под общей редакцией Ивановой Г. С. Рекомендовано Министерством общего и профессионального образования Российской Федерации в качестве учебник
Вид материала | Учебник |
СодержаниеType TMyClass=class Пример 5.76. Делегирование методов (графический редактор «Окружности и квадраты» - вариант 2) 1: C.Draw:=C.DrawSquare Begin if Cnil then Unit Figure Begin if Cnil then |
- Е. Л. Григоренко психогенетика под редакцией И. В. Равич-Щербо Рекомендовано Министерством, 27103.82kb.
- Г. В. Плеханова И. Н. Смирнов, В. Ф. Титов философия издание 2-е, исправленное и дополненное, 4810.28kb.
- К. Э. Фабри Основы зоопсихологии 3-е издание Рекомендовано Министерством общего и профессионального, 5154.41kb.
- Е. А. Климов введение в психологию труда рекомендовано Министерством общего и профессионального, 4594.17kb.
- Н. Ф. Самсонова Рекомендовано Министерством общего и профессионального образования, 6152.94kb.
- Ю. Г. Волков И. В. Мостовая социология под редакцией проф. В. И. Добренькова Рекомендовано, 6915.59kb.
- Е. Ф. Жукова Рекомендовано Министерством общего и профессионального образования Российской, 6286.83kb.
- О. А. Кривцун эстетика Рекомендовано Министерством общего и профессионального образования, 6381.8kb.
- В. И. Рудой классическая буддийская философия рекомендовано Министерством, общего, 6771.74kb.
- В. И. Ильинича Рекомендовано Министерством общего и профессионального Образования Российской, 6751.75kb.
5.5.Делегирование
В Delphi существует возможность объявления указателей на методы. Как уже упоминалось в разделе 5.1, в отличие от обычных подпрограмм (процедур и функций) методы в списке параметров содержат неявную ссылку на объект, которая обеспечивает методу доступ к полям и методам «своего» объекта. Соответственно, указатель на метод обеспечивает корректную передачу этой ссылки.
При объявлении типа «указатель на метод» используется специальный спецификатор of object:
Type <имя типа> = <заголовок метода> of object;
Например:
Type TMyMetod = function (ax:integer):word of object;
TMyClass=class
public FMetod:TMyMetod;
end;
Поле FMetod, описанное выше, имеет процедурный тип, его значением может быть имя метода данного или другого класса.
В Delphi доступ к полям, содержащим указатели на методы, обычно осуществляется через свойства:
^ Type TMyClass=class
private FMetod:TMyMetod;
public
property Metod:TMyMetod read FMetod write FMetod;
end;
Как уже упоминалось в разделе 5.4, такие свойства получили название процедурных. С использованием процедурных свойств в Delphi осуществляется делегирование (раздел 1.7), которое позволяет определять различное поведение объектов одного класса.
Так подключение обработчиков событий стандартных компонент представляет собой статическое делегирование, при котором поведение объекта определяется в процессе проектирования приложения.
При динамическом делегировании поведение объекта уточняется во время выполнения программы, что позволяет создавать объекты с динамически изменяющимся поведением.
^ Пример 5.76. Делегирование методов (графический редактор «Окружности и квадраты» - вариант 2)
В качестве примера рассмотрим создание приложения, которое в зависимости от положения радиокнопки по нажатию левой клавиши мыши рисует в специальном окне окружность или квадрат. Аналогичный пример уже рассматривался в разделе 5.2, но там нужный эффект достигался за счет переопределения методов. Классы TCircul и Tsquare включали свои методы рисования Draw. Смена типа фигуры реализовывалась за счет создания объекта другого класса. Изменить тип уже созданного объекта в рассмотренном там варианте решения нельзя.
Использование делегирования позволяет создать объект, который при изменении положения радиокнопки будет изменять тип и соответственно вид уже нарисованной фигуры (рис. 5. 14).
Рис. 5.81. Вид главного окна приложения «Окружности и квадраты»
Вариант 1. Простейший вариант создания такого объекта заключается в том, чтобы включить в описание класса TFigure обе процедуры рисования фигур DrawCircul (рисование окружности) и DrawSquare (рисование квадрата), осуществляя вызов процедуры рисования через свойство Draw - указатель на метод рисования. Для этого придется определить тип указателя на метод без параметров TDProc.
unit Figure;
interface
Uses extctrls,Graphics;
Type TDProc=procedure of object;
TMyFigure=class
private
x,y,r:word;
Image:TImage;
FDraw:TDProc; {поле свойства Draw}
public
Constructor Create(aImage:TImage;ax,ay,ar:Word); {конструктор}
property Draw:TDProc read FDraw write FDraw; {процедурное свойство}
Procedure Clear; {стирание фигуры}
Procedure DrawCircul; {рисование окружности}
Procedure DrawSquare; {рисование квадрата}
end;
Implementation
Constructor TMyFigure.Create;
Begin
inherited Create; Image:=aImage;
x:=ax; y:=ay; r:=ar;
End;
Procedure TMyFigure.Clear;
Begin Image.Canvas.Pen.Color:=clWhite;
Draw; {вызов метода по адресу, указанному в свойстве}
Image.Canvas.Pen.Color:=clBlack;
End;
Procedure TMyFigure.DrawCircul;
Begin Image.Canvas.Ellipse(x-r,y-r,x+r,y+r); End;
Procedure TMyFigure.DrawSquare;
Begin Image.Canvas.Rectangle(x-r,y-r,x+r,y+r); End;
end.
Вид выводимой фигуры определяется «нажатой» радиокнопкой:
case RadioGroup.ItemIndex of
0: C.Draw:=C.DrawCircul;
^ 1: C.Draw:=C.DrawSquare;
end;
Переопределение свойства Draw (указателя на метод рисования) выполняется в обработчике события RadioGroupClick (щелчок мышью на радиокнопках):
Procedure TMainForm.RadioGroupClick(Sender: TObject);
^ Begin if C<>nil then {если объект создан, то}
begin
C.Clear; {стереть изображение}
case RadioGroup.ItemIndex of {в зависимости от выбора}
0: C.Draw:=C.DrawCircul; { рисование окружности}
1: C.Draw:=C.DrawSquare; { рисование квадрата}
end;
C.Draw; {нарисовать фигуру}
end;
End;
Вариант 2. Более сложный вариант делегирования - подключение методов, определенных в других классах.
Определим специальный класс TDraw, который будет содержать методы рисования. Необходимые параметры эти методы будут получать через список аргументов, которые сгруппированы в запись типа TParam.
^ Unit Figure;
Interface
Uses extctrls,Graphics;
Type TParam=record
x,y,r:Word;
Color:TColor;
Image:TImage;
end;
TDProc=Procedure(AParam:TParam) of object;
TMyFigure=class
private FDraw:TDProc;
public
Param:TParam;
Procedure Clear;
Constructor Create(aImage:TImage;ax,ay,ar:Word;aColor:TColor);
property Draw:TDProc read FDraw write FDraw;
end;
TDraw=class
public
Procedure DrawCircul(AParam:TParam);
Procedure DrawSquare(AParam:TParam);
end;
Implementation
Constructor TMyFigure.Create;
begin
inherited Create;
Param.Image:=aImage; Param.Color:=aColor;
Param.x:=ax; Param.y:=ay; Param.r:=ar;
end;
Procedure TMyFigure.Clear;
Var TempColor:TColor;
Begin
TempColor:=Param.Color;
Param.Color:=Param.Image.Canvas.Brush.Color;
Draw(Param); {вызов делегированного метода}
Param.Color:=TempColor;
End;
Procedure TDraw.DrawCircul;
Begin
AParam.Image.Canvas.Pen.Color:=AParam.Color;
AParam.Image.Canvas.Ellipse(AParam.x-AParam.r, AParam.y-AParam.r,
AParam.x+AParam.r,AParam.y+AParam.r);
End;
Procedure TDraw.DrawSquare;
Begin
AParam.Image.Canvas.Pen.Color:=AParam.Color;
AParam.Image.Canvas.Rectangle(AParam.x-AParam.r,
AParam.y-AParam.r, AParam.x+AParam.r, AParam.y+AParam.r);
End;
End.
Переопределение свойства Draw в обработчике события RadioGroupClick теперь будет выполняться следующим образом:
Procedure TMainForm.RadioGroupClick(Sender: TObject);
^ Begin
if C<>nil then
begin C.Clear;
case RadioGroup.ItemIndex of
0: C.Draw:=G.DrawCircul; {делегирование метода}
1: C.Draw:=G.DrawSquare; {делегирование метода}
end;
C.Draw(C.Param); {вызов процедуры рисования}
end;
End;
Объект G класса TDraw может быть создан в секции инициализации модуля и уничтожен - в секции завершения:
initialization G:=TDraw.Create;
finalization G.Free;