์ฑ… ๋ฆฌ๋ทฐ/๋ฆฌํŒฉํ† ๋ง(์ž๋ฐ”์Šคํฌ๋ฆฝํŠธํŒ)

[๋ฆฌํŒฉํ„ฐ๋ง 2ํŒ] 12.10์„ ์ฝ๊ธฐ์— ์•ž์„œ ์œ„์ž„์ด๋ž€?

Rainbow๐ŸŒˆCoder 2022. 11. 28. 11:50
728x90

์ฐธ๊ณ ๊ธ€ 

https://umbum.dev/822

 

Composition VS Extends : delegation, decorator, wrapper

Effective Java : ์•„์ดํ…œ 18. (๊ธฐ๋Šฅ ํ™•์žฅ์ด ํ•„์š”ํ•  ๋•Œ)์ƒ์†๋ณด๋‹ค๋Š” ์ปดํฌ์ง€์…˜์„ ์‚ฌ์šฉํ•˜๋ผ [Effective Java] 4์žฅ ํด๋ž˜์Šค์™€ ์ธํ„ฐํŽ˜์ด์Šค ์ƒ์†์ด๋ž€? extends๋ฅผ ๋งํ•จ. (implements๋Š” ์•„๋‹˜. ์ด๊ฑด ๊ตฌํ˜„.) ์ปดํฌ์ง€์…˜์ด๋ž€? Compo

umbum.dev

๋งํฌ ์š”์•ฝ:

์ƒ์†์€ extends

์ปดํฌ์ง€์…˜์€ ํ•„์š”ํ•œ ๊ฐ์ฒด๋ฅผ ๋‚ด๋ถ€ private ๋ณ€์ˆ˜๋กœ ๋‘๋Š” ๊ฒƒ(ํด๋ž˜์Šค๊ฐ€ ๋‹ค๋ฅธ ํด๋ž˜์Šค์˜ ๊ตฌ์„ฑ์š”์†Œ๋กœ ์“ฐ์ธ๋‹ค๋Š” ๋œป)

์ƒ์†๊ณผ ์ปดํฌ์ง€์…˜์˜ ์ฐจ์ด๋Š” ์˜ค๋ฒ„๋ผ์ด๋”ฉํ•˜๋А๋ƒ ๋ฉ”์„œ๋“œ๋ฅผ ํ˜ธ์ถœํ•ด์„œ delegateํ•˜๋А๋ƒ์˜ ์ฐจ์ด๋กœ ๋ณผ ์ˆ˜ ์žˆ๋‹ค.

 

0. ์ƒ์† vs ๋ธ๋ฆฌ๊ฒŒ์ดํŠธ ์œ„์ž„(์ปดํฌ์ง€์…˜)

์ƒ์†์€, ์ˆ˜์ง์ ์ธ ๊ด€๊ณ„๋ฅผ ์ด์šฉํ•ด์„œ ์ฝ”๋“œ๋ฅผ ํ™•์žฅํ•˜๊ณ  ์žฌ์‚ฌ์šฉํ•œ๋‹ค๋Š” ๊ฐœ๋…์ด๋‹ค. ๋‹จ ํ•˜๋‚˜์˜ ํด๋ž˜์Šค๋งŒ ์ƒ์†๋ฐ›์„ ์ˆ˜ ์žˆ๋‹ค(๋‹ค์ค‘์ƒ์† ๋ถˆ๊ฐ€), ์œ„์ž„์ด ๋” ์ ์ ˆํ•œ๋ฐ๋„ ์ƒ์†์„ ๋‚จ์šฉํ•˜๋ฉด ์ด์น˜์— ์•ˆ ๋งž๋Š” ์ผ€์ด์Šค๊ฐ€ ๋ฐœ์ƒํ•  ์ˆ˜ ์žˆ๊ณ , ์œ ์ง€๋ณด์ˆ˜๊ฐ€ ์–ด๋ ต๋‹ค

 

๊ทธ๋Ÿฌ๋ฏ€๋กœ, ์ƒ์†์€ ํ•ด์•ผํ• (ํ•„์š”ํ•œ) ๊ฒฝ์šฐ๋ฅผ ๋‚จ๊ฒจ๋‘๋Š” ๊ฒƒ์ด ์ข‹๋‹ค.

 

์œ„์ž„์€, ๋ ˆ๊ณ ๋ฅผ ์กฐ๋ฆฝํ•˜๋Š” ๊ฒƒ์ฒ˜๋Ÿผ ํ•„์š”ํ•œ ๋ถ€ํ’ˆ์„ ์™ธ๋ถ€๋กœ๋ถ€ํ„ฐ ์ฃผ์ž…๋ฐ›์•„์„œ ์ฃผ์ž…๋ฐ›์€ ๊ทธ ๋ถ€ํ’ˆ์„ ํ™œ์šฉํ•˜๋Š” ๊ฒƒ์ด๋ผ๊ณ  ์ƒ๊ฐํ•˜๋ฉด ํŽธํ•˜๋‹ค.

 

 

