
Мост на Go
Мост — это структурный паттерн, который разделяет бизнес-логику или большой класс на несколько отдельных иерархий, которые потом можно развивать отдельно друг от друга.
Одна из этих иерархий (абстракция) получит ссылку на объекты другой иерархии (реализация) и будет делегировать им основную работу. Благодаря тому, что все реализации будут следовать общему интерфейсу, их можно будет взаимозаменять внутри абстракции.
Концептуальный пример
Представим, что у вас есть два типа компьютеров: Mac и Windows, а также два типа принтеров: Epson и HP. Компьютеры и принтеры должны работать между собой в любых комбинациях. Клиент не хочет думать об особенностях подключения принтеров к компьютерам.
Мы не хотим, чтобы при введении в эту систему новых принтеров количество кода увеличивалось по экспоненте. Вместо создания четырех структур для 2*2 комбинаций, мы создадим две иерархии:
- Иерархия абстракции: сюда будут входить наши компьютеры
- Иерархия реализации: сюда будут входить наши принтеры
Эти две иерархии общаются между собой посредством Моста, в котором Абстракция (компьютер) содержит ссылку на Реализацию (принтер). И абстракцию, и реализацию можно разрабатывать отдельно, не влияя друг на друга.
computer.go: Абстракция
package main
type computer interface {
print()
setPrinter(printer)
}
mac.go: Расширенная абстракция
package main
import "fmt"
type mac struct {
printer printer
}
func (m *mac) print() {
fmt.Println("Print request for mac")
m.printer.printFile()
}
func (m *mac) setPrinter(p printer) {
m.printer = p
}
windows.go: Расширенная абстракция
package main
import "fmt"
type windows struct {
printer printer
}
func (w *windows) print() {
fmt.Println("Print request for windows")
w.printer.printFile()
}
func (w *windows) setPrinter(p printer) {
w.printer = p
}
printer.go: Реализация
package main
type printer interface {
printFile()
}
epson.go: Конкретная реализация
package main
import "fmt"
type epson struct {
}
func (p *epson) printFile() {
fmt.Println("Printing by a EPSON Printer")
}
hp.go: Конкретная реализация
package main
import "fmt"
type hp struct {
}
func (p *hp) printFile() {
fmt.Println("Printing by a HP Printer")
}
main.go: Клиентский код
package main
import "fmt"
func main() {
hpPrinter := &hp{}
epsonPrinter := &epson{}
macComputer := &mac{}
macComputer.setPrinter(hpPrinter)
macComputer.print()
fmt.Println()
macComputer.setPrinter(epsonPrinter)
macComputer.print()
fmt.Println()
winComputer := &windows{}
winComputer.setPrinter(hpPrinter)
winComputer.print()
fmt.Println()
winComputer.setPrinter(epsonPrinter)
winComputer.print()
fmt.Println()
}
output.txt: Результат выполнения
Print request for mac
Printing by a HP Printer
Print request for mac
Printing by a EPSON Printer
Print request for windows
Printing by a HP Printer
Print request for windows