likes
comments
collection
share

十大经典设计模式及其Go语言实现1. 单例模式(Singleton Pattern) 特点:确保一个类只有一个实例,并

作者站长头像
站长
· 阅读数 21

1. 单例模式(Singleton Pattern)

  • 特点:确保一个类只有一个实例,并提供一个全局访问点。
  • 优点:保证资源或状态的全局唯一性,减少资源消耗。
  • 缺点:反模块化,测试困难。
  • 应用场景:配置管理器、连接池、日志记录器等。
  • 代码示例
type singleton struct {
    value string
}
var instance *singleton

// 定义一个结构体Singleton,用于存储单例的实例数据
type singleton struct {
    value string // 这里可以存储单例对象的任何数据
}

// 定义一个Once对象,用于确保初始化操作只执行一次
var once sync.Once

// 定义一个全局变量instance,用于存储单例的实例
var instance *singleton

// 初始化函数,由Once.Do调用
func initSingleton() {
    instance = &singleton{value: "unique instance"} // 这里初始化singleton实例
}

// getInstance函数用于获取单例的实例
func getInstance() *singleton {
    // 执行initSingleton,确保instance只被初始化一次
    once.Do(initSingleton)
    return instance // 返回单例的实例
}

func main() {
    // 获取单例实例
    singletonInstance := getInstance()
    fmt.Println(singletonInstance.value)
    // 再次获取单例实例,将返回相同的实例
    anotherInstance := getInstance()
    if singletonInstance == anotherInstance {
        fmt.Println("Both instances are the same")
    }
}

2. 工厂模式(Factory Pattern)

  • 特点:封装对象创建过程,由子类决定实例化哪个类。
  • 优点:提高模块独立性,易于扩展。
  • 缺点:类数量可能急剧增加,工厂类可能过于庞大。
  • 应用场景:数据库连接、GUI组件、支付网关等。
  • 代码示例
// 定义产品接口,声明了所有具体产品对象必须实现的操作
type Product interface {
    operation()
}

// 定义具体产品实现接口
type ConcreteProductA struct{}
func (p *ConcreteProductA) operation() {
    fmt.Println("Operation of ConcreteProductA")
}

// 定义抽象工厂接口,声明了工厂方法
type Creator interface {
    factoryMethod() Product
}

// 定义具体工厂实现接口
type CreatorA struct{}
func (c *CreatorA) factoryMethod() Product {
    return &ConcreteProductA{} // 返回具体产品的实例
}

func main() {
    // 使用工厂创建产品
    creatorA := &CreatorA{}
    productA := creatorA.factoryMethod()
    productA.operation()
}

3. 观察者模式(Observer Pattern)

  • 特点:定义对象间一对多的依赖关系,自动更新依赖对象。
  • 优点:降低耦合度,扩展性好。
  • 缺点:可能造成性能问题,系统难以维护。
  • 应用场景:事件监听系统、UI更新、消息系统等。
  • 代码示例
// 定义观察者接口,声明了更新方法
type Observer interface {
    Update(string)
}

// 定义主题,包含观察者列表和注册、通知观察者的方法
type Subject struct {
    observers []Observer
}

func (s *Subject) Attach(observer Observer) {
    s.observers = append(s.observers, observer)
}

func (s *Subject) Notify(message string) {
    for _, observer := range s.observers {
        observer.Update(message) // 通知所有观察者
    }
}

// 定义具体观察者实现接口
type ConcreteObserverA struct {
    name string
}

func (c *ConcreteObserverA) Update(message string) {
    fmt.Printf("%s received message: %s\n", c.name, message)
}

func main() {
    subject := &Subject{}
    observerA := &ConcreteObserverA{name: "Observer A"}
    subject.Attach(observerA) // 注册观察者
    subject.Notify("State changed to State 1") // 通知观察者
}

4. 装饰者模式(Decorator Pattern)

  • 特点:动态地给对象添加额外职责或功能。
  • 优点:增加职责动态且可撤销,多个装饰者可组合使用。
  • 缺点:系统复杂,可能影响性能。
  • 应用场景:日志记录、缓存、安全控制等。
  • 代码示例
// 定义组件接口,声明了组件的操作
type Component interface {
    operation()
}

// 定义具体组件实现接口
type ConcreteComponent struct{}
func (c *ConcreteComponent) operation() {
    fmt.Println("ConcreteComponent: performing basic operation")
}

