美文网首页golang进阶之路
Golang使用快慢指针找不知长度链表的中间节点

Golang使用快慢指针找不知长度链表的中间节点

作者: 不屈真实 | 来源:发表于2020-09-07 10:43 被阅读0次

线性表的链式存储,以及使用快慢指针找不知长度链表的中间节点

题目描述

首先,给定一个带有头结点 head 的不知长度链表,返回链表的中间结点。
如果有两个中间结点,则返回第二个中间结点。注意链表长度未知,这里测试定义的链表结构体存储的数据类型可以用字符串类型。
示例说明:
实例1
输入:“one”-"two"-"three"-"four"-"five"
输出:链表的中间节点为 ["three"]
实例2
输入:“tom”-"jack"-"kaka"-"James"-"vickor"-"wade"-"aha"-"tiger"
输出:链表的中间节点为 ["James","vickor"]

解题思路

设置两个指针,一个快指针,每次走两步,一个慢指针,每次走一步,当快指针为空(偶数个节点)或者快指针的next指针指向空时(奇数个节点),此时慢指针即为中间节点。
步骤:
首先定义线性表的链式存储,单个节点的定义如下
type Node struct { // 定义一个node结构体
data string
next *Node
}
紧接着定义相应的方法,主要有增加节点Append、获取长度GetLength、获取中间节点GetCenterNode等。
在链表尾部增加节点:
func (n *Node) Append(data string){
createNode := &Node{data,nil}
for n.next != nil{
n = n.next
}
n.next = createNode
}
获取整个链表的长度:
func (n *Node) GetLength() int{
if n.next==nil{
return 0
}
length:= 0
for n.next!=nil{
length ++
n = n.next
}
return length
}
获取整个链表中间节点:(这里有对不同情况分别讨论,如一些边界条件、链表长度为奇数偶数时分别讨论)
func (n Node)GetCenterNode()[]string{
str := make([]string,0,2)
if n.next==nil{
return nil
}
l := n.next
s := n.next
for s.next!= nil{
// 单数情况
if l.next==nil{
str = append(str, s.data)
return str
}
// 双数情况
if l.next != nil && l.next.next == nil{
str = append(str,s.data)
str = append(str,s.next.data)
return str
}
s =s.next
l = l.next.next
}
return str
}
首先初始化一个链表
// 初始化一个链表
func initLinkList ()
Node{
return &Node{"genesis node", nil}
}
通过随机数拼接生成一个长度为20的随机链表,用于测试
func Rand20List(n *Node){
for i:=0;i<3;i++{
n.Append("this is "+strconv.Itoa(i))
}
}
通过main函数进行测试,当然也可通过写测试函数进行测试。
func main(){
link := initLinkList()
var s int
for {
fmt.Println("1.查看链表")
fmt.Println("2.创建链表20个数据")
fmt.Println("3.链表长度")
fmt.Println("4.查看中间节点值")
fmt.Println("5.退出")
fmt.Println("请输入选择:")
fmt.Scan(&s)
switch s {
case 1:
fmt.Println(link.MapLinkList())
case 2:
Rand20List(link)
case 3:
fmt.Println(link.GetLength())
case 4:
fmt.Println(link.GetCenterNode())
case 5:
return
}
}
}

完整代码

完整线性表的链式存储,以及使用快慢指针找不知长度链表的中间节点代码如下:

package main
import (
    "fmt"
    "strconv"
)
// 定义一个node结构体
type Node struct {
    data string
    next *Node
}
func (n *Node) Append(data string){
    createNode := &Node{data,nil}
    for n.next != nil{
        n = n.next
    }
    n.next = createNode
}
func (n *Node) GetLength() int{
    if n.next==nil{
        return 0
    }
    length:= 0
    for n.next!=nil{
        length ++
        n = n.next
    }
    return length
}
func (n *Node) MapLinkList()[]string{
    var str []string
    if n.next == nil{
        return nil
    }
    for n.next != nil{
        str = append(str,n.next.data)
        n = n.next
    }
    return str
}
// 快慢指针法求不知道长度的链表的中间值
func (n *Node)GetCenterNode()[]string{
    str := make([]string,0,2)
    if n.next==nil{
        return nil
    }
    l := n.next
    s := n.next
    for s.next!= nil{
        // 单数情况
        if l.next==nil{
             str = append(str, s.data)
             return str
        }
        // 双数情况
        if l.next != nil && l.next.next == nil{
             str = append(str,s.data)
             str = append(str,s.next.data)
             return str
        }
        s =s.next
        l = l.next.next
    }
    return str
}
// 初始化一个链表
func initLinkList ()*Node{
    return &Node{"genesis node", nil}
}
func Rand20List(n *Node){
    for i:=0;i<3;i++{
        n.Append("this is "+strconv.Itoa(i))
    }
}
func main(){
    link := initLinkList()
    var s int
    for {
        fmt.Println("1.查看链表")
        fmt.Println("2.创建链表20个数据")
        fmt.Println("3.链表长度")
        fmt.Println("4.查看中间节点值")
        fmt.Println("5.退出")
        fmt.Println("请输入选择:")
        fmt.Scan(&s)
        switch s {
        case 1:
            fmt.Println(link.MapLinkList())
        case 2:
            Rand20List(link)
        case 3:
            fmt.Println(link.GetLength())
        case 4:
            fmt.Println(link.GetCenterNode())
        case 5:
            return
        }
    }
}

最后,通往北京的路有很多条,你也可以有其他更好的方法,欢迎一起交流学习,共同进步。

相关文章

  • Golang使用快慢指针找不知长度链表的中间节点

    线性表的链式存储,以及使用快慢指针找不知长度链表的中间节点 题目描述 首先,给定一个带有头结点 head 的不知长...

  • 数据结构与算法整理

    (1)链表的技巧 快慢指针(找环,环入口,环长度) 双指针(倒数K个节点) 合并链表(递归求解) 约瑟夫环(环形链...

  • 链表

    单向链表 链表反转 判断是否有环,找链表的中间节点 快慢指针 找环的入口(求两个链表的交点可以转化成这个问题) p...

  • 【Leetcode】19. 删除链表的倒数第N个节点

    要点:使用哑节点对应需要删除链表第一个节点情况优化:单次遍历:使用快慢指针。快慢指针的使用又打开了新世界大门! 第...

  • 0876-链表的中间结点

    链表的中间结点 方案一 使用快慢指针 借助单链表实现 C-源代码

  • 2019-06-17 根据有序的链表构建二叉搜索树

    每次找到中间节点,然后递归生成左右子树。 找到链表中间节点需要用快慢指针来找 slow fast:循环条件——fa...

  • 链表中间节点思路

    找链表的中间节点,可以依赖快慢指针来求解。思路是:一个慢指针,从head开始一次走 1 步;另外一个快指针,从he...

  • java算法:找到单链表的中间节点

    实现思路:通过快慢指针,快指针每次跳两个节点,慢指针每次跳一个节点,当快指针移动到末尾时,慢指针整好指向链表的中间节点。

  • 67. LeetCode 234. 回文链表

    标签: 链表 双指针 难度: 简单 题目描述 我的解法 【step1】先用快慢指针(p2, p3)确定中间节点...

  • 68. LeetCode 143. 重排链表

    标签: 链表 双指针 难度: 中等 题目描述 我的解法 【step1】先用快慢指针(p1, p2)确定中间节点...

网友评论

    本文标题:Golang使用快慢指针找不知长度链表的中间节点

    本文链接:https://www.haomeiwen.com/subject/syykektx.html