一、概念介绍

下面这副图是我们单链表运煤车队。

每节运煤车就是单链表里的元素,每节车厢里的煤炭就是元素中保存的数据。前后车通过锁链相连,作为单链表运煤车,从1号车厢开始,每节车厢都知道后面拉着哪一节车厢,却不知道前面是哪节车厢拉的自己。第一节车厢没有任何车厢拉它,我们就叫它车头,第五节车厢后面拉其他车厢,我们称为车尾。

作为单链表它最大的特点就是能随意增加车队的长度,也能随意减少车队的长度。这是比数组公交车最大的优点。

二、Go语言实现讲解

1、节点



每节车厢都由车体、绳索和煤炭构成。在Go语言中表示这种自定义组合体的类型就是结构,当然为了通用性,我们这里要把车厢转换成节点也就是元素,煤炭转换成数据,绳索转换成指针。

    type Node struct {
data Object
next *Node
}

用张图来描述这种对应关系。



这里结构体Node表示车厢,data表示煤炭用Object类型,next是牵着下节车厢的绳索,用指针表示。

对于车厢来说,除了放煤炭外,还能放瓜果、衣物、饭菜等等,所以这里data的类型必须通用,当然Go里是没有Java里的Object类型的,所以我们就自己定义了一个。

    type Object interface{}

2、链表



光有车厢还不够,我们还要描述车厢组成的车队。每个单链表车队都有车头、车尾和车厢数量,我们同样以结构来表现。

    type List struct {
size uint64 // 车辆数量
head *Node // 车头
tail *Node // 车尾
}

3、方法

(1)初始化

第一步就是组装单链表车队,这就是初始化。不过开始的时候车队是个空壳,一个车厢没有。

    func (list *List) Init() {
(*list).size = 0 // 此时链表是空的
(*list).head = nil // 没有车头
(*list).tail = nil // 没有车尾
}

(2)添加元素

当前的单链表是空的,所以我们要向里面添加元素。

    func (list *List) Append(node *Node) {
(*list).head = node // 这是单链表的第一个元素,也是链表的头部
(*list).tail = node // 同时是单链表的尾部
(*list).size = 1 // 单链表有了第一个元素
}

现在单链表有了第一个元素,我还想再添加一个元素,当然是添加到单链表尾部。

    func (list *List) Append(node *Node) {
if (*list).size == 0 { // 无元素的时候添加
(*list).head = node // 这是单链表的第一个元素,也是链表的头部
(*list).tail = node // 同时是单链表的尾部
(*list).size = 1 // 单链表有了第一个元素
} else { // 有元素了再添加
oldTail := (*list).tail
(*oldTail).next = node // node放到尾部元素后面
(*list).tail = node // node成为新的尾部
(*list).size++ // 元素数量增加
}
}

分析上面的代码存在3处疑点,元芳你怎么看?属下认为

第一,node如果为空,则添加无任何意义;

第二,代码中存在重复的地方;

这第三么,卑职如何才能知道新增结果?

下面大卫哥顺着元芳的思路改进下代码。

    func (list *List) Append(node *Node) bool {
if node == nil {
return false
} (*node).next = nil
// 将新元素放入单链表中
if (*list).size == 0 {
(*list).head = node
} else {
oldTail := (*list).tail
(*oldTail).next = node
} // 调整尾部位置,及链表元素数量
(*list).tail = node // node成为新的尾部
(*list).size++ // 元素数量增加 return true
}

(3)插入元素

一天领导让大卫哥把他小舅子安排到第一个,于是大卫哥为了拍好领导马屁。绞尽脑汁弄了一个方法。

   func (list *List) Insert(node *Node) bool {
if node == nil {
return false
} (*node).next = (*list).head // 领导小舅子排到之前第一名前面
(*list).head = node // 领导小舅子成为第一名
(*list).size++
return true
}

