
本教程详细阐述了如何利用go语言,将具有父子关系的扁平化表格数据高效地转换为内存中的树形结构。通过定义简洁的节点类型、运用哈希映射实现父节点的快速查找与关联,以及设计递归函数进行树的构建与可视化展示,我们提供了一套完整的解决方案,旨在帮助开发者理解并实现此类数据转换,并附带了可运行的代码示例。
在许多应用场景中,数据天然具有层级关系,例如组织架构、文件系统或评论回复链。然而,这些层级数据在数据库或CSV文件中常常以扁平化的表格形式存储,每行记录包含一个唯一的ID、名称以及一个指向其父节点的ID。我们的目标就是将这种扁平化的父子关系数据转换为内存中的树形结构,以便于进行层级遍历、查找和展示。
例如,以下表格展示了一个简化的部门层级结构:
| OrgID | OrgName | parentID |
|---|---|---|
| A001 | Dept | 0 |
| A002 | subDept1 | A001 |
| A003 | sub_subDept | A002 |
| A006 | gran_subDept | A003 |
| A004 | subDept2 | A001 |
其中 parentID 为 "0" 的节点被视为根节点。我们希望将其转换为如下所示的树形结构:
Dept --subDept1 ----sub_subDept ------gran_subDept --subDept2
为了在Go语言中表示树形结构,我们需要定义一个节点类型,该节点包含其自身的名称以及一个指向其所有子节点的切片。同时,为了高效地根据ID查找节点,我们还需要一个全局的哈希映射。
立即学习“go语言免费学习笔记(深入)”;
package main
import (
"bufio"
"fmt"
"io"
"os"
"strings"
)
// Node 定义了树中的一个节点
type Node struct {
Name string // 节点的名称
Children []*Node // 节点的子节点列表
}
var (
// nodeTable 用于通过ID快速查找已创建的节点
nodeTable = map[string]*Node{}
// root 指向树的根节点。本示例假设只有一个根节点。
root *Node
)解释:
add 函数负责创建新节点,并将其正确地插入到树中。它根据 parentID 判断节点是根节点还是普通子节点。
// add 函数用于创建新节点并将其添加到树结构中
// id: 当前节点的唯一标识
// name: 当前节点的显示名称
// parentId: 当前节点的父节点标识
func add(id, name, parentId string) {
fmt.Printf("add: id=%v name=%v parentId=%v\n", id, name, parentId)
// 创建一个新的节点实例
node := &Node{Name: name, Children: []*Node{}}
// 根据 parentId 判断节点类型
if parentId == "0" {
// 如果 parentId 为 "0",则当前节点是根节点
root = node
} else {
// 否则,当前节点是子节点,需要找到其父节点
parent, ok := nodeTable[parentId]
if !ok {
// 如果父节点不存在,打印错误并跳过
fmt.Printf("add: parentId=%v: not found, skipping node %v\n", parentId, id)
return
}
// 将当前节点添加到父节点的 Children 列表中
parent.Children = append(parent.Children, node)
}
// 将新创建的节点添加到 nodeTable 中,以便后续查找
nodeTable[id] = node
}解释:
scan 函数负责从标准输入读取数据,解析每一行,并调用 add 函数来构建树。
// scan 函数从标准输入读取数据,解析每行并调用 add 函数构建树
func scan() {
input := os.Stdin
reader := bufio.NewReader(input)
lineCount := 0
for {
lineCount++
line, err := reader.ReadString('\n') // 读取一行直到换行符
if err == io.EOF {
break // 文件结束
}
if err != nil {
fmt.Printf("error reading lines: %v\n", err)
return
}
// 使用 strings.Fields 分割字符串,默认按空格或制表符分割
tokens := strings.Fields(line)
if t := len(tokens); t != 3 {
// 检查每行是否有3个字段 (OrgID, OrgName, parentID)
fmt.Printf("bad input line %v: tokens=%d [%v], expected 3 fields\n", lineCount, t, line)
continue
}
// 调用 add 函数处理当前行数据
add(tokens[0], tokens[1], tokens[2])
}
}解释:
showNode 函数使用递归的方式遍历树并打印节点名称,通过 prefix 参数实现层级缩进。show 函数作为入口,调用 showNode 从根节点开始展示。
Flex是一个基于组件的开发框架,可以生成一个由Flash Player运行的富互联网应用程序。Flex将基于标准的语言和各种可扩展用户界面及数据访问组件结合起来,使得开发人员能够构建具有丰富数据演示、强大客户端逻辑和集成多媒体的应用程序。 Flex是一个建立在Flash平台上的富客户端应用开发工具包,Flex 作为富 Internet 应用(RIA)时代的新技术代表,自从 2007 年 Adobe 公司将其开源以来,Flex 就以前所未有的速度在成长。感兴趣的朋友可以过来看看
0
// showNode 递归地显示树中的每个节点及其子节点
// node: 当前要显示的节点
// prefix: 用于缩进的字符串,表示当前节点的层级
func showNode(node *Node, prefix string) {
if prefix == "" {
// 根节点不加前缀,单独打印
fmt.Printf("%v\n", node.Name)
} else {
// 子节点加上缩进前缀
fmt.Printf("%v%v\n", prefix, node.Name)
}
// 递归遍历所有子节点
for _, n := range node.Children {
showNode(n, prefix+"--") // 子节点的缩进增加 "--"
}
}
// show 函数是显示树结构的入口
func show() {
if root == nil {
fmt.Printf("show: root node not found, tree is empty\n")
return
}
fmt.Printf("\nRESULT:\n")
showNode(root, "") // 从根节点开始显示,初始前缀为空
}解释:
将上述所有部分组合起来,构成一个完整的Go程序:
package main
import (
"bufio"
"fmt"
"io"
"os"
"strings"
)
// Node 定义了树中的一个节点
type Node struct {
Name string // 节点的名称
Children []*Node // 节点的子节点列表
}
var (
// nodeTable 用于通过ID快速查找已创建的节点
nodeTable = map[string]*Node{}
// root 指向树的根节点。本示例假设只有一个根节点。
root *Node
)
// add 函数用于创建新节点并将其添加到树结构中
// id: 当前节点的唯一标识
// name: 当前节点的显示名称
// parentId: 当前节点的父节点标识
func add(id, name, parentId string) {
fmt.Printf("add: id=%v name=%v parentId=%v\n", id, name, parentId)
// 创建一个新的节点实例
node := &Node{Name: name, Children: []*Node{}}
// 根据 parentId 判断节点类型
if parentId == "0" {
// 如果 parentId 为 "0",则当前节点是根节点
root = node
} else {
// 否则,当前节点是子节点,需要找到其父节点
parent, ok := nodeTable[parentId]
if !ok {
// 如果父节点不存在,打印错误并跳过
fmt.Printf("add: parentId=%v: not found, skipping node %v\n", parentId, id)
return
}
// 将当前节点添加到父节点的 Children 列表中
parent.Children = append(parent.Children, node)
}
// 将新创建的节点添加到 nodeTable 中,以便后续查找
nodeTable[id] = node
}
// scan 函数从标准输入读取数据,解析每行并调用 add 函数构建树
func scan() {
input := os.Stdin
reader := bufio.NewReader(input)
lineCount := 0
for {
lineCount++
line, err := reader.ReadString('\n') // 读取一行直到换行符
if err == io.EOF {
break // 文件结束
}
if err != nil {
fmt.Printf("error reading lines: %v\n", err)
return
}
// 使用 strings.Fields 分割字符串,默认按空格或制表符分割
tokens := strings.Fields(line)
if t := len(tokens); t != 3 {
// 检查每行是否有3个字段 (OrgID, OrgName, parentID)
fmt.Printf("bad input line %v: tokens=%d [%v], expected 3 fields\n", lineCount, t, line)
continue
}
// 调用 add 函数处理当前行数据
add(tokens[0], tokens[1], tokens[2])
}
}
// showNode 递归地显示树中的每个节点及其子节点
// node: 当前要显示的节点
// prefix: 用于缩进的字符串,表示当前节点的层级
func showNode(node *Node, prefix string) {
if prefix == "" {
// 根节点不加前缀,单独打印
fmt.Printf("%v\n", node.Name)
} else {
// 子节点加上缩进前缀
fmt.Printf("%v%v\n", prefix, node.Name)
}
// 递归遍历所有子节点
for _, n := range node.Children {
showNode(n, prefix+"--") // 子节点的缩进增加 "--"
}
}
// show 函数是显示树结构的入口
func show() {
if root == nil {
fmt.Printf("show: root node not found, tree is empty\n")
return
}
fmt.Printf("\nRESULT:\n")
showNode(root, "") // 从根节点开始显示,初始前缀为空
}
// main 函数是程序的入口点
func main() {
fmt.Printf("main: reading input from stdin\n")
scan() // 读取并解析输入数据
fmt.Printf("main: reading input from stdin -- done\n")
show() // 显示构建好的树结构
fmt.Printf("main: end\n")
}
保存代码: 将上述完整代码保存为 tree_builder.go 文件。
准备输入数据: 创建一个文本文件,例如 input.txt,将您的扁平化表格数据粘贴进去。确保每行包含三个由空格分隔的字段:OrgID OrgName parentID。
A001 Dept 0 A002 subDept1 A001 A003 sub_subDept A002 A006 gran_subDept A003 A004 subDept2 A001
编译程序: 打开终端或命令行,导航到 tree_builder.go 文件所在的目录,然后执行以下命令:
go build -o tree_builder
这会生成一个名为 tree_builder (Windows 上是 tree_builder.exe) 的可执行文件。
运行程序: 通过管道将 input.txt 的内容作为标准输入传递给程序:
cat input.txt | ./tree_builder
或者在 Windows 上:
type input.txt | .\tree_builder.exe
您将看到程序打印的调试信息和最终的树形结构输出:
main: reading input from stdin add: id=A001 name=Dept parentId=0 add: id=A002 name=subDept1 parentId=A001 add: id=A003 name=sub_subDept parentId=A002 add: id=A006 name=gran_subDept parentId=A003 add: id=A004 name=subDept2 parentId=A001 main: reading input from stdin -- done RESULT: Dept --subDept1 ----sub_subDept ------gran_subDept --subDept2 main: end
以上就是使用Go语言将扁平化数据构建为层级树结构的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号