【面试必刷TOP101】反转链表 & 链表内指定区间反转

简介: 【面试必刷TOP101】反转链表 & 链表内指定区间反转

题目:反转链表_牛客题霸_牛客网 (nowcoder.com)

题目的接口:

package main
import "fmt"
import . "nc_tools"
/*
 * type ListNode struct{
 *   Val int
 *   Next *ListNode
 * }
 */
/**
 * 代码中的类名、方法名、参数名已经指定,请勿修改,直接返回方法规定的值即可
 *
 * 
 * @param head ListNode类 
 * @return ListNode类
*/
func ReverseList( head *ListNode ) *ListNode {
    // write code here
}

解题思路:

这道题我能想到两个思路,第一个思路是不断遍历原链表,然后通过尾插的方式新建一个链表,但是这样的时间复杂度会变得很高,因为我们需要从遍历找尾开始,再尾插,而且代码也比较繁杂。

第二个方法就是通过反转指针的方式,只需要遍历一次链表,将每一个节点的指针反转过来就行了,具体操作是这样的:用 pNext 记录下一个节点,让当前节点指向 newNode, 如图:

第一步:(head 从指向 pNext 改成指向 newNode)

也就是现在的新链表是这样的:

第二步:(newNode = head,让 newNode 更新成新链表的头节点)

最后再让 head = pNext,就能继续往后遍历链表了。

代码:

package main
import . "nc_tools"
/*
 * type ListNode struct{
 *   Val int
 *   Next *ListNode
 * }
 */
/**
 * 代码中的类名、方法名、参数名已经指定,请勿修改,直接返回方法规定的值即可
 *
 * 
 * @param head ListNode类 
 * @return ListNode类
*/
func ReverseList( head *ListNode ) *ListNode {
    if head == nil || head.Next == nil {
        return head
    }
    var newHead *ListNode
    for head != nil {
        pNext := head.Next   // 保存下一个节点
        head.Next = newHead; // 将当前节点指向一个新节点
        newHead = head       // 更新反转链表
        head = pNext         // 更新当前节点
    }
    return newHead
}

过啦!!!

题目:链表内指定区间反转_牛客题霸_牛客网 (nowcoder.com)

题目的接口:

package main
import . "nc_tools"
/*
 * type ListNode struct{
 *   Val int
 *   Next *ListNode
 * }
 */
/**
 * 代码中的类名、方法名、参数名已经指定,请勿修改,直接返回方法规定的值即可
 *
 * 
 * @param head ListNode类 
 * @param m int整型 
 * @param n int整型 
 * @return ListNode类
*/
func reverseBetween( head *ListNode ,  m int ,  n int ) *ListNode {
    // write code here
}

解题思路:

这道题的核心思路如下:

首先先找到需要反转节点的位置,然后开始操作:

pre 节点作为反转链表的前一个节点,并保持不变,cur 节点作为反转链表的第一个节点,tmp 节点不断将下一个节点转移到前面,也就是反转的过程,我们以题目的样例为例:

进行一次操作之后,pre 保持在反转链表的前一个节点,cur 保持在反转链表的头结点位置,tmp 就出现在下一个节点的位置,而 3 节点就被反转到了前面

继续进行操作:

还是一样的:

代码:

package main
import . "nc_tools"
/*
 * type ListNode struct{
 *   Val int
 *   Next *ListNode
 * }
 */
/**
 * 代码中的类名、方法名、参数名已经指定,请勿修改,直接返回方法规定的值即可
 *
 * 
 * @param head ListNode类 
 * @param m int整型 
 * @param n int整型 
 * @return ListNode类
*/
func reverseBetween( head *ListNode ,  m int ,  n int ) *ListNode {
    if head == nil || head.Next == nil {
        return head
    }
    // 找到反转链表的起始位置的前一个位置
    Node := &ListNode{ Next: head, }
    pre := Node
    for i := 1; i < m; i++ {
        pre = pre.Next
    }
    // 反转操作的主逻辑
    cur := pre.Next
    for i := m; i < n; i++ {
        tmp := cur.Next
        cur.Next = tmp.Next
        tmp.Next = pre.Next
        pre.Next = tmp
    }
    return Node.Next
}

过啦!!!

写在最后:

以上就是本篇文章的内容了,感谢你的阅读。

如果感到有所收获的话可以给博主点一个哦。

如果文章内容有遗漏或者错误的地方欢迎私信博主或者在评论区指出~

相关文章
|
8月前
|
索引
【力扣刷题】两数求和、移动零、相交链表、反转链表
【力扣刷题】两数求和、移动零、相交链表、反转链表
56 2
【力扣刷题】两数求和、移动零、相交链表、反转链表
|
3月前
|
存储 算法 安全
HashMap常见面试题(超全面):实现原理、扩容机制、链表何时升级为红黑树、死循环
HashMap常见面试题:红黑树、散列表,HashMap实现原理、扩容机制,HashMap的jd1.7与jdk1.8有什么区别,寻址算法、链表何时升级为红黑树、死循环
|
5月前
|
存储 算法 Python
【面试题】合井K个升序链表
【面试题】合井K个升序链表
37 0
|
5月前
|
存储 Java
【Java集合类面试十】、HashMap中的循环链表是如何产生的?
在多线程环境下,HashMap在扩容时如果发生条件竞争,元素的插入顺序可能形成循环链表,导致死循环。
|
7月前
|
存储 算法 数据可视化
【模拟面试问答】深入解析力扣163题:缺失的区间(线性扫描与双指针法详解)
【模拟面试问答】深入解析力扣163题:缺失的区间(线性扫描与双指针法详解)
|
8月前
<数据结构>五道LeetCode链表题分析.环形链表,反转链表,合并链表,找中间节点.
<数据结构>五道LeetCode链表题分析.环形链表,反转链表,合并链表,找中间节点
80 1
|
8月前
|
存储 算法 索引
链表面试题
链表面试题
链表面试题
|
7月前
|
存储 SQL 算法
LeetCode 83题:删除排序链表中的重复元素【面试】
LeetCode 83题:删除排序链表中的重复元素【面试】
|
8月前
【一刷《剑指Offer》】面试题 17:合并两个排序的链表
【一刷《剑指Offer》】面试题 17:合并两个排序的链表
|
8月前
【一刷《剑指Offer》】面试题 16:反转链表
【一刷《剑指Offer》】面试题 16:反转链表