来托关系插队的人越来越多,领导的关系户不能动,只能插后面人的队了。于是大卫哥又修改了代码,增加了位置参数。

    func (list *List) Insert(i uint,node *Node) bool {
// 空的节点、索引超出范围和空链表都无法做插入操作
if node == nil || i > (*list).size || (*list).size == 0 {
return false
} if i == 0 { // 直接排第一,也就领导小舅子才可以
(*node).next = (*list).head
(*list).head = node
} else {
// 找到前一个元素
preItem := (*list).head
for j := 1 ; j < i; j++ { // 数前面i个元素
preItem = (*preItem).next
}
// 原来元素放到新元素后面,新元素放到前一个元素后面
(*node).next = (*preItem).next
(*preItem).next = preItem
} (*list).size++ return true
}

(4)删除元素

插队的关系户太多,影响了正常排队的人,被人投诉,大卫哥只好想办法删除一些。

    func (list *List) Remove(i uint, node *Node) bool {
if i >= (*list).size {
return false
} if i == 0 { // 删除头部
node = (*list).head
(*list).head = (*node).next
if (*list).size == 1 { // 如果只有一个元素,那尾部也要调整
(*list).tail = nil
}
} else {
preItem := (*list).head
for j := 1; j < i; j++ {
preItem = (*preItem).next
} node = (*preItem).next
(*preItem).next = (*node).next if i == ((*list).size - 1) { // 若删除的尾部,尾部指针需要调整
(*list).tail = preItem
}
}
(*list).size--
return true
}

(5)获取

为了获取某个位置的元素,我们需要一个方法。

    func (list *List) Get(i uint) *Node {
if i >= (*list).size {
return nil
} item := (*list).head
for j := 0; j < i ; j++ { // 从head数i个
item = (*item).next
} return item
}

到这里基本框架已经出来了,不过还有几个接口还没实现,作为课后作业。

三、小结

单链表就和列车类似,一个接着一个,所以本节从列车类比介绍了单链表的Go语言实现。在接口实现部分大卫哥以序号作为链表中每个节点的操作关键字。在实际应用中,我们往往以data中的某一个字段作为操作关键字。所以这也衍生出链表的不同接口,大家可以参考大卫哥留的链接中的代码实现作为理解。同时有些实现将表头独立出来并不存放数据,这在一定程度上简化了代码的实现。

代码下载

四、习题

(1)补全GetSize,RemoveAll,GetHead和GetTail的定义和实现。

(2)以data作为参数,考虑单链表的实现。

(3)将单链表的head独立出来,此时的head是独立的,不存放data,如下图,考虑单链表的实现,并比较这种实现。



(4)如果将head和tail都独立出来,都不存放data,此时的单链表如何实现?这样实现的代码在插入删除操作时候是不是更容易点?

