go语言|数据结构:单链表(1)

简介: go语言|数据结构:单链表(1)

链表


一种物理存储单元上非连续、非顺序的存储结构,数据元素的逻辑顺序是通过链表中的指针链接次序实现的。链表由一系列结点(链表中每一个元素称为结点)组成,结点可以在运行时动态生成。每个结点包括两个部分:一个是存储数据元素的数据域,另一个是存储下一个结点地址的指针域。使用链表结构可以避免在使用数组时需要预先知道数据大小的缺点,链表结构可以充分利用计算机内存空间,实现灵活的内存动态管理。但是链表失去了数组随机读取的优点,同时链表由于增加了结点的指针域,空间开销比较大。


102dbb7a9f67499f9e2b21a3504ae8fb.png



单链表结构


 利用 struct 可以包容多种数据类型,结构体内也可以包含多个成员,这些成员可以是基本类型、自定义类型、数组类型,也可以是指针类型。这里可以使用指针类型成员来存放下一个结点的地址。如以下定义,成员 data 用来存放结点中的数据(整数类型),next 是指针类型的成员,它指向 ListNode struct 类型数据,也就是下一个结点的数据类型。


1. type ListNode struct {
2.  data int
3.  next *ListNode
4. }



创建节点

节点声明和赋值有以下几种格式:

package main
import "fmt"
type ListNode struct {
  data int
  next *ListNode
}
func main() {
  var head *ListNode
  head = new(ListNode)
  head.data = 1
  var node1 = new(ListNode)
  node1.data = 2
  var node2 = &ListNode{3, nil}
  var node3 = &ListNode{data: 4}
  fmt.Println(*head)
  fmt.Println(*node1)
  fmt.Println(*node2)
  fmt.Println(*node3)
}
/* 输出:
{1 <nil>}
{2 <nil>}
{3 <nil>}
{4 <nil>}
*/


遍历链表

一个for循环即可,结构描述的链表没有空链表的,不论data是何种类型,一旦声明即使不马上赋值也会有类型默认值,比如new(ListNode)即赋值了ListNode{0, nil}。

func showNode(p *ListNode) {
  fmt.Print(*p)
  for p.next != nil {
    p = p.next
    fmt.Print("->", *p)
  }
  fmt.Println()
}



头插法

新结点放在链表的最前面

package main
import "fmt"
type ListNode struct {
  data int
  next *ListNode
}
func showNode(p *ListNode) {
  fmt.Print(*p)
  for p.next != nil {
    p = p.next
    fmt.Print("->", *p)
  }
  fmt.Println()
}
func main() {
  var head = &ListNode{0, nil}
  for i := 1; i < 5; i++ {
    var node = ListNode{data: i}
    node.next = head
    head = &node
  }
  showNode(head)
}
/* 输出:
{4 0xc000084250}->{3 0xc000084240}->{2 0xc000084230}->{1 0xc000084220}->{0 <nil>}
*/



尾插法

新结点追加到链表的最后面

package main
import "fmt"
type ListNode struct {
  data int
  next *ListNode
}
func showNode(p *ListNode) {
  fmt.Print(*p)
  for p.next != nil {
    p = p.next
    fmt.Print("->", *p)
  }
  fmt.Println()
}
func main() {
  var head, tail *ListNode
  head = &ListNode{0, nil}
  tail = head
  for i := 1; i < 5; i++ {
    var node = ListNode{data: i}
    (*tail).next = &node
    tail = &node
  }
  showNode(head)
}
/* 输出:
{0 0xc000084220}->{1 0xc000084230}->{2 0xc000084240}->{3 0xc000084250}->{4 <nil>}
*/



遍历方法

方法的定义:参数表放在函数名前

package main
import "fmt"
type ListNode struct {
  data int
  next *ListNode
}
func (p *ListNode) travel() {
  fmt.Print(p.data)
  for p.next != nil {
    p = p.next
    fmt.Print("->", p.data)
  }
  fmt.Println("<nil>")
}
func main() {
  var head = &ListNode{0, nil}
    head.travel()
  for i := 1; i < 10; i++ {
    var node = ListNode{data: i}
    node.next = head
    head = &node
  }
  head.travel()
  var root *ListNode
  root = new(ListNode)
  root.travel()
}
/* 输出:
0<nil>
9->8->7->6->5->4->3->2->1->0<nil>
0<nil>
*/



