上一篇桥接设计模式:https://blog.csdn.net/weixin_40165163/article/details/90731470
github:https://github.com/zhumengyifang/GolangDesignPatterns
wiki百科: 组合模式,又叫部分整体模式,是用于把一组相似的对象当作一个单一的对象。组合模式依据树形结构来组合对象,用来表示部分以及整体层次。这种类型的设计模式属于结构模式,它创建了对象组的树形结构。
这种模式创建了一个包含自己对象组的类。该类提供了修改相同对象组的方式。
意图:将对象组合成树形结构以表示“部分-整体”的层次结构。组合模式使得用户对单个对象和组合对象的使用具有一致性。
主要解决:它在我们树形结构的问题中,模糊了简单元素和复杂元素的概念,客户程序可以向处理简单元素一样来处理复杂元素,从而使得客户程序与复杂元素的内部结构解偶。
何时使用:
如何解决:树枝和叶子实现同一接口,树枝内部组合该接口。
关键代码:树枝内部组合该接口,并且含有内部属性slice,里面放置元素
优点:
缺点:在使用组合模式时,其叶子和树枝的声明都是实现类,而不是接口,违反了依赖倒置原则。
使用场景:部分、整体场景。
一个员工(Employee)Struct,该类被当作组合模型类。CompositePatternDemo,我们的演示类使用员工Struce来添加部门层次结构,并打印所有的员工
代码:
package CompositePattern
import "fmt"
type Employee struct {
name string
dept string
salary int
subordinates []*Employee
}
func GetEmployee(name string, dept string, sal int) *Employee {
employee := new(Employee)
employee.name = name
employee.dept = dept
employee.salary = sal
return employee
}
func (e *Employee) Add(ee *Employee) {
e.subordinates = append(e.subordinates, ee)
}
func (e *Employee) Remove(ee *Employee) {
target := e.subordinates[:0]
for _, item := range e.subordinates {
if item == ee {
target = append(target, item)
}
}
e.subordinates = target
}
func (e *Employee) GetSubordinates() []*Employee {
return e.subordinates
}
func (e *Employee) ToString() {
fmt.Println("name:", e.name, "dept:", e.dept, "salary:", e.salary)
}
func (e *Employee) PrintSubordinates() {
fmt.Println("=============")
e.ToString()
for _, headEmployee := range e.GetSubordinates() {
headEmployee.ToString()
for _, employee := range headEmployee.GetSubordinates() {
employee.ToString()
}
}
fmt.Println("=============")
}
测试:
func testCompositePattern() {
ceo := CompositePattern.GetEmployee("John", "CEO", 30000)
headSales := CompositePattern.GetEmployee("Robert", "Head Sales", 20000)
headMarketing := CompositePattern.GetEmployee("Michel", "Head Marketing", 20000)
ceo.Add(headSales)
ceo.Add(headMarketing)
salesExecutive1 := CompositePattern.GetEmployee("Richard", "Sales", 10000)
salesExecutive2 := CompositePattern.GetEmployee("Rob", "Sales", 10000)
headSales.Add(salesExecutive1)
headSales.Add(salesExecutive2)
clerk1 := CompositePattern.GetEmployee("Laura", "Marketing", 10000)
clerk2 := CompositePattern.GetEmployee("Bob", "Marketing", 10000)
headMarketing.Add(clerk1)
headMarketing.Add(clerk2)
ceo.PrintSubordinates()
ceo.Remove(headSales)
ceo.Remove(headMarketing)
ceo.PrintSubordinates()
}
输出:
=============
name: John dept: CEO salary: 30000
name: Robert dept: Head Sales salary: 20000
name: Richard dept: Sales salary: 10000
name: Rob dept: Sales salary: 10000
name: Michel dept: Head Marketing salary: 20000
name: Laura dept: Marketing salary: 10000
name: Bob dept: Marketing salary: 10000
=============
=============
name: John dept: CEO salary: 30000
=============
结论:跟树类似,只不过可有无限多个叶子节点,这取决于实现的业务场景。
下一篇装饰器设计模式:https://blog.csdn.net/weixin_40165163/article/details/90740155