第一节 如何用Go实现单链表的更多相关文章

  1. 动态单链表的传统存储方式和10种常见操作-C语言实现

    顺序线性表的优点:方便存取(随机的),特点是物理位置和逻辑为主都是连续的(相邻).但是也有不足,比如:前面的插入和删除算法,需要移动大量元素,浪费时间,那么链式线性表 (简称链表) 就能解决这个问题. ...

  2. Java单链表反转 详细过程

    版权声明:本文为博主原创文章,未经博主允许不得转载. https://blog.csdn.net/guyuealian/article/details/51119499 Java单链表反转 Java实 ...

  3. 【数据结构】单链表&&静态链表详解和代码实例

    喜欢的话可以扫码关注我们的公众号哦,更多精彩尽在微信公众号[程序猿声] 01 单链表(Singly Linked List ) 1.1 什么是单链表? 单链表是一种链式存储的结构.它动态的为节点分配存 ...

  4. c++单链表基本功能

    head_LinkNode.h /*单链表类的头文件*/#include<assert.h>#include"compare.h"typedef int status; ...

  5. C++ 单链表基本操作

    链表一直是面试的高频题,今天先总结一下单链表的使用,下节再总结双向链表的.本文主要有单链表的创建.插入.删除节点等. 1.概念 单链表是一种链式存取的数据结构,用一组地址任意的存储单元存放线性表中的数 ...

  6. js数据结构与算法--单链表的实现与应用思考

    链表是动态的数据结构,它的每个元素由一个存储元素本身的节点和一个指向下一个元素的引用(也称指针或链接)组成. 现实中,有一些链表的例子. 第一个就是寻宝的游戏.你有一条线索,这条线索是指向寻找下一条线 ...

  7. 数据结构(一) 单链表的实现-JAVA

    数据结构还是很重要的,就算不是那种很牛逼的,但起码得知道基础的东西,这一系列就算是复习一下以前学过的数据结构和填补自己在这一块的知识的空缺.加油.珍惜校园中自由学习的时光.按照链表.栈.队列.排序.数 ...

  8. JAVA实现具有迭代器的线性表(单链表)

    一,迭代器的基本知识: 1,为什么要用迭代器?(迭代:即对每一个元素进行一次“问候”) 比如说,我们定义了一个ADT(抽象数据类型),作为ADT的一种实现,如单链表.而单链表的基本操作中,大部分需要用 ...

  9. 数据结构C语言版--单链表的基本功能实现

    /* * 构造一个链式存储的线性表(当输入9999时,结束构造过程),然后输出该线性表 * 并统计该线性链表的长度 . *注:new和delete是C++的运算符 malloc和free是C++/C的 ...

随机推荐

  1. Python初学者第十三天 三级菜单程序小作业

    13day 作业题目: 三级菜单 作业需求: 数据结构: menu = { '北京':{ '海淀':{ '五道口':{ 'soho':{}, '网易':{}, 'google':{} }, '中关村' ...

  2. winform中webBrowser模拟网页操作中遇到的问题

    我们通过网页上传一些特殊数据的时候,由于必填项众多,数量量大的时候,会发现工作相当繁琐,前段时间做了一个winform内嵌webBrowser模拟网页上传文档的小工具,发现了许多问题,总结一下: 先说 ...

  3. unity3d项目版本管理设置

    unity3d老是有一堆乱七八糟的文件,好像不提交也不行,特别是那烦人的meta文件,哪到底unity项目提交到版本管理哪些东西可以忽略呢?应该设置些什么东西呢? 菜单,Edit => Proj ...

  4. 判断元素(expected_conditions)

    判断元素 如何判断一个元素是否存在,如何判断 alert 弹窗出来了,如何判断动态的元素等等一系列的判断,在 selenium 的 expected_conditions 模块收集了一系列的场景判断方 ...

  5. SGU---462 Electrician 最大生成树

    题目链接: https://cn.vjudge.net/problem/SGU-462 题目大意: 有N条电线需要接入电网,第i条电线计划连接ai和bi两个地点,电线有两个属性:ri(电线稳定度)和c ...

  6. Vue Spa切换页面时更改标题

    在Vue组件化开发过程中,因为是单页面开发,但是有时候需要页面的title根据情况改变,于是上网查了一下,各种说法花(wo)里(kan)胡(bu)哨(dong), 于是想到一个黑科技 documet. ...

  7. linux下搭建LAMP

    PHP命令找不到:  export PATH=$PATH:/usr/local/php/bin https://www.centos.bz/forum/thread-69-1-1.html 步骤: w ...

  8. 利用JDK自带工具keyTool生成安全证书

    前言:说一下最近做的工作,主要利用iText给网页中生成好的html报表转化为pdf格式的文件,并且在其中加入水印,数字签名等等,这部分主要介绍安全证书的目的就是为了做数字签名部分用的. 下面利用jd ...

  9. 基于MySql数据库的单表与多表联合查询

    这里以学生 班级 身份证 以及课程为例 1,启动MySql数据库  开启服务 2.1.0新建一张班级表 备注:CHARSET = UTF8 (指定编码格式为utf8 防止中文乱码) /*班级表*/ C ...

  10. import 本质

    一. 模块:用来从逻辑上来组织python代码(变量,函数,类,逻辑,实现一个功能),本质就是,py结尾的python文件 1.1 导入方法: import module import module1 ...