Структурные паттерны
Adapter, Composite, Decorator
OOP2Struc1°. Adapter (Адаптер) — структурный паттерн. Известен также под именем Wrapper (Обертка). Частота использования: выше средней. Назначение: преобразует интерфейс одного класса в интерфейс другого, который ожидают клиенты. Адаптер обеспечивает совместную работу классов с несовместимыми интерфейсами, которая без него была бы невозможна. Участники: • Target (Целевой класс) — определяет зависящий от предметной области интерфейс, которым пользуется Client; • Client (Клиент) — вступает во взаимоотношения с объектами, удовлетворяющими интерфейсу Target; • Adaptee (Адаптируемый класс) — определяет существующий интерфейс, который нуждается в адаптации; • Adapter (Адаптер) — адаптирует интерфейс Adaptee к интерфейсу Target. В данном задании рассматривается вариант адаптера, использующий композицию. Такой вариант называется адаптером объекта, поскольку адаптируется объект типа Adaptee, входящий в виде ссылочного поля в класс Adapter. Задание 1. Абстрактный класс Target содержит три абстрактных метода: GetA, GetB и Request (не имеют параметров, возвращают значение целого типа). Класс ConcreteTarget является потомком класса Target; он содержит поля a и b целого типа, которые инициализируются в конструкторе, имеющем одноименные параметры. Методы GetA и GetB класса ConcreteTarget возвращают значения полей a и b соответственно, а метод Request возвращает сумму этих полей. Класс Adaptee содержит два целочисленных поля a и b, конструктор с параметрами a и b, задающий значения этих полей, метод GetAll, возвращающий текущие значения полей (либо с помощью выходных параметров, либо с помощью возвращаемого значения — массива или кортежа), и метод SpecificRequest без параметров, возвращающий произведение полей a и b. Реализовать класс Adapter, адаптирующий класс Adaptee к интерфейсу класса Target. Класс должен быть адаптером объекта: он порождается от класса Target и включает ссылку ad на экземпляр адаптируемого объекта Adaptee. Ссылка ad инициализируется в конструкторе класса Adapter путем вызова конструктора класса Adaptee с параметрами a и b, совпадающими с одноименными параметрами конструктора класса Adapter. Метод Request класса Adapter должен вызывать метод SpecificRequest объекта ad, а методы GetA и GetB — возвращать значения полей a и b объекта ad, используя его метод GetAll. Тестирование разработанной системы классов. Дано целое число N (≤ 6) и набор из N троек (C, A, B), где C является символом «+» или «*», а элементы A и B являются целыми числами. Создать структуру данных (например, массив) с элементами-ссылками типа Target и заполнить ее объектами типа ConcreteTarget (для троек с символом «+») и Adapter (для троек с символом «*») с полями, равными A и B. Перебирая элементы полученного набора в обратном порядке, вывести для каждого из них значения полей a, b и результат выполнения метода Request.
[C++] class Adaptee
{
// Do not change the implementation of the class
int a, b;
public:
Adaptee(int a, int b) : a(a), b(b) {}
void GetAll(int& a, int& b);
int SpecificRequest();
};
void Adaptee::GetAll(int& a, int& b)
{
a = this->a;
b = this->b;
}
int Adaptee::SpecificRequest()
{
return a * b;
}
class Target
{
public:
virtual int GetA() = 0;
virtual int GetB() = 0;
virtual int Request() = 0;
virtual ~Target()
{
Show("Target");
}
};
// Implement the ConcreteTarget and Adapter classes
[C#] public class Adaptee
{
// Do not change the implementation of the class
int a, b;
public Adaptee(int a, int b)
{
this.a = a;
this.b = b;
}
public void GetAll(out int a, out int b)
{
a = this.a;
b = this.b;
}
public int SpecificRequest()
{
return a * b;
}
}
public abstract class Target
{
public abstract int GetA();
public abstract int GetB();
public abstract int Request();
}
// Implement the ConcreteTarget and Adapter classes
[Java] class Adaptee {
// Do not change the implementation of the class
private int a,b;
public Adaptee(int a,int b) {
this.a = a;
this.b = b;
}
public int[] getAll() {
int res[] = {this.a,this.b};
return res;
}
public int specificRequest() {
return a * b;
}
}
abstract class Target {
public abstract int getA();
public abstract int getB();
public abstract int request();
}
// Implement the ConcreteTarget and Adapter classes
[Python] class Adaptee:
# Do not change the implementation of the class
def __init__(self, a, b):
self.__a = a
self.__b = b
def getAll(self):
return self.__a, self.__b
def specificRequest(self):
return self.__a * self.__b
class ConcreteTarget:
def __init__(self, a, b):
pass
# Implement the "constructor"
def getA(self):
pass
# Implement the method
def getB(self):
pass
# Implement the method
def request(self):
pass
# Implement the method
# Implement the Adapter class
[Ruby] class Adaptee
# Do not change the implementation of the class
def initialize(a,b)
@a = a
@b = b
end
def getAll
return @a, @b
end
def specificRequest
@a * @b
end
end
class ConcreteTarget
def initialize(a,b)
# Implement the "constructor"
end
def getA
# Implement the method
end
def getB
# Implement the method
end
def request
# Implement the method
end
end
# Implement the Adapter class
[Julia] # Do not change the implementation
# of the Adaptee struct
# and its associated functions
struct Adaptee
a::Int
b::Int
end
function getAll(ad::Adaptee)
ad.a, ad.b
end
function specificRequest(ad::Adaptee)
ad.a * ad.b
end
abstract type Target
end
# Implement the ConcreteTarget and Adapter structs
[PascalABC.NET] type
// Не изменяйте реализацию класса
Adaptee = class
private
a, b: integer;
public
constructor Create(a, b: integer);
begin
self.a := a;
self.b := b;
end;
procedure GetAll(var a, b: integer);
begin
a := self.a;
b := self.b;
end;
function SpecificRequest: integer;
begin
Result := a * b;
end;
end;
Target = abstract class
public
function GetA: integer; abstract;
function GetB: integer; abstract;
function Request: integer; abstract;
end;
// Реализуйте классы ConcreteTarget и Adapter
OOP2Struc2°. Adapter (Адаптер) — структурный паттерн. В данном задании рассматривается вариант адаптера, использующий множественное наследование. Такой вариант называется адаптером класса, поскольку класс Adapter порождается от класса Adaptee, наследуя его реализацию (а также от класса Target, наследуя его интерфейс). Если язык не поддерживает множественное наследование, то адаптер класса можно реализовать с помощью интерфейсов; для этого надо преобразовать абстрактный класс Target в интерфейс ITarget, сделать класс Adapter потомком класса Adaptee и добавить к нему интерфейс ITarget. Задание 2. Выполнить предыдущее задание (см. OOP2Struc1), реализовав Adapter как адаптер класса. Для языков, не поддерживающих множественное наследование, определить интерфейс ITarget с методами GetA, GetB и Request и добавить интерфейс ITarget к классам ConcreteTarget и Adapter (абстрактный класс Target теперь не требуется; для хранения исходных данных надо использовать коллекцию с элементами интерфейсного типа ITarget).
[C++] class Adaptee
{
// Do not change the implementation of the class
int a, b;
public:
Adaptee(int a, int b) : a(a), b(b) {}
void GetAll(int& a, int& b);
int SpecificRequest();
};
void Adaptee::GetAll(int& a, int& b)
{
a = this->a;
b = this->b;
}
int Adaptee::SpecificRequest()
{
return a * b;
}
class Target
{
public:
virtual int GetA() = 0;
virtual int GetB() = 0;
virtual int Request() = 0;
virtual ~Target()
{
Show("Target");
}
};
// Implement the ConcreteTarget and Adapter classes;
// for the Adapter class, use multiple inheritance
// (from the Target and Adaptee classes)
[C#] public class Adaptee
{
// Do not change the implementation of the class
int a, b;
public Adaptee(int a, int b)
{
this.a = a;
this.b = b;
}
public void GetAll(out int a, out int b)
{
a = this.a;
b = this.b;
}
public int SpecificRequest()
{
return a * b;
}
}
public abstract class Target
{
// Convert this abstract class into the ITarget interface
public abstract int GetA();
public abstract int GetB();
public abstract int Request();
}
// Implement the ConcreteTarget and Adapter classes.
// These classes must implement the ITarget interface;
// the Adapter class must be a descendant of the Adaptee class
[Java] class Adaptee {
// Do not change the implementation of the class
private int a,b;
public Adaptee(int a,int b) {
this.a = a;
this.b = b;
}
public int[] getAll() {
int res[] = {this.a,this.b};
return res;
}
public int specificRequest() {
return a * b;
}
}
abstract class Target {
// Convert this abstract class into the ITarget interface
public abstract int getA();
public abstract int getB();
public abstract int request();
}
// Implement the ConcreteTarget and Adapter classes.
// These classes must implement the ITarget interface;
// the Adapter class must be a descendant of the Adaptee class
[Python] class Adaptee:
# Do not change the implementation of the class
def __init__(self, a, b):
self.__a = a
self.__b = b
def getAll(self):
return self.__a, self.__b
def specificRequest(self):
return self.__a * self.__b
class ConcreteTarget:
def __init__(self, a, b):
pass
# Implement the "constructor"
def getA(self):
pass
# Implement the method
def getB(self):
pass
# Implement the method
def request(self):
pass
# Implement the method
# Implement the Adapter class;
# the Adapter class must be
# a descendant of the Adaptee class
[Ruby] class Adaptee
# Do not change the implementation of the class
def initialize(a,b)
@a = a
@b = b
end
def getAll
return @a, @b
end
def specificRequest
@a * @b
end
end
class ConcreteTarget
def initialize(a,b)
# Implement the "constructor"
end
def getA
# Implement the method
end
def getB
# Implement the method
end
def request
# Implement the method
end
end
# Implement the Adapter class;
# the Adapter class must be a descendant
# of the Adaptee class
Указание (Julia). Поскольку в Julia не могут наследоваться конкретные типы, адаптер класса в этом языке реализовать нельзя. Можно решить задачу, используя адаптер объекта (т. е. тем же способом, что и задачу OOP2Struc1).
[Julia] # Do not change the implementation
# of the Adaptee struct
# and its associated functions
struct Adaptee
a::Int
b::Int
end
function getAll(ad::Adaptee)
ad.a, ad.b
end
function specificRequest(ad::Adaptee)
ad.a * ad.b
end
abstract type Target
end
# Implement the ConcreteTarget and Adapter structs
[PascalABC.NET] type
// Не изменяйте реализацию класса
Adaptee = class
private
a, b: integer;
public
constructor Create(a, b: integer);
begin
self.a := a;
self.b := b;
end;
procedure GetAll(var a, b: integer);
begin
a := self.a;
b := self.b;
end;
function SpecificRequest: integer;
begin
Result := a * b;
end;
end;
// Преобразуйте абстрактный класс в интерфейс ITarget
Target = abstract class
function GetA: integer; abstract;
function GetB: integer; abstract;
function Request: integer; abstract;
end;
// Реализуйте классы ConcreteTarget и Adapter.
// Эти классы должны реализовывать интерфейс ITarget;
// класс Adapter должен быть потомком класса Adaptee
OOP2Struc3°. Adapter (Адаптер) — структурный паттерн. Задание 3. Дан абстрактный класс Shape, предоставляющий интерфейс для графических объектов: метод GetInfo без параметров, возвращающий строку с именем объекта и координатами левой верхней и правой нижней вершины ограничивающего прямоугольника (считается, что ось OY направлена вниз), и метод MoveBy(a, b) с двумя целочисленными параметрами, определяющими вектор, на который надо сместить данный графический объект (метод не возвращает значений). В классе Shape методы GetInfo и MoveBy являются абстрактными. Также дан конкретный класс RectShape — потомок класса Shape, реализующий прямоугольник и имеющий конструктор с параметрами (x1, y1, x2, y2), которые задают координаты левой верхней и правой нижней вершины этого прямоугольника. Метод GetInfo для данного класса возвращает строку вида «R(x1,y1)(x2,y2)» с текущими значениями координат (например, «R(1,−4)(3,2)»). Дан класс TextView для работы с текстовыми объектами. Он содержит поля x, y (координаты точки привязки — левого верхнего угла текстовой области), width, height (ширина и высота текстовой области) и методы GetOrigin (возвращает координаты точки привязки), SetOrigin (изменяет точку привязки), GetSize (возвращает размеры текстовой области) и SetSize (изменяет размеры текстовой области). Методы GetOrigin и GetSize возвращают результаты либо с помощью выходных параметров, либо с помощью возвращаемого значения — массива или кортежа. Конструктор класса не имеет параметров (поля x и y полагаются равными 0, поля width и height — равными 1). Реализовать класс TextShape, адаптирующий класс TextView к интерфейсу класса Shape. Класс должен быть адаптером объекта: он порождается от класса Shape и включает поле tview, являющееся ссылкой на экземпляр адаптируемого объекта TextView (других полей класс TextShape не содержит). Метод GetInfo класса TextShape должен возвращать строку вида «T(x1,y1)(x2,y2)» с текущими значениями левой верхней и правой нижней вершины ограничивающего прямоугольника. Включить в класс TextShape конструктор с параметрами (x1, y1, x2, y2), задающими координаты левой верхней и правой нижней вершины ограничивающего прямоугольника для текстовой области (эти параметры должны использоваться в конструкторе при инициализации поля tview). Тестирование разработанной системы классов. Дано целое число N (≤ 8) и набор из N пятерок (C, X1, Y1, X2, Y2), где C является символом «R» или «T», а остальные элементы являются целыми числами. Кроме того, даны целые числа A и B. Создать структуру данных (например, массив) с элементами типа Shape и заполнить ее объектами типа RectShape (для пятерок с символом «R») и TextShape (для пятерок с символом «T»), используя значения X1, Y1, X2, Y2 в качестве параметров соответствующего конструктора. Применить к каждому элементу созданного набора метод MoveBy с параметрами A и B и вывести строковые представления элементов набора с помощью метода GetInfo (перебирая элементы в исходном порядке).
[C++] class TextView
{
// Do not change the implementation of the class
int x = 0, y = 0;
int width = 1, height = 1;
public:
void GetOrigin(int& x, int& y);
void SetOrigin(int x, int y);
void GetSize(int& width, int& height);
void SetSize(int width, int height);
};
void TextView::GetOrigin(int& x, int& y)
{
x = this->x;
y = this->y;
}
void TextView::SetOrigin(int x, int y)
{
this->x = x;
this->y = y;
}
void TextView::GetSize(int& width, int& height)
{
width = this->width;
height = this->height;
}
void TextView::SetSize(int width, int height)
{
this->width = width;
this->height = height;
}
class Shape
{
public:
virtual string GetInfo() = 0;
virtual void MoveBy(int a, int b) = 0;
virtual ~Shape()
{
Show("Shape");
}
};
// Implement the RectShape and TextShape descendant classes
[C#] public class TextView
{
// Do not change the implementation of the class
int x, y;
int width = 1, height = 1;
public void GetOrigin(out int x, out int y)
{
x = this.x;
y = this.y;
}
public void SetOrigin(int x, int y)
{
this.x = x;
this.y = y;
}
public void GetSize(out int width, out int height)
{
width = this.width;
height = this.height;
}
public void SetSize(int width, int height)
{
this.width = width;
this.height = height;
}
}
public abstract class Shape
{
public abstract string GetInfo();
public abstract void MoveBy(int a, int b);
}
// Implement the RectShape and TextShape descendant classes
[Java] class TextView {
// Do not change the implementation of the class
private int x = 0,y = 0;
private int width = 1,height = 1;
public int[] getOrigin() {
int res[] = {this.x,this.y};
return res;
}
public void setOrigin(int x,int y) {
this.x = x;
this.y = y;
}
public int[] getSize() {
int res[] = {this.width,this.height};
return res;
}
public void setSize(int width,int height) {
this.width = width;
this.height = height;
}
}
abstract class Shape {
public abstract String getInfo();
public abstract void moveBy(int a,int b);
}
// Implement the RectShape and TextShape descendant classes
[Python] class TextView:
# Do not change the implementation of the class
def __init__(self):
self.__x = 0
self.__y = 0
self.__width = 1
self.__height = 1
def getOrigin(self):
return [self.__x, self.__y]
def setOrigin(self, x, y):
self.__x = x
self.__y = y
def getSize(self):
return [self.__width, self.__height]
def setSize(self, width, height):
self.__width = width
self.__height = height
class RectShape:
def __init__(self, x1, y1, x2, y2):
pass
# Implement the "constructor"
def getInfo(self):
pass
# Implement the method
def moveBy(self, a, b):
pass
# Implement the method
# Implement the TextShape class
[Ruby] class TextView
# Do not change the implementation of the class
def initialize
@x = 0
@y = 0
@width = 1
@height = 1
end
def getOrigin
return @x, @y
end
def setOrigin(x,y)
@x = x
@y = y
end
def getSize
return @width, @height
end
def setSize(width,height)
@width = width
@height = height
end
end
class RectShape
def initialize(x1,y1,x2,y2)
# Implement the "constructor"
end
def getInfo
# Implement the method
end
def moveBy(a,b)
# Implement the method
end
end
# Implement the TextShape class
[Julia] # Do not change the implementation
# of the TextView struct
# and its associated functions
mutable struct TextView
x::Int
y::Int
width::Int
height::Int
end
TextView() = TextView(0, 0, 1, 1)
function getOrigin(t::TextView)
t.x,t.y
end
function setOrigin!(t::TextView, x::Int, y::Int)
t.x = x
t.y = y
end
function getSize(t::TextView)
t.width, t.height
end
function setSize!(t::TextView, width::Int, height::Int)
t.width = width
t.height = height
end
abstract type Shape
end
# Implement the RectShape and TextShape descendant structs
[PascalABC.NET] type
// Не изменяйте реализацию класса
TextView = class
private
x, y: integer;
width: integer := 1;
height: integer := 1;
public
procedure GetOrigin(var x, y: integer);
begin
x := self.x;
y := self.y;
end;
procedure SetOrigin(x, y: integer);
begin
self.x := x;
self.y := y;
end;
procedure GetSize(var width, height: integer);
begin
width := self.width;
height := self.height;
end;
procedure SetSize(width, height: integer);
begin
self.width := width;
self.height := height;
end;
end;
Shape = abstract class
public
function GetInfo: string; abstract;
procedure MoveBy(a, b: integer); abstract;
end;
// Реализуйте классы-потомки RectShape и TextShape
OOP2Struc4°. Adapter (Адаптер) — структурный паттерн. Задание 4. Выполнить предыдущее задание (см. OOP2Struc3), реализовав TextShape как адаптер класса. Для языков, не поддерживающих множественное наследование, преобразовать абстрактный класс Shape в интерфейс IShape и добавить его к классам RectShape и TextShape (для хранения исходных данных в этом случае надо использовать коллекцию с элементами интерфейсного типа IShape).
[C++] class TextView
{
// Do not change the implementation of the class
int x = 0, y = 0;
int width = 1, height = 1;
public:
void GetOrigin(int& x, int& y);
void SetOrigin(int x, int y);
void GetSize(int& width, int& height);
void SetSize(int width, int height);
};
void TextView::GetOrigin(int& x, int& y)
{
x = this->x;
y = this->y;
}
void TextView::SetOrigin(int x, int y)
{
this->x = x;
this->y = y;
}
void TextView::GetSize(int& width, int& height)
{
width = this->width;
height = this->height;
}
void TextView::SetSize(int width, int height)
{
this->width = width;
this->height = height;
}
class Shape
{
public:
virtual string GetInfo() = 0;
virtual void MoveBy(int a, int b) = 0;
virtual ~Shape()
{
Show("Shape");
}
};
// Implement the RectShape and TextShape classes;
// for the TextShape class, use multiple inheritance
// (from the Shape and TextView classes)
[C#] public class TextView
{
// Do not change the implementation of the class
int x, y;
int width = 1, height = 1;
public void GetOrigin(out int x, out int y)
{
x = this.x;
y = this.y;
}
public void SetOrigin(int x, int y)
{
this.x = x;
this.y = y;
}
public void GetSize(out int width, out int height)
{
width = this.width;
height = this.height;
}
public void SetSize(int width, int height)
{
this.width = width;
this.height = height;
}
}
public abstract class Shape
{
// Convert this abstract class into the IShape interface
public abstract string GetInfo();
public abstract void MoveBy(int a, int b);
}
// Implement the RectShape and TextShape classes.
// These classes must implement the IShape interface;
// the TextShape class must be a descendant of the TextView class
[Java] class TextView {
// Do not change the implementation of the class
private int x = 0,y = 0;
private int width = 1,height = 1;
public int[] getOrigin() {
int res[] = {this.x,this.y};
return res;
}
public void setOrigin(int x,int y) {
this.x = x;
this.y = y;
}
public int[] getSize() {
int res[] = {this.width,this.height};
return res;
}
public void setSize(int width,int height) {
this.width = width;
this.height = height;
}
}
abstract class Shape {
// Convert this abstract class into the IShape interface
public abstract String getInfo();
public abstract void moveBy(int a,int b);
}
// Implement the RectShape and TextShape classes.
// These classes must implement the IShape interface;
// the TextShape class must be a descendant of the TextView class
[Python] class TextView:
# Do not change the implementation of the class
def __init__(self):
self.__x = 0
self.__y = 0
self.__width = 1
self.__height = 1
def getOrigin(self):
return [self.__x, self.__y]
def setOrigin(self, x, y):
self.__x = x
self.__y = y
def getSize(self):
return [self.__width, self.__height]
def setSize(self, width, height):
self.__width = width
self.__height = height
class RectShape:
def __init__(self, x1, y1, x2, y2):
pass
# Implement the "constructor"
def getInfo(self):
pass
# Implement the method
def moveBy(self, a, b):
pass
# Implement the method
# Implement the TextShape class;
# the TextShape class must be
# a descendant of the TextView class
[Ruby] class TextView
# Do not change the implementation of the class
def initialize
@x = 0
@y = 0
@width = 1
@height = 1
end
def getOrigin
return @x, @y
end
def setOrigin(x,y)
@x = x
@y = y
end
def getSize
return @width, @height
end
def setSize(width,height)
@width = width
@height = height
end
end
class RectShape
def initialize(x1,y1,x2,y2)
# Implement the "constructor"
end
def getInfo
# Implement the method
end
def moveBy(a,b)
# Implement the method
end
end
# Implement the TextShape class;
# the TextShape class must be
# a descendant of the TextView class
Указание (Julia). Поскольку в Julia не могут наследоваться конкретные типы, адаптер класса в этом языке реализовать нельзя. Можно решить задачу, используя адаптер объекта (т. е. тем же способом, что и задачу OOP2Struc3).
[Julia] # Do not change the implementation
# of the TextView struct
# and its associated functions
mutable struct TextView
x::Int
y::Int
width::Int
height::Int
end
TextView() = TextView(0, 0, 1, 1)
function getOrigin(t::TextView)
t.x,t.y
end
function setOrigin!(t::TextView, x::Int, y::Int)
t.x = x
t.y = y
end
function getSize(t::TextView)
t.width, t.height
end
function setSize!(t::TextView, width::Int, height::Int)
t.width = width
t.height = height
end
abstract type Shape
end
# Implement the RectShape and TextShape descendant structs
[PascalABC.NET] type
// Не изменяйте реализацию класса
TextView = class
private
x, y: integer;
width: integer := 1;
height: integer := 1;
public
procedure GetOrigin(var x, y: integer);
begin
x := self.x;
y := self.y;
end;
procedure SetOrigin(x, y: integer);
begin
self.x := x;
self.y := y;
end;
procedure GetSize(var width, height: integer);
begin
width := self.width;
height := self.height;
end;
procedure SetSize(width, height: integer);
begin
self.width := width;
self.height := height;
end;
end;
type
// Преобразуйте абстрактный класс в интерфейс IShape
Shape = abstract class
function GetInfo: string; abstract;
procedure MoveBy(a, b: integer); abstract;
end;
// Реализуйте классы RectShape и TextShape.
// Эти классы должны реализовывать интерфейс IShape;
// класс TextShape должен быть потомком класса TextView
OOP2Struc5°. Composite (Компоновщик) — структурный паттерн. Частота использования: выше средней. Назначение: компонует объекты в древовидные структуры для представления иерархий «часть-целое». Позволяет клиентам единообразно трактовать индивидуальные и составные объекты. Участники: • Component (Компонент) — объявляет интерфейс для компонуемых объектов; предоставляет подходящую реализацию операций по умолчанию, общую для всех классов; объявляет интерфейс для доступа к дочерним компонентам и управления ими; • Leaf (Лист) — представляет листовой узел композиции, не имеющий потомков; • Composite (Составной объект) — хранит дочерние компоненты составного узла; реализует относящиеся к управлению потомками операции в интерфейсе класса Component; • Client (Клиент) — манипулирует объектами композиции через интерфейс Component. Объекты-листы и составные объекты могут также содержать дополнительные методы, определяющие их особое поведение, однако паттерн Composite ориентирован прежде всего на применение тех методов, которые являются общими у всех объектов композиции (хотя и реализуются по-разному для листов и составных объектов). Задание 1. Реализовать иерархию классов, включающую абстрактный класс Component с методами AddComponent и Operation и конкретные классы Leaf и Composite. Метод AddComponent с параметром-ссылкой c типа Component добавляет компонент c в набор дочерних компонентов (имеет смысл только для класса Composite; для класса Leaf не выполняет никаких действий), метод Operation возвращает строковое представление данного компонента и всех его потомков (при наличии). В классе Component метод AddComponent не выполняет никаких действий, а метод Operation является абстрактным. Классы Composite и Leaf содержат символьное поле data; метод Operation класса Leaf возвращает это поле (преобразованное к строковому типу), метод Operation класса Composite возвращает строку, начинающуюся с символа data, после которого стоит открывающая круглая скобка «(», затем указываются данные, полученные методом Operation для каждого дочернего компонента (без пробелов), а затем указывается закрывающая круглая скобка «)». Класс Composite хранит свои дочерние компоненты в структуре данных children (например, массиве) с элементами-ссылками типа Component (можно считать, что любой объект типа Composite содержит не более 15 дочерних компонентов). Конструктор классов Leaf и Composite содержит один символьный параметр, которым инициализируется поле data. Тестирование разработанной системы классов. Дано целое число N (≤ 15) и набор из N символов, являющихся прописными или строчными латинскими буквами. Создать набор данных comp (например, массив) из N ссылок на объекты Component и заполнить этот набор, создавая для каждой прописной буквы из исходного набора символов объект типа Composite, а для каждой строчной буквы — объект типа Leaf, и указывая эту букву в качестве параметра конструктора объекта. Кроме того, дан набор из N целых чисел, определяющих связи между объектами набора comp: число с индексом K (K = 0, …, N − 1) определяет индекс родительского объекта для объекта с индексом K (при этом гарантируется, что родительский объект обязательно имеет тип Composite). Если объект не имеет родителя, то соответствующий элемент в исходном наборе чисел равен −1. Если несколько объектов имеют общего родителя, то они должны добвляться к нему в порядке их следования в наборе comp. Используя данные из исходного набора чисел и вызывая метод AddComponent для требуемых объектов типа Composite, установить связи между объектами набора comp. Затем, перебирая объекты из набора comp с порядке возрастания их индексов, вызвать для каждого из них метод Operation и вывести возвращаемое этим методом строковое описание объекта.
[C++] class Component
{
public:
virtual void AddComponent(shared_ptr<Component> c) {}
virtual string Operation() = 0;
virtual ~Component()
{
Show("Component");
}
};
// Implement the Leaf and Composite descendant classes
[C#] public abstract class Component
{
public virtual void AddComponent(Component c) {}
public abstract string Operation();
}
// Implement the Leaf and Composite descendant classes
[Java] abstract class Component {
public void addComponent(Component c) {}
public abstract String operation();
}
// Implement the Leaf and Composite descendant classes
[Python] class Composite:
def __init__(self, data):
pass
# Implement the "constructor"
def addComponent(self, c):
pass
# Implement the method
def operation(self):
pass
# Implement the method
# Implement the Leaf class
[Ruby] class Composite
def initialize(data)
# Implement the "constructor"
end
def addComponent(c)
# Implement the method
end
def operation
# Implement the method
end
end
# Implement the Leaf class
[Julia] abstract type Component
end
struct Leaf <: Component
data::Char
end
struct Composite <: Component
data::Char
children::Vector{Component}
end
Composite(data::Char) = Composite(data, Vector{Component}())
# Implement the required functions
# for the Leaf and Composite structs
# (addComponent and operation)
[PascalABC.NET] type
Component = abstract class
public
procedure AddComponent(c: Component); virtual;
begin
end;
function Operation: string; abstract;
end;
// Реализуйте классы-потомки Leaf и Composite
OOP2Struc6°. Composite (Компоновщик) — структурный паттерн. Задание 2. Реализовать иерархию классов, включающую абстрактный класс Device (устройство) с методами Add, GetName и GetTotalPrice и конкретные классы SimpleDevice (простое устройство) и CompoundDevice (составное устройство). Метод Add с параметром-ссылкой d типа Device добавляет устройство d в набор дочерних устройств (имеет смысл только для класса CompoundDevice; для класса SimpleDevice не выполняет никаких действий), метод GetName возвращает строковое имя данного устройства, метод GetTotalPrice возвращает стоимость данного устройства и всех его потомков (целое число). В классе Device метод Add не выполняет никаких действий, а методы GetName и GetTotalPrice являются абстрактными. Классы SimpleDevice и CompoundDevice содержат строковое поле name и целочисленное поле price; класс CompoundDevice хранит свои дочерние устройства в виде массива или другой структуры данных с элементами-ссылками типа Device (можно считать, что любой объект типа CompoundDevice содержит не более 15 дочерних устройств). Конструктор классов SimpleDevice и CompoundDevice содержит параметры name и price, которые используются для инициализации одноименных полей. Тестирование разработанной системы классов. Дано целое число N (≤ 15) и N пар вида (name, price), где name — некоторая строка, а price — положительное целое число. Первый символ строки name является латинской буквой; если буква заглавная, то строка определяет составное устройство, а если строчная — простое устройство. Кроме того, дан набор из N целых чисел, определяющих связи между исходными устройствами: число с индексом K (K = 0, …, N − 1) определяет индекс родительского устройства для исходного устройства с индексом K (при этом гарантируется, что родительское устройство обязательно имеет тип CompoundDevice). Если устройство не имеет родителя, то соответствующий элемент в исходном наборе чисел равен −1. Перебирая созданные устройства в порядке, соответствующем порядку их характеристик (name, price) и используя методы GetName и GetTotalPrice, вывести для каждого устройства название и полную стоимость.
[C++] class Device
{
public:
virtual void Add(shared_ptr<Device> d) {}
virtual string GetName() = 0;
virtual int GetTotalPrice() = 0;
virtual ~Device()
{
Show("Device");
}
};
// Implement the SimpleDevice
// and CompoundDevice descendant classes
[C#] public abstract class Device
{
public virtual void Add(Device d) {}
public abstract string GetName();
public abstract int GetTotalPrice();
}
// Implement the SimpleDevice
// and CompoundDevice descendant classes
[Java] abstract class Device {
public void add(Device d) {}
public abstract String getName();
public abstract int getTotalPrice();
}
// Implement the SimpleDevice
// and CompoundDevice descendant classes
[Python] class SimpleDevice:
def __init__(self, name, price):
pass
# Implement the "constructor"
def getName(self):
pass
# Implement the method
def getTotalPrice(self):
pass
# Implement the method
# Implement the CompoundDevice class
[Ruby] class SimpleDevice
def initialize(name,price)
# Implement the "constructor"
end
def getName
# Implement the method
end
def getTotalPrice
# Implement the method
end
end
# Implement the CompoundDevice class
[Julia] abstract type Device
end
struct SimpleDevice <: Device
name::String
price::Int
end
struct CompoundDevice <: Device
name::String
price::Int
children::Vector{Device}
end
CompoundDevice(name::String, price::Int) =
CompoundDevice(name, price, Vector{Device}())
# Implement the required functions
# for the SimpleDevice and CompoundDevice structs
# (add!, getName and getTotalPrice)
[PascalABC.NET] type
Device = abstract class
public
procedure Add(d: Device); virtual;
begin
end;
function GetName: string; abstract;
function GetTotalPrice: integer; abstract;
end;
// Реализуйте классы-потомки SimpleDevice и CompoundDevice
OOP2Struc7°. Decorator (Декоратор) — структурный паттерн. Известен также под именем Wrapper (Обертка). Частота использования: средняя. Назначение: динамически добавляет объекту новые возможности, приводящие к изменению его состояния и/или поведения. Является гибкой альтернативой порождению подклассов с целью расширения функциональности. Участники: • Component (Компонент) — определяет интерфейс для объектов, к которым могут быть динамически добавлены новые возможности; • ConcreteComponent (Конкретный компонент) — определяет объект, к которому могут быть добавлены новые возможности; • Decorator (Декоратор) — хранит ссылку на объект Component и определяет интерфейс, соответствующий интерфейсу Component; • ConcreteDecoratorA и ConcreteDecoratorB (Конкретные декораторы) — добавляют к компоненту новые возможности, изменяющие его состояние и/или поведение. Задание 1. Реализовать иерархию классов, которая включает абстрактный класс Component с абстрактным методом Operation (не имеет параметров, возвращает строку), абстрактный класс Decorator, который является потомком класса Component и содержит защищенное поле comp — ссылку на объект типа Component, и конкретные классы ConcreteComponent (потомок класса Component), ConcreteDecoratorA и ConcreteDecoratorB (потомки класса Decorator). Класс ConcreteComponent содержит строковое поле text, которое инициализируется в конструкторе с помощью одноименного параметра. Метод Operation класса ConcreteComponent возвращает строку text. Классы ConcreteDecoratorA и ConcreteDecoratorB имеют поле comp, являющееся ссылкой на объект типа Component; класс ConcreteDecoratorA содержит также символьное поле ch. В конструкторах классов ConcreteDecoratorA и ConcreteDecoratorB поля инициализируются с помощью одноименных параметров. Метод Operation конкретного декоратора A возвращает строку, полученную путем добавления символа ch перед и после текста, возвращенного методом Operation объекта comp. Метод Operation конкретного декоратора B возвращает строку, полученную путем добавления символа «(» перед текстом, возвращенным методом Operation объекта comp, и символа «)» после этого текста. Таким образом, каждый декоратор изменяет поведение метода Operation исходного объекта Component, добавляя к возвращаемому значению дополнительный префикс и суффикс; при этом декоратор A корректирует состояние исходного объекта, путем добавления к нему поля ch. Тестирование разработанной системы классов. Дан символ C, целое число N (≤ 9) и N пар строк (S, D), причем строка S является непустой, а строка D содержит только буквы «A» и «B» и может быть пустой. Создать набор из N объектов типа Component, формируя каждый элемент этого набора на основе соответствующей пары строк (S, D) следующим образом: вначале создать объект типа ConcreteComponent, вызвав его конструктор с параметром S, а затем последовательно применять к результирующему объекту декораторы A или В, причем количество и порядок декораторов определяется строкой D (например, в случае строки «AAB» к исходному объекту типа ConcreteComponent надо последовательно применить декораторы A, A и B). В качестве второго параметра конструктора всех декораторов A использовать исходный символ C. Перебирая созданный набор из N объектов в обратном порядке, вызвать для каждого из них метод Operation и вывести его возвращаемое значение.
[C++] class Component
{
public:
virtual string Operation() = 0;
virtual ~Component()
{
Show("Component");
}
};
// Implement the ConcreteComponent descendant class
class Decorator : public Component
{
protected:
shared_ptr<Component> comp;
public:
~Decorator() override
{
Show("Decorator");
}
};
// Implement the ConcreteDecoratorA
// and ConcreteDecoratorB descendant classes
[C#] public abstract class Component
{
public abstract string Operation();
}
// Implement the ConcreteComponent descendant class
public abstract class Decorator : Component
{
protected Component comp;
}
// Implement the ConcreteDecoratorA
// and ConcreteDecoratorB descendant classes
[Java] abstract class Component {
public abstract String operation();
}
// Implement the ConcreteComponent descendant class
abstract class Decorator extends Component {
protected Component comp;
}
// Implement the ConcreteDecoratorA
// and ConcreteDecoratorB descendant classes
[Python] class ConcreteComponent:
def __init__(self, text):
pass
# Implement the "constructor"
def operation(self):
pass
# Implement the method
class ConcreteDecoratorA:
def __init__(self, comp, ch):
pass
# Implement the "constructor"
def operation(self):
pass
# Implement the method
# Implement the ConcreteDecoratorB class
[Ruby] class ConcreteComponent
def initialize(text)
# Implement the "constructor"
end
def operation
# Implement the method
end
end
class ConcreteDecoratorA
def initialize(comp, ch)
# Implement the "constructor"
end
def operation
# Implement the method
end
end
# Implement the ConcreteDecoratorB class
Указание (Julia). Поскольку в Julia конкретные типы не могут наследоваться, а по условию задачи тип Decorator должен быть типом-предком, его надо описать как абстрактный тип, не содержащий полей. Поэтому поле comp необходимо добавить в конкретные структуры ConcreteDecoratorA и ConcreteDecoratorB .
[Julia] abstract type Component
end
abstract type Decorator <: Component
end
# Implement the ConcreteComponent, ConcreteDecoratorA,
# ConcreteDecoratorB descendant structs
# and their associated functions
[PascalABC.NET] type
Component = abstract class
public
function Operation: string; abstract;
end;
// Реализуйте класс-потомок ConcreteComponent
Decorator = abstract class(Component)
protected
comp: Component;
end;
// Реализуйте классы-потомки ConcreteDecoratorA и ConcreteDecoratorB
OOP2Struc8°. Decorator (Декоратор) — структурный паттерн. Задание 2. Реализовать иерархию классов, включающую абстрактный класс Function с абстрактными методами GetName (без параметров; возвращает строку) и GetValue(x) (с целочисленным параметром; возвращает целое число) и пять конкретных классов — потомков Function: FX, FDouble, FTriple, FSquare, FCube. Класс FX не имеет полей; его метод GetName возвращает строку «X», а функция GetValue(x) возвращает свой параметр x. Остальные конкретные классы являются декораторами; все они содержат ссылочное поле f типа Function и конструктор с параметром-ссылкой f типа Function, который инициализирует это поле. Метод GetName данных классов вызывает метод GetName класса f и возвращает его результат, снабженный дополнительным префиксом и суффиксом: «2*(» и «)» для FDouble, «3*(» и «)» для FTriple, «(» и «)^2» для FSquare, «(» и «)^3» для FCube. Метод GetValue(x) данных классов вызывает метод GetValue(x) класса f и возвращает его результат, преобразованный следующим образом: результат умножается на 2 для FDouble, умножается на 3 для FTriple, возводится в квадрат для FSquare, возводится в куб для FCube. Тестирование разработанной системы классов. Дано целое число N (≤ 10) и набор из N строк; каждая строка содержит комбинацию из букв «D», «T», «S» и «C» и может быть пустой. Кроме того, даны два целых числа X1 и X2. Создать набор из N объектов типа Function, формируя каждый элемент следующим образом: вначале создать объект типа FX, а затем последовательно применять к результирующему объекту декораторы FDouble, FTriple, FSquare, FCube, причем количество и порядок декораторов определяется символами соответствующей строки из исходного набора (например, в случае строки «TCSS» к исходному объекту типа FX надо последовательно применить декораторы FTriple, FCube и два декоратора FSquare). Перебирая созданный набор из N объектов в исходном порядке, вызвать для каждого из них метод GetName, метод GetValue с параметром X1, метод GetValue с параметром X2 и вывести их возвращаемые значения.
[C++] class Function
{
public:
virtual string GetName() = 0;
virtual int GetValue(int x) = 0;
virtual ~Function()
{
Show("Function");
}
};
// Implement the FX, FDouble, FTriple, FSquare
// and FCube descendant classes
[C#] public abstract class Function
{
public abstract string GetName();
public abstract int GetValue(int x);
}
// Implement the FX, FDouble, FTriple, FSquare
// and FCube descendant classes
[Java] abstract class Function {
public abstract String getName();
public abstract int getValue(int x);
}
// Implement the FX, FDouble, FTriple, FSquare
// and FCube descendant classes
[Python] class FX:
def getName(self):
pass
# Implement the method
def getValue(self, x):
pass
# Implement the method
# Implement the FDouble, FTriple, FSquare
# and FCube classes
[Ruby] class FX
def getName
# Implement the method
end
def getValue(x)
# Implement the method
end
end
# Implement the FDouble, FTriple, FSquare
# and FCube classes
[Julia] abstract type Function
end
struct FX <: Function
end
struct FDouble <: Function
f::Function
end
# Implement the FTriple, FSquare
# and FCube descendant structs
function getName(::FX)
"X"
end
function getValue(::FX, x::Int)
x
end
function getName(fd::FDouble)
"2*(" * getName(fd.f) * ")"
end
function getValue(fd::FDouble, x::Int)
2 * getValue(fd.f, x)
end
# Implement the getName and getValue functions
# for the FTriple, FSquare and FCube structs
Примечание (PascalABC.NET). В PascalABC.NET нельзя использовать идентификатор Function в качестве имени класса, так как он является служебным словом, поэтому при определении абстрактного класса необходимо использовать другой идентификатор (например, Func ).
[PascalABC.NET] type
Func = abstract class
public
function GetName: string; abstract;
function GetValue(x: integer): integer; abstract;
end;
// Реализуйте классы-потомки FX, FDouble,
// FTriple, FSquare и FCube
Proxy, Bridge, Flyweight
OOP2Struc9°. Proxy (Заместитель) — структурный паттерн. Известен также под именем Surrogate (Суррогат). Частота использования: выше средней. Назначение: является суррогатом другого объекта и контролирует доступ к нему. Участники: • Subject (Субъект) — определяет общий для RealSubject и Proxy интерфейс, так что класс Proxy можно использовать везде, где ожидается RealSubject; • RealSubject (Реальный субъект) — определяет реальный объект, представленный заместителем; • Proxy (Заместитель) — хранит ссылку, которая позволяет заместителю обратиться к реальному субъекту; контролирует доступ к реальному субъекту и может отвечать за его создание и удаление; прочие обязанности зависят от вида заместителя (удаленный заместитель отвечает за отправление запроса реальному субъекту в другом адресном пространстве; виртуальный заместитель может отложить создание реального субъекта и вызывать некоторые его методы самостоятельно; защищающий заместитель проверяет, имеет ли вызывающий объект необходимые для выполнения запроса права; кэширующий заместитель обеспечивает временное хранение результатов ранее выполненных высокозатратных запросов и совместный доступ к этим результатам; имеются и другие виды заместителей). Задание 1. Реализовать иерархию классов, включающую абстрактный класс Subject с абстрактными методами RequestA, RequestB, RequestC, RequestD и конкретные классы RealSubject и Proxy — потомки класса Subject. Указанные методы не имеют параметров и возвращают строку. Класс RealSubject не имеет полей, его методы возвращают строки «A (Real)», «B (Real)», «C (Real)», «D (Real)». Класс Proxy является заместителем класса RealSubject, комбинирующим черты виртуального и защищающего заместителя. Предполагается, что запросы A и B являются простыми и могут быть реализованы в самом заместителе, в то время как запросы C и D являются сложными и доступны только в классе RealSubject. Кроме того, запросы A и C являются безопасными, а запросы B и D — потенциально опасными, и в некоторых ситуациях их целесообразно заблокировать. Поэтому класс Proxy содержит два логических поля: deferredMode (отложенный режим) и protectedMode (защищенный режим), которые задаются в его конструкторе с помощью одноименных параметров и определяют его поведение по отношению к реальному субъекту. Кроме того, класс Proxy содержит поле rsubj — ссылку на объект типа RealSubject. Если поле deferredMode равно False, то объект RealSubject создается в конструкторе класса Proxy (и связывается со ссылкой rsubj), если deferredMode равно True, то начальное значение ссылки rsubj является пустым. Если поле protectedMode равно False и при этом ссылка rsubj не является пустой, то все запросы переадресуются объекту, на который ссылается rsubj. Если protectedMode равно False, а ссылка rsubj является пустой, то простые запросы A и B выполняются самим объектом Proxy и при этом возвращаются строки «A (Proxy)» и «B (Proxy)», а в случае вызова запроса C или D объект RealSubject создается и связывается со ссылкой rsubj, после чего этот запрос переадресуется ему. Если protectedMode равно True, то выполнение запросов A и C не отличается от ранее описанного, а при попытке вызова запросов B и D они отменяются (независимо от значения ссылки rsubj), причем соответствующие методы возвращают строки «B denied» и «D denied». Тестирование разработанной системы классов. Дан набор из трех целых чисел, принимающих значения от −1 до 3, и строка, содержащая только символы «A», «B», «C», «D» — имена запросов. Создать массив из трех ссылочных элементов типа Subject, инициализировав элементы конкретными объектами в зависимости от значения исходных чисел: −1 — RealSubject, 0 — Proxy(False, False), 1 — Proxy(True, False), 2 — Proxy(False, True), 3 — Proxy(True, True). Для каждого из созданных объектов выполнить набор запросов, определяемый исходной строкой, и вывести результат, возвращаемый каждым запросом.
[C++] class Subject
{
public:
virtual string RequestA() = 0;
virtual string RequestB() = 0;
virtual string RequestC() = 0;
virtual string RequestD() = 0;
virtual ~Subject()
{
Show("Subject");
}
};
// Implement the RealSubject and Proxy descendant classes
[C#] public abstract class Subject
{
public abstract string RequestA();
public abstract string RequestB();
public abstract string RequestC();
public abstract string RequestD();
}
// Implement the RealSubject and Proxy descendant classes
[Java] abstract class Subject {
public abstract String requestA();
public abstract String requestB();
public abstract String requestC();
public abstract String requestD();
}
// Implement the RealSubject and Proxy descendant classes
[Python] class RealSubject:
def requestA(self):
pass
# Implement the method
def requestB(self):
pass
# Implement the method
def requestC(self):
pass
# Implement the method
def requestD(self):
pass
# Implement the method
# Implement the Proxy class
[Ruby] class RealSubject
def requestA
# Implement the method
end
def requestB
# Implement the method
end
def requestC
# Implement the method
end
def requestD
# Implement the method
end
end
# Implement the Proxy class
Указание (Julia). Для возможности присваивания полю rsubj структуры Proxy значения nothing его тип надо определить как Union{RealSubject, Nothing} .
[Julia] abstract type Subject
end
struct RealSubject <: Subject
end
mutable struct Proxy <: Subject
deferredMode::Bool
protectedMode::Bool
rsubj::Union{RealSubject, Nothing}
end
# Implement the constructor for the Proxy struct
# and the required functions
# for the RealSubject and Proxy structs
# (requestA, requestB, requestC, requestD)
[PascalABC.NET] type
Subject = abstract class
public
function RequestA: string; abstract;
function RequestB: string; abstract;
function RequestC: string; abstract;
function RequestD: string; abstract;
end;
// Реализуйте классы-потомки RealSubject и Proxy
OOP2Struc10°. Bridge (Мост) — структурный паттерн. Известен также под именем Handle/Body (Описатель/Тело). Частота использования: средняя. Назначение: отделяет абстракцию от ее реализации так, чтобы то и другое можно было изменять независимо. Участники: • Abstraction (Абстракция) — определяет интерфейс абстракции; хранит ссылку на объект типа Implementor; • RefinedAbstraction (Уточненная абстракция) — расширяет интерфейс, определенный абстракцией Abstraction; • Implementor (Реализатор) — определяет интерфейс для классов реализации; не обязан точно соответствовать интерфейсу класса Abstraction (обычно предоставляет только примитивные операции, в то время как класс Abstraction определяет операции более высокого уровня, базирующиеся на этих примитивах); • ConcreteImplementorA и ConcreteImplementorB (Конкретные реализаторы) — содержат конкретную реализацию интерфейса класса Implementor. Задание 1. Реализовать иерархию классов-реализаторов, содержащую абстрактного реализатора Implementor и два конкретных реализатора ConcreteImplementorA и ConcreteImplementorB. Классы отвечают за представление горизонтальных линий и текста и включают методы DrawLine(size) и DrawText(text) (аналоги метода OperationImp на диаграмме классов), которые возвращают строковые значения. Параметр size определяет размер линии (в символах), параметр text — выводимый текст. В классе Implementor методы DrawLine и DrawText являются абстрактными. Конкретный реализатор A представляет линию в виде набора символов «−», а текст отображает в нижнем регистре. Конкретный реализатор B представляет линию в виде набора символов «=», а текст отображает в верхнем регистре. Оба конкретных реализатора имеют конструкторы без параметров, не выполняющие дополнительных действий. Реализовать класс Abstraction, предназначенный для отображения и корректировки строки заголовка. Конструктор класса принимает параметры imp типа Implementor (ссылку на используемый реализатор) и size целого типа (размер заголовка). Класс также содержит метод Show без параметров (аналог метода Operation на диаграмме классов), возвращающий строку-заголовок, и метод SetSize(n), задающий размер заголовка равным значению n (целое неотрицательное число). Класс Abstraction реализует простейший вариант заголовка, представляющий собой линию указанного размера. Реализовать класс RefinedAbstraction, который является усовершенствованным вариантом класса Abstraction и позволяет включать в заголовок текст. Конструктор класса RefinedAbstraction содержит, кроме параметров imp и size, имеющих тот же смысл, что и для конструктора класса Abstraction, строковый параметр caption. Заголовок формируется следующим образом: вначале указывается линия размера 1, затем строка caption, затем линия такого размера, чтобы суммарный размер заголовка был равен size. Для малых значений size строка caption может урезаться справа. Переопределить нужным образом метод Show в классе RefinedAbstraction. Тестирование разработанной системы классов. Дано целое положительное число size (начальный размер заголовка) и строка caption (необязательный элемент заголовка). Также даны пять целых положительных чисел (новые размеры заголовков). Создать экземпляры классов Abstraction и RefinedAbstraction с указанными параметрами и каждым из конкретных реализаторов A и B и вывести соответствующие заголовки методом Show. Затем, используя каждый из новых размеров, изменить размер каждого заголовка и вывести измененные заголовки. Порядок вывода заголовков для каждого размера: Abstraction с реализатором A, Abstraction с реализатором B, RefinedAbstraction с реализатором A, RefinedAbstraction с реализатором B. Для хранения созданных объектов использовать массив из четырех ссылочных элементов типа Abstraction.
[C++] class Implementor
{
public:
virtual string DrawLine(int size) = 0;
virtual string DrawText(string text) = 0;
virtual ~Implementor()
{
Show("Implementor");
}
};
// Implement the ConcreteImplementorA
// and ConcreteImplementorB descendant classes
class Abstraction
{
protected:
int size;
shared_ptr<Implementor> imp;
public:
Abstraction(shared_ptr<Implementor> imp, int size) :
imp(imp), size(size) {}
virtual ~Abstraction()
{
Show("Abstraction");
}
// Complete the implementation of the class Abstraction
};
// Implement the RefinedAbstraction descendant class
[C#] public abstract class Implementor
{
public abstract string DrawLine(int size);
public abstract string DrawText(string text);
}
// Implement the ConcreteImplementorA
// and ConcreteImplementorB descendant classes
public class Abstraction
{
protected int size;
protected Implementor imp;
public Abstraction(Implementor imp, int size)
{
this.imp = imp;
this.size = size;
}
// Complete the implementation of the class
}
// Implement the RefinedAbstraction descendant class
[Java] abstract class Implementor {
public abstract String drawLine(int size);
public abstract String drawText(String text);
}
// Implement the ConcreteImplementorA
// and ConcreteImplementorB descendant classes
class Abstraction {
protected int size;
protected Implementor imp;
public Abstraction(Implementor imp,int size) {
this.imp = imp;
this.size = size;
}
// Complete the implementation of the class
}
// Implement the RefinedAbstraction descendant class
[Python] class ConcreteImplementorA:
def drawLine(self, size):
pass
# Implement the method
def drawText(self, text):
pass
# Implement the method
# Implement the ConcreteImplementorB class
class Abstraction:
def __init__(self, imp, size):
self.__imp = imp
self.__size = size
# Complete the implementation of the class
# Implement the RefinedAbstraction descendant class
[Ruby] class ConcreteImplementorA
def drawLine(size)
# Implement the method
end
def drawText(text)
# Implement the method
end
end
# Implement the ConcreteImplementorB class
class Abstraction
def initialize(imp,size)
@imp = imp
@size = size
end
# Complete the implementation of the class
end
# Implement the RefinedAbstraction descendant class
Указание (Julia). Поскольку в Julia конкретные типы не могут наследоваться, а по условию задачи структура RefinedAbstraction должна унаследовать возможности структуры Abstraction , в данном случае можно использовать вместо наследования композицию, добавив в определение структуры RefinedAbstraction поле abstraction::Abstraction . При этом можно не определять общего абстрактного предка для конкретных структур Abstraction и RefinedAbstraction , а вместо этого при создании массива для хранения объектов этих структур использовать выражение Union{Abstraction, RefinedAbstraction} для описания типа элементов массива.
[Julia] abstract type Implementor
end
# Implement the ConcreteImplementorA,
# ConcreteImplementorB descendant structs
# and their associated functions
# (drawLine and drawText)
mutable struct Abstraction
imp::Implementor
size::Int
end
struct RefinedAbstraction
abstraction::Abstraction
caption::String
end
# Implement a constructor for the RefinedAbstraction struct
# and the required functions (setSize! and show)
# for the Abstraction and RefinedAbstraction structs
[PascalABC.NET] type
Implementor = abstract class
public
function DrawLine(size: integer): string; abstract;
function DrawText(text: string): string; abstract;
end;
// Реализуйте классы-потомки ConcreteImplementorA и ConcreteImplementorB
Abstraction = class
protected
size: integer;
imp: Implementor;
public
constructor Create(imp: Implementor; size: integer);
begin
self.imp := imp;
self.size := size;
end;
// Завершите реализацию класса
end;
// Реализуйте класс-потомок RefinedAbstraction
OOP2Struc11°. Flyweight (Приспособленец) — структурный паттерн. Частота использования: низкая. Назначение: применяет разделение (т. е. совместное использование одного и того же экземпляра) для эффективной поддержки множества объектов, сохраняя основную часть их состояния во внешних данных клиента. Участники: • Flyweight (Приспособленец) — объявляет интерфейс, с помощью которого приспособленцы могут получать внешнее состояние или воздействовать на него; • ConcreteFlyweight (Конкретный приспособленец) — реализует интерфейс класса Flyweight и добавляет при необходимости внутреннее состояние (объект класса ConcreteFlyweight должен быть разделяемым, любое сохраняемое им состояние должно быть внутренним, то есть не зависящим от контекста); • UnsharedConcreteFlyweight (Неразделяемый конкретный приспособленец) — реализует интерфейс класса Flyweight, но не является разделяемым; данный класс может сохранять состояние, зависящее от контекста; • FlyweightFactory (Фабрика приспособленцев) — создает объекты-приспособленцы и управляет ими; при запросе клиентом приспособленца объект FlyweightFactory предоставляет существующий экземпляр или создает новый, если готового еще нет; • Client (Клиент) — хранит ссылки на одного или нескольких приспособленцев; вычисляет или хранит внешнее состояние приспособленцев. Задание 1. Реализовать иерархию классов, которая включает абстрактный класс Flyweight с абстрактным методом Operation(state), который имеет логический параметр state и возвращает символьное значение, и конкретные классы ConcreteFlyweight и UnsharedConcreteFlyweight — потомки класса Flyweight. Предполагается, что клиент будет использовать большое количество объектов класса ConcreteFlyweight, поэтому этот класс должен поддерживать разделение. Метод Operation(state) класса ConcreteFlyweight возвращает символ «A», регистр которого зависит от значения state (если state равен True, то используется верхний регистр, если state равен False, то нижний). Конструктор класса ConcreteFlyweight не имеет параметров и не выполняет дополнительных действий. Класс UnsharedConcreteFlyweight не является разделяемым; он хранит дополнительное символьное поле inf, которое инициализируется в конструкторе, имеющем соответствующий символьный параметр. Метод Operation(state) класса UnsharedConcreteFlyweight возвращает символ inf, его регистр определяется параметром state (как в методе Operation класса ConcreteFlyweight). Реализовать класс-фабрику FlyweightFactory, содержащий поле-ссылку cf типа ConcreteFlyweight и поле count целого типа, а также конструктор без параметров (при создании данного объекта поле cf содержит пустую ссылку, а поле count — значение 0). Класс содержит метод GetFlyweight(inf) с символьным параметром и возвращаемым значением-ссылкой типа Flyweight. Если параметр отличен от символа «A» или «a», то фабрика создает и возвращает новый объект типа UnsharedConcreteFlyweight, используя конструктор с параметром inf. Если параметр представляет собой символ «A» или «a», то в случае, если поле cf содержит ссылку на существующий объект типа ConcreteFlyweight, возвращается этот объект, если же поле cf является пустым, то объект типа ConcreteFlyweight создается, сохраняется в поле cf и возвращается методом GetFlyweight. Если в методе GetFlyweight создается новый объект, то поле count увеличивается на 1; таким образом, данное поле хранит общее количество созданных объектов. Значение поля count можно получить с помощью метода GetCount без параметров. Реализовать также класс Client, предназначенный для создания, хранения и обработки наборов объектов типа Flyweight. Класс содержит поля f — объект типа FlyweightFactory и fw — структуру ссылочных данных (например, массив) с элементами типа Flyweight (можно считать, что число элементов fw не превосходит 30). Конструктор класса Client не имеет параметров, в нем создается объект f и инициализируется структура данных fw. Класс содержит методы MakeFlyweights(inf), ShowFlyweights(state) и GetFlyweightCount. Параметр inf метода MakeFlyweights является строкой, определяющей набор создаваемых объектов типа Flyweight (для создания требуемого набора в методе MakeFlyweights в цикле вызывается метод GetFlyweight объекта f с параметром — очередным символом строки inf). Ссылки на созданные объекты сохраняются в структуре fw; при каждом вызове метода MakeFlyweights прежнее содержимое структуры fw очищается. Метод ShowFlyweights(state) возвращает строку, состоящую из символов, возвращаемых методом Operation(state) для каждого элемента структуры данных fw. Метод GetFlyweightCount без параметров возвращает количество объектов, созданных к данному моменту фабрикой f; для этого используется ее метод GetCount. Тестирование разработанной системы классов. Даны пять текстовых строк; длина каждой строки не превосходит 30, большинство символов в этих строках являются символами «A» в верхнем или нижнем регистре. Создать объект типа Client и для каждой исходной строки s вызвать метод MakeFlyweights(s) и вывести значения, возвращаемые методами ShowFlyweights(True), ShowFlyweights(False) и GetFlyweightCount.
[C++] class Flyweight
{
public:
virtual char Operation(bool state) = 0;
virtual ~Flyweight()
{
Show("Flyweight");
}
};
// Implement the ConcreteFlyweight
// and UnsharedConcreteFlyweight descendant classes
// Implement the FlyweightFactory and Client classes
[C#] public abstract class Flyweight
{
public abstract char Operation(bool state);
}
// Implement the ConcreteFlyweight
// and UnsharedConcreteFlyweight descendant classes
// Implement the FlyweightFactory and Client classes
[Java] abstract class Flyweight {
public abstract char operation(boolean state);
}
// Implement the ConcreteFlyweight
// and UnsharedConcreteFlyweight descendant classes
// Implement the FlyweightFactory and Client classes
[Python] class ConcreteFlyweight:
def operation(self, state):
pass
# Implement the method
# Implement the UnsharedConcreteFlyweight class
# Implement the FlyweightFactory and Client classes
[Ruby] class ConcreteFlyweight
def operation(state)
# Implement the method
end
end
# Implement the UnsharedConcreteFlyweight class
# Implement the FlyweightFactory and Client classes
[Julia] abstract type Flyweight
end
struct ConcreteFlyweight <: Flyweight
end
struct UnsharedConcreteFlyweight <: Flyweight
inf::Char
end
mutable struct FlyweightFactory
cf::Union{ConcreteFlyweight, Nothing}
count::Int
end
struct Client
f::FlyweightFactory
fw::Vector{Flyweight}
end
# Implement constructors for the FlyweightFactory
# and Client structs
# Implement all required functions
[PascalABC.NET] type
Flyweight = abstract class
public
function Operation(state: boolean): char; abstract;
end;
// Реализуйте классы-потомки ConcreteFlyweight и UnsharedConcreteFlyweight
// Реализуйте классы FlyweightFactory и Client
|