// 定义装饰者抽象类,包含组件接口类型的字段
type Decorator struct {
    component Component
}

// 装饰者实现组件的操作,委托给组件
func (d *Decorator) operation() {
    if d.component != nil {
        d.component.operation()
    }
}

// 定义具体装饰者实现装饰者抽象类
type ConcreteDecoratorA struct {
    Decorator
}
func (cda *ConcreteDecoratorA) operation() {
    cda.Decorator.operation() // 调用装饰的组件操作
    fmt.Println("ConcreteDecoratorA: added additional responsibilities")
}

func main() {
    component := &ConcreteComponent{}
    decoratorA := &ConcreteDecoratorA{Decorator{component}}
    decoratorA.operation() // 装饰后的操作
}

5. 策略模式(Strategy Pattern)

  • 特点:定义算法家族,封装起来,互相替换。
  • 优点:算法变化独立于客户端,易于添加新算法。
  • 缺点:客户端需要了解策略类差异。
  • 应用场景:算法选择、支付方式、排序算法等。
  • 代码示例
// 定义策略接口,声明了策略的算法方法
type Strategy interface {
    algorithm()
}

// 定义具体策略实现接口
type ConcreteStrategyA struct{}
func (c *ConcreteStrategyA) algorithm() {
    fmt.Println("Executing Algorithm A")
}

// 定义上下文环境,包含策略接口类型的字段
type Context struct {
    strategy Strategy
}

// 上下文环境执行策略的方法
func (c *Context) executeStrategy() {
    c.strategy.algorithm()
}

func main() {
    context := &Context{}
    strategyA := &ConcreteStrategyA{}
    context.strategy = strategyA // 设置策略
    context.executeStrategy() // 执行策略
}

6. 适配器模式(Adapter Pattern)

  • 特点:使不兼容接口能一起工作。
  • 优点:增加兼容性,客户端代码无需修改。
  • 缺点:系统复杂,可能引入性能开销。
  • 应用场景:系统集成、第三方库集成等。
  • 代码示例
// 定义客户端期望的接口
type Target interface {
    request()
}

// 定义一个已经存在的类,有自己的接口
type Adaptee struct{}
func (a *Adaptee) specificRequest() {
    fmt.Println("Adaptee performs a specific request")
}

// 定义适配器,作为Target接口和Adaptee类之间的桥梁
type Adapter struct {
    adaptee *Adaptee
}
func (a *Adapter) request() {
    if a.adaptee != nil {
        a.adaptee.specificRequest() // 委托调用Adaptee的方法
    }
}

func main() {
    adaptee := &Adaptee{}
    adapter := &Adapter{adaptee: adaptee}
    var target Target = adapter // 客户端通过适配器使用Adaptee
    target.request()
}

7. 代理模式(Proxy Pattern)

  • 特点:为另一个对象提供代替或占位符,控制访问。
  • 优点:降低耦合度,增加可控性,代码可扩展。
  • 缺点:增加系统复杂性,可能引入性能开销。
  • 应用场景:访问控制、延迟初始化、远程代理等。
  • 代码示例
// 定义主题接口,声明了请求方法
type Subject interface {
    request()
}

// 定义真实主题,实现了主题接口
type RealSubject struct{}
func (r *RealSubject) request() {
    fmt.Println("Real Subject")
}

// 定义代理主题,包含对真实主题的引用
type Proxy struct {
    realSubject *RealSubject
}
func (p *Proxy) request() {
    if p.realSubject == nil {
        p.realSubject = &RealSubject{} // 如果没有真实主题,则创建一个
    }
    p.realSubject.request() // 调用真实主题的请求方法
}

8. 命令模式(Command Pattern)

  • 特点:将请求或操作封装为对象,解耦发送者和接收者。
  • 优点:降低耦合度,增加操作灵活性,易于扩展。
  • 缺点:可能产生大量命令类。
  • 应用场景:事务处理、撤销操作、日志请求等。
  • 代码示例
// 定义命令接口,声明了执行方法
type Command interface {
    Execute()
}

// 定义接收者,将执行命令的实际请求
type Receiver struct{}
func (r *Receiver) Action() {
    fmt.Println("Receiver: Action")
}

// 定义具体命令,实现命令接口,包含接收者的引用
type ConcreteCommand struct {
    receiver *Receiver
}
func (c *ConcreteCommand) Execute() {
    c.receiver.Action() // 执行接收者的操作
}

