Personal-Study/Design Patterns
[TS Design Patterns] 생성 패턴 - 빌더
Aaron-Kim
2023. 10. 21. 21:03
Builder
- Lets you construct complex objects step by step
- Allows us to produce different types and representations of an object
using the same construction code
- Suggests that us extract the object construction code out of its own class
and move it to separate objects called builders
- 예시
/**
* The Builder interface specifies methods for creating the different parts of
* the Product objects.
*/
interface Builder {
producePartA(): void;
producePartB(): void;
producePartC(): void;
}
/**
* The Concrete Builder classes follow the Builder interface and provide
* specific implementations of the building steps. Your program may have several
* variations of Builders, implemented differently.
*/
class ConcreteBuilder1 implements Builder {
// @ts-ignore
private product: Product1;
/**
* A fresh builder instance should contain a blank product object, which is
* used in further assembly.
*/
constructor() {
this.reset();
}
public reset(): void {
this.product = new Product1();
}
/**
* All production steps work with the same product instance.
*/
public producePartA(): void {
this.product.parts.push('PartA1');
}
public producePartB(): void {
this.product.parts.push('PartB1');
}
public producePartC(): void {
this.product.parts.push('PartC1');
}
/**
* Concrete Builders are supposed to provide their own methods for
* retrieving results. That's because various types of builders may create
* entirely different products that don't follow the same interface.
* Therefore, such methods cannot be declared in the base Builder interface
* (at least in a statically typed programming language).
*
* Usually, after returning the end result to the client, a builder instance
* is expected to be ready to start producing another product. That's why
* it's a usual practice to call the reset method at the end of the
* `getProduct` method body. However, this behavior is not mandatory, and
* you can make your builders wait for an explicit reset call from the
* client code before disposing of the previous result.
*/
public getProduct(): Product1 {
const result = this.product;
this.reset();
return result;
}
}
/**
* It makes sense to use the Builder pattern only when your products are quite
* complex and require extensive configuration.
*
* Unlike in other creational patterns, different concrete builders can produce
* unrelated products. In other words, results of various builders may not
* always follow the same interface.
*/
class Product1 {
public parts: string[] = [];
public listParts(): void {
console.log(`Product parts: ${this.parts.join(', ')}\n`);
}
}
/**
* The Director is only responsible for executing the building steps in a
* particular sequence. It is helpful when producing products according to a
* specific order or configuration. Strictly speaking, the Director class is
* optional, since the client can control builders directly.
*/
class Director {
// @ts-ignore
private builder: Builder;
/**
* The Director works with any builder instance that the client code passes
* to it. This way, the client code may alter the final type of the newly
* assembled product.
*/
public setBuilder(builder: Builder): void {
this.builder = builder;
}
/**
* The Director can construct several product variations using the same
* building steps.
*/
public buildMinimalViableProduct(): void {
this.builder.producePartA();
}
public buildFullFeaturedProduct(): void {
this.builder.producePartA();
this.builder.producePartB();
this.builder.producePartC();
}
}
/**
* The client code creates a builder object, passes it to the director and then
* initiates the construction process. The end result is retrieved from the
* builder object.
*/
function clientCode(director: Director) {
const builder = new ConcreteBuilder1();
director.setBuilder(builder);
console.log('Standard basic product:');
director.buildMinimalViableProduct();
builder.getProduct().listParts();
console.log('Standard full featured product:');
director.buildFullFeaturedProduct();
builder.getProduct().listParts();
// Remember, the Builder pattern can be used without a Director class.
console.log('Custom product:');
builder.producePartA();
builder.producePartC();
builder.getProduct().listParts();
}
const director = new Director();
clientCode(director);
// pagres-org/react-design-pattern
// users.ts
class User {
name: string;
age?: number;
phone?: number | string;
address?: string;
constructor(name: string) {
this.name = name;
}
print() {
return `[USER]: name=${this.name}, age=${this.age}, phone=${this.phone}, address=${this.address}`;
}
}
export class UserBuilder {
user: User;
constructor(name: string) {
this.user = new User(name);
}
setAge(age: number) {
this.user.age = age;
return this;
}
setPhone(phone: number | string) {
this.user.phone = phone;
return this;
}
setAddress(address: string) {
this.user.address = address;
return this;
}
build() {
return this.user;
}
}
const user = new UserBuilder('ding-co')
.setAddress('Seoul')
.setPhone('010-0000-0000')
.setAge(29)
.build();
console.log(user);
- 활용
- Nest.js 쿼리 빌더
async getPosts() {
const postsDAO = await this.postRepository
.createQueryBuilder('post')
.loadRelationCountAndMap('post.commentCount', 'post.comments')
.orderBy('post.created_at', 'DESC')
.getMany();
return postsDAO;
}
반응형