1. ํด๋ž˜์Šค ์ƒ์†์˜ ์œ ์šฉํ•จ

//์ƒ์†
class Printer {
	print() {
		console.log('๊ธฐ๋ณธ์ ์ธ ์ถœ๋ ฅ');
	}
}

class ColorPrinter extends Printer {
	print() {
		console.log('์ปฌ๋Ÿฌ ์ถœ๋ ฅ!');
	}
}

class BlackPrinter extends Printer {
	print() {
		console.log('ํ‘๋ฐฑ ์ถœ๋ ฅ!');
	}
}

const printers = [new Printer(), new ColorPrinter(), new BlackPrinter()];
printers.forEach((printer) => printer.print());

์ถœ๋ ฅ : ๋‹คํ˜•์„ฑ์˜ ๋Œ€ํ‘œ์ ์ธ ์˜ˆ์‹œ๋ผ๊ณ  ํ•  ์ˆ˜ ์žˆ๋‹ค.

 

2. ์ƒ์† ๋Œ€์‹  composition(์ปดํฌ์ง€์…˜:์œ„์ž„)์„ ์“ด๋‹ค๋ฉด

//์ปดํฌ์ง€์…˜(์œ„์ž„)
class Printer {
	#delegate; //๋ฌผ๋ก  ์ด๋Ÿฐ ์ด๋ฆ„์€ ๋‚˜์œ ๋ƒ„์ƒˆ๊ฐ€ ๋‚˜๋ฏ€๋กœ, printerHeader ๋“ฑ์œผ๋กœ ๋ฐ”๊พธ๋Š” ๊ฒƒ์ด ์ข‹๋‹ค.
	constructor(delegate) {
		this.#delegate = delegate;
	}
	print() {
    //์ „๋‹ฌ๋ฐ›์€ ๋ธ๋ฆฌ๊ฒŒ์ดํŠธ๊ฐ€ ์žˆ๋‹ค๋ฉด ์‹คํ–‰, ์—†๋‹ค๋ฉด ๊ธฐ๋ณธ ์ถœ๋ ฅ
		this.#delegate ? this.#delegate.print() : console.log('๊ธฐ๋ณธ์ ์ธ ์ถœ๋ ฅ');
	}
}

class ColorPrinterHeader {
	print() {
		console.log('์ปฌ๋Ÿฌ ์ถœ๋ ฅ!');
	}
}

class BlackPrinterHeader {
	print() {
		console.log('ํ‘๋ฐฑ ์ถœ๋ ฅ!');
	}
}


//ํ”„๋ฆฐํ„ฐ๋ฅผ ๋งŒ๋“ค ๋•Œ, ๋ธ๋ฆฌ๊ฒŒ์ดํŠธ ์šฉ ํด๋ž˜์Šค(์œ„์ž„ํ•  ์ˆ˜ ์žˆ๋Š” ์ธ์Šคํ„ด์Šค)๋ฅผ ์ฃผ์ž…ํ•ด์ค€๋‹ค
const printers = [new Printer(), new Printer(new ColorPrinterHeader()), new Printer(new BlackPrinterHeader())];
printers.forEach((printer) => printer.print());

์ถœ๋ ฅ : ์ด๋ ‡๋“ฏ ์œ„์ž„์„ ์ด์šฉํ•˜๋ฉด ๋™์ผํ•œ ํด๋ž˜์Šค๋ฅผ ์ด์šฉํ•ด์„œ ์ „๋‹ฌ๋œ ์œ„์ž„์ž์— ๋”ฐ๋ผ์„œ ๋‹ค๋ฅธ ํ–‰๋™์„ ํ•˜๋„๋ก ๋งŒ๋“ค ์ˆ˜ ์žˆ๋‹ค.

์ปดํฌ์ง€์…˜์„ ์ด์šฉํ•˜๋ฉด ๋ ˆ๊ณ ๋ฅผ ์กฐ๋ฆฝํ•˜๋“ฏ์ด ํ•„์š”ํ•œ ๊ธฐ๋Šฅ๋“ค์„ ํ•˜๋‚˜ํ•˜๋‚˜์”ฉ ๋ฌถ์–ด์„œ ์กฐ๋ฆฝํ•ด๋‚˜๊ฐˆ ์ˆ˜ ์žˆ๋‹ค.

 

 

3. ์ปดํฌ์ง€์…˜ ์ฝ”๋“œ๋ฅผ ts๋กœ ์ „ํ™˜ํ•œ๋‹ค๋ฉด

