Нужна клёвая книга о паттернах на русском? Вот она »
Прототип

Прототип на Go

Прототип — это порождающий паттерн, который позволяет копировать объекты любой сложности без привязки к их конкретным классам.

Все классы—Прототипы имеют общий интерфейс. Поэтому вы можете копировать объекты, не обращая внимания на их конкретные типы и всегда быть уверены, что получите точную копию. Клонирование совершается самим объектом-прототипом, что позволяет ему скопировать значения всех полей, даже приватных.

Концептуальный пример

Давайте попробуем разобрать паттерн Прототип, используя для примера файловую систему ОС. Файловая система является рекурсивной – папки содержат файлы и другие папки, которые, в свою очередь, могут содержать файлы и папки, и так далее.

Каждый файл и папка могут быть представлены интерфейсом inode. Он имеет функцию clone.

Обе структуры файла и папки — file и folder — реализуют функции print и clone, поскольку они имеют тип inode. Также, обратите внимание на функцию clone в file и folder. Функция clone в обеих случаях возвращает копию соответствующего файла или папки. Во время клонирования мы добавляем ключевое слово «_clone» в поле имени.

inode.go: Интерфейс прототипа

package main

type inode interface {
	print(string)
	clone() inode
}

file.go: Конкретный прототип

package main

import "fmt"

type file struct {
	name string
}

func (f *file) print(indentation string) {
	fmt.Println(indentation + f.name + "_clone")
}

func (f *file) clone() inode {
	return &file{name: f.name}
}

folder.go: Конкретный прототип

package main

import "fmt"

type folder struct {
	childrens []inode
	name      string
}

func (f *folder) print(indentation string) {
	fmt.Println(indentation + f.name)
	for _, i := range f.childrens {
		i.print(indentation + indentation)
	}
}

func (f *folder) clone() inode {
	cloneFolder := &folder{name: f.name + "_clone"}
	var tempChildrens []inode
	for _, i := range f.childrens {
		copy := i.clone()
		tempChildrens = append(tempChildrens, copy)
	}
	cloneFolder.childrens = tempChildrens
	return cloneFolder
}

main.go: Клиентский код

package main

import "fmt"

func main() {
	file1 := &file{name: "File1"}
	file2 := &file{name: "File2"}
	file3 := &file{name: "File3"}

	folder1 := &folder{
		childrens: []inode{file1},
		name:      "Folder1",
	}

	folder2 := &folder{
		childrens: []inode{folder1, file2, file3},
		name:      "Folder2",
	}
	fmt.Println("\nPrinting hierarchy for Folder2")
	folder2.print("  ")

	cloneFolder := folder2.clone()
	fmt.Println("\nPrinting hierarchy for clone Folder")
	cloneFolder.print("  ")
}

output.txt: Результат выполнения

Printing hierarchy for Folder2
  Folder2
    Folder1
        File1
    File2
    File3

Printing hierarchy for clone Folder
  Folder2_clone
    Folder1_clone
        File1_clone
    File2_clone
    File3_clone
По материалам: Golang By Example

Прототип на других языках программирования

Прототип на Java Прототип на C# Прототип на C++ Прототип на PHP Прототип на Python Прототип на Ruby Прототип на Swift Прототип на TypeScript