0

0

如何在Golang中实现访问者模式操作复合对象

P粉602998670

P粉602998670

发布时间:2025-10-19 20:09:01

|

555人浏览过

|

来源于php中文网

原创

答案:Go语言通过接口实现访问者模式,分离数据结构与操作。定义Element和Visitor接口,元素实现Accept方法接收访问者,访问者为每种元素提供Visit方法。具体元素如File和Folder包含Accept逻辑,Folder递归调用子元素。具体访问者如SizeVisitor统计文件大小,PrintVisitor打印层级结构。使用时创建访问者实例并调用根元素的Accept方法触发遍历。新增行为只需扩展访问者,无需修改元素类,符合开闭原则。该模式适用于树形结构,利用接口和组合弥补无继承缺陷,实现双重分派,解耦结构与算法。

如何在golang中实现访问者模式操作复合对象

在Go语言中实现访问者模式操作复合对象,关键是分离数据结构与作用于其上的操作。这种设计让你可以在不修改原有对象结构的前提下,为它们添加新的行为。下面说明如何用Go实现这一模式。

定义元素接口和访问者接口

复合对象通常由多个不同类型的数据节点组成。先定义一个通用的元素接口,让所有具体元素实现它。

Element接口声明了Accept方法,用于接收访问者:

type Element interface {
    Accept(visitor Visitor)
}

Visitor接口则针对每种具体元素类型定义Visit方法:

立即学习go语言免费学习笔记(深入)”;

type Visitor interface {
    VisitFile(f *File)
    VisitFolder(f *Folder)
}

这样,当一个元素调用Accept时,会反向调用访问者的对应Visit方法,实现“双重分派”。

实现具体元素类型

假设我们要处理文件系统中的文件和文件夹,可以定义两个结构体:

网易外贸通
网易外贸通

网易旗下专为外贸企业打造的一站式海外营销管理平台

下载
type File struct {
    Name string
    Size int
}

func (f *File) Accept(visitor Visitor) { visitor.VisitFile(f) }

type Folder struct { Name string Children []Element }

func (f *Folder) Accept(visitor Visitor) { visitor.VisitFolder(f) for _, child := range f.Children { child.Accept(visitor) } }

注意Folder在被访问后,还会递归地让子元素接受访问,从而实现对整个树形结构的操作。

实现具体访问者

现在可以定义不同的访问者来执行特定任务。比如统计总大小:

type SizeVisitor struct {
    TotalSize int
}

func (v SizeVisitor) VisitFile(f File) { v.TotalSize += f.Size }

func (v SizeVisitor) VisitFolder(f Folder) { // 文件夹本身不占空间,也可根据需要计入元数据开销 }

或者打印结构树:

type PrintVisitor struct {
    Level int
}

func (v PrintVisitor) VisitFile(f File) { indent := strings.Repeat(" ", v.Level) fmt.Printf("%s- File: %s (%d bytes)\n", indent, f.Name, f.Size) }

func (v PrintVisitor) VisitFolder(f Folder) { indent := strings.Repeat(" ", v.Level) fmt.Printf("%s+ Folder: %s\n", indent, f.Name) v.Level++ }

使用时只需创建访问者实例并启动遍历:

root := &Folder{
    Name: "root",
    Children: []Element{
        &File{Name: "a.txt", Size: 100},
        &Folder{
            Name: "sub",
            Children: []Element{
                &File{Name: "b.txt", Size: 200},
            },
        },
    },
}

sizeVisitor := &SizeVisitor{} root.Accept(sizeVisitor) fmt.Printf("Total size: %d\n", sizeVisitor.TotalSize)

printVisitor := &PrintVisitor{} root.Accept(printVisitor)

这种方式让新增操作变得非常灵活。比如以后要加权限检查、备份操作或序列化功能,只需实现新的Visitor,无需改动File或Folder代码。

基本上就这些。Go没有继承,但通过接口和组合能很好地支持访问者模式。只要把握“元素接受访问者,访问者处理元素”的交互逻辑,就能清晰地解耦数据结构与行为。对于树形或图形结构的复合对象特别实用。

相关专题

更多
golang如何定义变量
golang如何定义变量

golang定义变量的方法:1、声明变量并赋予初始值“var age int =值”;2、声明变量但不赋初始值“var age int”;3、使用短变量声明“age :=值”等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

180

2024.02.23

golang有哪些数据转换方法
golang有哪些数据转换方法

golang数据转换方法:1、类型转换操作符;2、类型断言;3、字符串和数字之间的转换;4、JSON序列化和反序列化;5、使用标准库进行数据转换;6、使用第三方库进行数据转换;7、自定义数据转换函数。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

228

2024.02.23

golang常用库有哪些
golang常用库有哪些

golang常用库有:1、标准库;2、字符串处理库;3、网络库;4、加密库;5、压缩库;6、xml和json解析库;7、日期和时间库;8、数据库操作库;9、文件操作库;10、图像处理库。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

341

2024.02.23

golang和python的区别是什么
golang和python的区别是什么

golang和python的区别是:1、golang是一种编译型语言,而python是一种解释型语言;2、golang天生支持并发编程,而python对并发与并行的支持相对较弱等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

209

2024.03.05

golang是免费的吗
golang是免费的吗

golang是免费的。golang是google开发的一种静态强类型、编译型、并发型,并具有垃圾回收功能的开源编程语言,采用bsd开源协议。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

393

2024.05.21

golang结构体相关大全
golang结构体相关大全

本专题整合了golang结构体相关大全,想了解更多内容,请阅读专题下面的文章。

200

2025.06.09

golang相关判断方法
golang相关判断方法

本专题整合了golang相关判断方法,想了解更详细的相关内容,请阅读下面的文章。

191

2025.06.10

golang数组使用方法
golang数组使用方法

本专题整合了golang数组用法,想了解更多的相关内容,请阅读专题下面的文章。

293

2025.06.17

c++ 根号
c++ 根号

本专题整合了c++根号相关教程,阅读专题下面的文章了解更多详细内容。

25

2026.01.23

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Go语言实战之 GraphQL
Go语言实战之 GraphQL

共10课时 | 0.8万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号