//์œ„์ž„์€ ์‚ฌ์‹ค ํƒ€์ž…์Šคํฌ๋ฆฝํŠธ๋กœ ํ•  ๋•Œ ๋”์šฑ ํ’๋ถ€ํ•œ ์‹œ๋„ˆ์ง€๋ฅผ ๊ฐ€์งˆ ์ˆ˜ ์žˆ๋‹ค.
class Printer {
	//์ด๋ ‡๋“ฏ ๋ธ๋ฆฌ๊ฒŒ์ดํŠธ ์ฃผ์ž…ํ•  ๋•Œ๋ถ€ํ„ฐ private ๋‹ฌ์•„์ฃผ๋ฉด #์„ ์‹น๋‹ค ์ œ๊ฑฐํ•ด๋„ ๋œ๋‹ค! ํ›จ์”ฌ ๊น”๋”ํ•ด์ง„๋‹ค.
	//PrinterHeader์€ ์˜ต์…”๋„๋กœ ์ง€์ •ํ•˜์—ฌ ์ปดํฌ์ง€์…˜(์œ„์ž„: ๋ธ๋ฆฌ๊ฒŒ์ดํŠธ)๊ฐ€ ์žˆ์–ด๋„ ์—†์–ด๋„ ๋จ์„ ๋ช…์‹œํ•ด์ค€๋‹ค.
	constructor(private delegate?: PrinterHeader) { //๋˜ํ•œ ์•„๋ฌด๊ฑฐ๋‚˜ ์ „๋‹ฌํ•  ์ˆ˜ ์—†๊ฒŒ ํƒ€์ž…์„ ์ง€์ •ํ•  ์ˆ˜ ์žˆ๋‹ค.
		this.delegate = delegate;
	}
	print() {
		this.delegate ? this.delegate.print() : console.log('๊ธฐ๋ณธ์ ์ธ ์ถœ๋ ฅ');
	}
}

//PrinterHeader๊ฐ€ ๋˜๋ ค๋ฉด ์ด ๊ทœ๊ฒฉ(interface: ์ธํ„ฐํŽ˜์ด์Šค)์„ ๋”ฐ๋ผ์•ผ ํ•ด! ๊ทœ์น™์„ ์ •ํ•ด์ค€๋‹ค.
//ํด๋ž˜์Šค๋Š” ๊ฐ์ฒด์˜ ์ฒญ์‚ฌ์ง„์ด๋ผ๋ฉด ์ธํ„ฐํŽ˜์ด์Šค๋Š” ํด๋ž˜์Šค์˜ ์ฒญ์‚ฌ์ง„์ด๋‹ค.
//ํด๋ž˜์Šค๊ฐ€ ํ•ด์•ผํ•˜๋Š” ํ–‰๋™์„ ๊ฒฐ์ • : ์ฆ‰, ํด๋ž˜์Šค๊ฐ€ ์–ด๋–ค ๋ฉ”์†Œ๋“œ๋ฅผ ๊ฐ€์งˆ์ง€ ๊ฒฐ์ •ํ•˜๋‹ค.
//ํด๋ž˜์Šค๋Š” ์ธํ„ฐํŽ˜์ด์Šค๋ฅผ implement(๊ตฌํ˜„)ํ•œ๋‹ค.
interface PrinterHeader {
	//ํ”„๋ฆฐํŠธ๋ผ๋Š” ํ•จ์ˆ˜๋ฅผ ๊ฐ€์ง€๊ณ  ์žˆ์–ด์•ผ ํ•˜๋ฉฐ, ์•„๋ฌด ๊ฒƒ๋„ ๋ฐ˜ํ™˜ํ•˜์ง€ ์•Š์•„์•ผ ํ•œ๋‹ค.
	print(): void;
}


class ColorPrinterHeader implements PrinterHeader {
	print() {
		console.log('์ปฌ๋Ÿฌ ์ถœ๋ ฅ!');
	}
}

class BlackPrinterHeader implements PrinterHeader {
	print() {
		console.log('ํ‘๋ฐฑ ์ถœ๋ ฅ!');
	}
}

const printers = [new Printer(), new Printer(new ColorPrinterHeader()), new Printer(new BlackPrinterHeader())];
printers.forEach((printer) => printer.print());

๊ทธ๋Ÿฐ๋ฐ Printer์˜ delegate๋ฅผ null์ด ๋  ์ˆ˜ ์—†๊ฒŒ private๋กœ ๋‘๊ณ  ์•„๋ž˜์™€ ๊ฐ™์ด ๋ฐ”๊ฟ€ ์ˆ˜๋„ ์žˆ๋‹ค.

 

class Printer {
	private delegate: PrinterHeader;
	constructor(delegate?: PrinterHeader) {
		this.delegate = delegate ? delegate : new DefaultPrinterHeader();
	}
	print() {
		this.delegate.print();
	}
}

interface PrinterHeader {
	print(): void;
}

class DefaultPrinterHeader implements PrinterHeader {
	print() {
		console.log('๊ธฐ๋ณธ ์ถœ๋ ฅ!');
	}
}


class ColorPrinterHeader implements PrinterHeader {
	print() {
		console.log('์ปฌ๋Ÿฌ ์ถœ๋ ฅ!');
	}
}

class BlackPrinterHeader implements PrinterHeader {
	print() {
		console.log('ํ‘๋ฐฑ ์ถœ๋ ฅ!');
	}
}

const printers = [new Printer(), new Printer(new ColorPrinterHeader()), new Printer(new BlackPrinterHeader())];
printers.forEach((printer) => printer.print());

 

728x90