// 定义调用者,负责调用命令对象的执行方法
type Invoker struct {
    command Command
}
func (i *Invoker) Invoke() {
    i.command.Execute() // 执行命令
}

func main() {
    receiver := &Receiver{}
    command := &ConcreteCommand{receiver: receiver} // 创建具体命令对象,并注入接收者
    invoker := &Invoker{command: command} // 创建调用者对象,并注入具体命令对象
    invoker.Invoke() // 调用者执行命令
}

9. 组合模式(Composite Pattern)

  • 特点:将对象组合成树状结构,一致对待单个对象和组合。
  • 优点:简化客户端代码,更好的层次结构表示。
  • 缺点:设计复杂,需要合理设计接口和类。
  • 应用场景:文件系统、组织结构、GUI组件等。
  • 代码示例
// 定义组件接口,作为组合中对象的一致性协议
type Component interface {
    Operation()
    Add(Component)
    Remove(Component)
    GetChild(int) Component
}

// 定义叶节点,实现组件接口
type Leaf struct {
    name string
}
func (l *Leaf) Operation() {
    fmt.Println("Leaf:", l.name)
}
func (l *Leaf) Add(c Component) {
    fmt.Println("Cannot add to a leaf")
}
func (l *Leaf) Remove(c Component) {
    fmt.Println("Cannot remove from a leaf")
}
func (l *Leaf) GetChild(i int) Component {
    return nil
}

// 定义组合节点,实现组件接口
type Composite struct {
    name string
    Children []Component
}
func (c *Composite) Operation() {
    fmt.Println("Composite:", c.name)
    for _, child := range c.Children {
        child.Operation()
    }
}
func (c *Composite) Add(component Component) {
    c.Children = append(c.Children, component)
}
func (c *Composite) Remove(component Component) {
    for i, child := range c.Children {
        if child == component {
            c.Children = append(c.Children[:i], c.Children[i+1:]...)
            break
        }
    }
}
func (c *Composite) GetChild(i int) Component {
    if i < 0 || i >= len(c.Children) {
        return nil
    }
    return c.Children[i]
}

func main() {
    leafA := &Leaf{name: "Leaf A"}
    leafB := &Leaf{name: "Leaf B"}
    composite := &Composite{name: "Composite Root"}
    composite.Add(leafA) // 向组合中添加叶节点A
    composite.Add(leafB) // 向组合中添加叶节点B
    composite.Operation() //

10. 迭代器模式(Iterator Pattern)

  • 特点:顺序访问聚合对象元素,不暴露内部表示。
  • 优点:抽象化集合访问,支持多种遍历方式,增加灵活性。
  • 缺点:增加系统复杂性,需要额外代码实现迭代器。
  • 应用场景:遍历集合、数据结构、数据库查询等。
  • 代码示例
// 定义迭代器接口,声明了遍历集合的方法
type Iterator interface {
    Next() bool   // 移动到下一个元素
    Current() interface{} // 返回当前元素
}

// 定义具体迭代器实现接口
type ConcreteIterator struct {
    items []string   // 存储聚合对象的元素列表
    index int        // 当前迭代到的元素索引
}
func (c *ConcreteIterator) Next() bool {
    if c.index < len(c.items) {
        c.index++
        return true
    }
    return false
}
func (c *ConcreteIterator) Current() interface{} {
    if c.index > 0 && c.index <= len(c.items) {
        return c.items[c.index-1]
    }
    return nil
}

// 定义聚合对象接口,声明了创建迭代器的方法
type Aggregate interface {
    CreateIterator() Iterator // 创建并返回迭代器
}

// 定义具体聚合对象实现接口
type ConcreteAggregate struct {
    items []string // 聚合对象存储的元素列表
}
func (a *ConcreteAggregate) CreateIterator() Iterator {
    return &ConcreteIterator{items: a.items, index: 0} // 返回一个新的迭代器实例
}

func main() {
    aggregate := &ConcreteAggregate{items: []string{"Item1", "Item2", "Item3"}}
    iterator := aggregate.CreateIterator() // 使用聚合对象创建迭代器
    // 使用迭代器遍历聚合对象中的所有元素
    for iterator.Next() {
        fmt.Println(iterator.Current())
    }
}

这些设计模式是软件工程中常用的解决方案,可以帮助开发者提高代码的可重用性、可读性和可靠性。

转载自:https://juejin.cn/post/7410672790020784164
评论
请登录