链表长度

注意:函数与方法的区别

package main
import "fmt"
type ListNode struct {
  data int
  next *ListNode
}
func (head *ListNode) size() int {
  size := 1
  for head = head.next; head != nil; size++ {
    head = head.next
  }
  return size
}
func Len(head *ListNode) int {
  size := 1
  for head = head.next; head != nil; size++ {
    head = head.next
  }
  return size
}
func main() {
  var head = &ListNode{0, nil}
  fmt.Println(Len(head))
  fmt.Println(head.size())
  for i := 1; i < 10; i++ {
    var node = ListNode{data: i}
    node.next = head
    head = &node
  }
  fmt.Println(Len(head))
  fmt.Println(head.size())
}
/* 输出:
1
1
10
10
*/


链表转数组

package main
import (
  "fmt"
)
type ListNode struct {
  data int
  next *ListNode
}
func (head *ListNode) size() int {
  size := 1
  for head = head.next; head != nil; size++ {
    head = head.next
  }
  return size
}
func (head *ListNode) tolist() []int {
  var res []int
  res = make([]int, 0, head.size())
  for head.next != nil {
    res = append(res, head.data)
    head = head.next
  }
  res = append(res, head.data)
  return res
}
func (head *ListNode) tolist2() []int {
  var res []int
  res = make([]int, 0, head.size())
  res = append(res, head.data)
  head = head.next
  for head != nil {
    res = append(res, head.data)
    head = head.next
  }
  return res
}
func main() {
  var head = &ListNode{0, nil}
  for i := 1; i < 10; i++ {
    var node = ListNode{data: i}
    node.next = head
    head = &node
  }
  fmt.Println(head.tolist())
  var root, tail *ListNode
  root = &ListNode{0, nil}
  tail = root
  for i := 1; i < 10; i++ {
    var node = ListNode{data: i}
    (*tail).next = &node
    tail = &node
  }
  fmt.Println(root.tolist2())
}
/* 输出:
[9 8 7 6 5 4 3 2 1 0]
[0 1 2 3 4 5 6 7 8 9]
*/


数组转链表

package main
import "fmt"
type ListNode struct {
  data int
  next *ListNode
}
func (p *ListNode) travel() {
  fmt.Print(p.data)
  for p.next != nil {
    p = p.next
    fmt.Print("->", p.data)
  }
  fmt.Println("<nil>")
}
func toNode(list []int) *ListNode {
  var head, tail *ListNode
  head = &ListNode{list[0], nil}
  tail = head
  for i := 1; i < len(list); i++ {
    var node = ListNode{data: list[i]}
    (*tail).next = &node
    tail = &node
  }
  return head
}
func main() {
  var lst = []int{1, 3, 2, 3, 5, 6, 6, 8, 9}
  toNode(lst).travel()
}
/* 输出:
1->3->2->3->5->6->6->8->9<nil>
*/



例题


实战一:

给定一个已排序链表,删除重复节点,原始链表中多次出现的数字只能保留一次。


示例1

输入: 1->1->2

输出: 1->2


示例2

输入: 1->1->2->3->3

输出: 1->2->3



实战二:

给定一个排序链表,删除所有重复的节点,留原始链表有过重复的数字一个也不留。

示例1

输入: 1->2->3->3->4->4->5

输出: 1->2->5

示例2

输入: 1->1->1->2->3

输出: 2->3



实战三:

给定两个有序链表(升序),合并为一个新的有序链表并返回。

示例1

输入:1->2>4->8

  1->3->3->5->5

输出:1->1->2->3->3->4->5->5->8

示例2

输入:0->2>4->8

  1->3->5->7->9

输出:0->1->2->3->4->5->6->7->8->9



