mirror of
https://github.com/JakubVojvoda/design-patterns-cpp.git
synced 2025-12-17 12:54:36 +03:00
add Builder pattern
This commit is contained in:
149
builder/Builder.cpp
Normal file
149
builder/Builder.cpp
Normal file
@@ -0,0 +1,149 @@
|
|||||||
|
/*
|
||||||
|
* C++ Design Patterns: Builder
|
||||||
|
* Author: Jakub Vojvoda [github.com/JakubVojvoda]
|
||||||
|
* 2016
|
||||||
|
*
|
||||||
|
* Source code is licensed under MIT licence
|
||||||
|
* (for more details see LICENCE)
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <iostream>
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Product
|
||||||
|
* the final object that will be created using Builder
|
||||||
|
*/
|
||||||
|
class Product {
|
||||||
|
public:
|
||||||
|
void makeA(const std::string &part) {
|
||||||
|
partA = part;
|
||||||
|
}
|
||||||
|
void makeB(const std::string &part) {
|
||||||
|
partB = part;
|
||||||
|
}
|
||||||
|
void makeC(const std::string &part) {
|
||||||
|
partC = part;
|
||||||
|
}
|
||||||
|
std::string get() {
|
||||||
|
return (partA + " " + partB + " " + partC);
|
||||||
|
}
|
||||||
|
// ...
|
||||||
|
|
||||||
|
private:
|
||||||
|
std::string partA;
|
||||||
|
std::string partB;
|
||||||
|
std::string partC;
|
||||||
|
// ...
|
||||||
|
};
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Builder
|
||||||
|
* abstract interface for creating products
|
||||||
|
*/
|
||||||
|
class Builder {
|
||||||
|
public:
|
||||||
|
virtual ~Builder() {
|
||||||
|
// ...
|
||||||
|
}
|
||||||
|
|
||||||
|
Product get() {
|
||||||
|
return product;
|
||||||
|
}
|
||||||
|
|
||||||
|
virtual void buildPartA() = 0;
|
||||||
|
virtual void buildPartB() = 0;
|
||||||
|
virtual void buildPartC() = 0;
|
||||||
|
// ...
|
||||||
|
|
||||||
|
protected:
|
||||||
|
Product product;
|
||||||
|
};
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Concrete Builder X and Y
|
||||||
|
* create real products and stores them in the composite structure
|
||||||
|
*/
|
||||||
|
class ConcreteBuilderX : public Builder {
|
||||||
|
public:
|
||||||
|
void buildPartA() {
|
||||||
|
product.makeA("A-X");
|
||||||
|
}
|
||||||
|
void buildPartB() {
|
||||||
|
product.makeB("B-X");
|
||||||
|
}
|
||||||
|
void buildPartC() {
|
||||||
|
product.makeC("C-X");
|
||||||
|
}
|
||||||
|
// ...
|
||||||
|
};
|
||||||
|
|
||||||
|
class ConcreteBuilderY : public Builder {
|
||||||
|
public:
|
||||||
|
void buildPartA() {
|
||||||
|
product.makeA("A-Y");
|
||||||
|
}
|
||||||
|
void buildPartB() {
|
||||||
|
product.makeB("B-Y");
|
||||||
|
}
|
||||||
|
void buildPartC() {
|
||||||
|
product.makeC("C-Y");
|
||||||
|
}
|
||||||
|
// ...
|
||||||
|
};
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Director
|
||||||
|
* responsible for managing the correct sequence of object creation
|
||||||
|
*/
|
||||||
|
class Director {
|
||||||
|
public:
|
||||||
|
Director() : builder(nullptr) {}
|
||||||
|
|
||||||
|
~Director() {
|
||||||
|
if (builder) {
|
||||||
|
delete builder;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void set(Builder *b) {
|
||||||
|
if (builder) {
|
||||||
|
delete builder;
|
||||||
|
}
|
||||||
|
builder = b;
|
||||||
|
}
|
||||||
|
|
||||||
|
Product get() {
|
||||||
|
return builder->get();
|
||||||
|
}
|
||||||
|
|
||||||
|
void construct() {
|
||||||
|
builder->buildPartA();
|
||||||
|
builder->buildPartB();
|
||||||
|
builder->buildPartC();
|
||||||
|
// ...
|
||||||
|
}
|
||||||
|
// ...
|
||||||
|
|
||||||
|
private:
|
||||||
|
Builder *builder;
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
int main()
|
||||||
|
{
|
||||||
|
Director director;
|
||||||
|
director.set(new ConcreteBuilderX);
|
||||||
|
director.construct();
|
||||||
|
|
||||||
|
Product product1 = director.get();
|
||||||
|
std::cout << "1st product parts: " << product1.get() << std::endl;
|
||||||
|
|
||||||
|
director.set(new ConcreteBuilderY);
|
||||||
|
director.construct();
|
||||||
|
|
||||||
|
Product product2 = director.get();
|
||||||
|
std::cout << "2nd product parts: " << product2.get() << std::endl;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
12
builder/README.md
Normal file
12
builder/README.md
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
## Builder
|
||||||
|
|
||||||
|
Builder pattern has creational purpose and separates the construction of a complex object
|
||||||
|
from its representation so that the same construction process can create different
|
||||||
|
representations. It is object pattern, ie. relationships can be changed at run-time
|
||||||
|
and are more dynamic. Often is used for building composite structures but constructing
|
||||||
|
objects requires more domain knowledge of the client than using a Factory.
|
||||||
|
|
||||||
|
### When to use
|
||||||
|
|
||||||
|
* the algorithm for creating a object should be independent of the parts and how they're assembled
|
||||||
|
* the construction process must allow different representations for the object that's constructed
|
||||||
Reference in New Issue
Block a user