目录
相关文章
|
30天前
|
存储 监控 算法
员工上网行为监控中的Go语言算法:布隆过滤器的应用
在信息化高速发展的时代,企业上网行为监管至关重要。布隆过滤器作为一种高效、节省空间的概率性数据结构,适用于大规模URL查询与匹配,是实现精准上网行为管理的理想选择。本文探讨了布隆过滤器的原理及其优缺点,并展示了如何使用Go语言实现该算法,以提升企业网络管理效率和安全性。尽管存在误报等局限性,但合理配置下,布隆过滤器为企业提供了经济有效的解决方案。
79 8
员工上网行为监控中的Go语言算法:布隆过滤器的应用
|
10天前
|
算法 安全 Go
Go语言中的加密和解密是如何实现的?
Go语言通过标准库中的`crypto`包提供丰富的加密和解密功能,包括对称加密(如AES)、非对称加密(如RSA、ECDSA)及散列函数(如SHA256)。`encoding/base64`包则用于Base64编码与解码。开发者可根据需求选择合适的算法和密钥,使用这些包进行加密操作。示例代码展示了如何使用`crypto/aes`包实现对称加密。加密和解密操作涉及敏感数据处理,需格外注意安全性。
33 14
|
10天前
|
Go 数据库
Go语言中的包(package)是如何组织的?
在Go语言中,包是代码组织和管理的基本单元,用于集合相关函数、类型和变量,便于复用和维护。包通过目录结构、文件命名、初始化函数(`init`)及导出规则来管理命名空间和依赖关系。合理的包组织能提高代码的可读性、可维护性和可复用性,减少耦合度。例如,`stringutils`包提供字符串处理函数,主程序导入使用这些函数,使代码结构清晰易懂。
50 11
|
10天前
|
存储 安全 Go
Go语言中的map数据结构是如何实现的?
Go 语言中的 `map` 是基于哈希表实现的键值对数据结构,支持快速查找、插入和删除操作。其原理涉及哈希函数、桶(Bucket)、动态扩容和哈希冲突处理等关键机制,平均时间复杂度为 O(1)。为了确保线程安全,Go 提供了 `sync.Map` 类型,通过分段锁实现并发访问的安全性。示例代码展示了如何使用自定义结构体和切片模拟 `map` 功能,以及如何使用 `sync.Map` 进行线程安全的操作。
|
15天前
|
监控 安全 算法
深度剖析核心科技:Go 语言赋能局域网管理监控软件进阶之旅
在局域网管理监控中,跳表作为一种高效的数据结构,能显著提升流量索引和查询效率。基于Go语言的跳表实现,通过随机化索引层生成、插入和搜索功能,在高并发场景下展现卓越性能。跳表将查询时间复杂度优化至O(log n),助力实时监控异常流量,保障网络安全与稳定。示例代码展示了其在实际应用中的精妙之处。
36 9
|
24天前
|
算法 安全 Go
Go 语言中实现 RSA 加解密、签名验证算法
随着互联网的发展,安全需求日益增长。非对称加密算法RSA成为密码学中的重要代表。本文介绍如何使用Go语言和[forgoer/openssl](https://github.com/forgoer/openssl)库简化RSA加解密操作,包括秘钥生成、加解密及签名验证。该库还支持AES、DES等常用算法,安装简便,代码示例清晰易懂。
58 12
|
27天前
|
监控 算法 安全
解锁企业计算机监控的关键:基于 Go 语言的精准洞察算法
企业计算机监控在数字化浪潮下至关重要,旨在保障信息资产安全与高效运营。利用Go语言的并发编程和系统交互能力,通过进程监控、网络行为分析及应用程序使用记录等手段,实时掌握计算机运行状态。具体实现包括获取进程信息、解析网络数据包、记录应用使用时长等,确保企业信息安全合规,提升工作效率。本文转载自:[VIPShare](https://www.vipshare.com)。
32 1
|
1月前
|
Go 数据安全/隐私保护 UED
优化Go语言中的网络连接:设置代理超时参数
优化Go语言中的网络连接:设置代理超时参数
|
1月前
|
存储 Go 索引
go语言中数组和切片
go语言中数组和切片
46 7