面试算法-链表-反转链表(golang、c++)

目录

1、题目

2、解题思路

[2.1 遍历、迭代](#2.1 遍历、迭代)

[2.2 递归](#2.2 递归)

3、源代码

[3.1 c++](#3.1 c++)

[3.2 golang](#3.2 golang)

4、复杂度分析

[4.1 遍历、迭代法](#4.1 遍历、迭代法)

[4.2 迭代法](#4.2 迭代法)

1、题目

链表是一种常用的数据结构,链表的特点是插入、删除节点的效率非常高,因为他不需要移动其他任何元素,只需要改变节点的指向接口,但是他的缺点也很明显,访问任意节点,都需要从链表头遍历,时间复杂度O(n)。

题目:给你单链表的头节点 head ,请你反转链表,并返回反转后的链表。

示例一:

复制代码
输入:head = [1,2,3,4,5]
输出:[5,4,3,2,1]

示例二:

复制代码
输入:head = []
输出:[]

2、解题思路

2.1 遍历、迭代

这种方法,说的直白点,就是硬拧,把链表指向掉头。

解题过程:

  1. 从头结点开始,遍历每个节点。
  2. 保存当前节点的下一个节点cur_next。
  3. 将当前节点的next指向前一个节点pre,
  4. 将pre指向当前节点,将当前节点执行cur_next。
  5. 返回新链表头结点。

2.2 递归

递归的本质在于反向工作,假设有链表:

n1->n2-> n3->......->nk->nk+1->......->nm.

nk之后的链表已经逆序完成,现在只需要将nk+1节点的next指向nk即可,为了避免nk、nk+1两个节点互指,也需要将nk的next指向null。

3、源代码

3.1 c++

  • 遍历、迭代:
cpp 复制代码
struct ListNode
{
    int data;
    ListNode *next;
};
// 反转链表-遍历、迭代
ListNode *reverseList(ListNode *head)
{
    ListNode *pre = nullptr;
    ListNode *cur = head;
    if (head == nullptr || head->next == nullptr)
    {
        return head;
    }
    while (cur)
    {
        ListNode *next = cur->next;
        cur->next = pre;
        pre = cur;
        cur = next;
    }
    return pre;
}
  • 递归法:
cpp 复制代码
// 反转链表-递归
ListNode *reverseList2(ListNode *head)
{
    ListNode *cur = nullptr;
    if (head == nullptr || head->next == nullptr)
    {
        return head;
    }
    cur = reverseList2(head->next);
    head->next->next = head;
    head->next = nullptr;
    return cur;
}

3.2 golang

  • 遍历法:
Go 复制代码
func Reverse(head *ListNode) *ListNode {
	var pre, next *ListNode
	cur := head
	for cur != nil {
		next = cur.Next
		cur.Next = pre
		pre = cur
		cur = next
	}
	return pre
}
  • 递归法:
Go 复制代码
// 反转链表--递归
func Reverse2(head *ListNode) *ListNode {
	if head == nil || head.Next == nil {
		return head
	}
	newNode := Reverse2(head.Next)
	head.Next.Next = head
	head.Next = nil // 防止循环
	return newNode
}

4、复杂度分析

4.1 遍历、迭代法

时间复杂度:遍历法,需要遍历整个链表,因此时间复杂度为:O(n),n为链表长度。

空间复杂度:程序运行整个过程中,没有申请新的内存,因此空间复杂度为:O(1)。

4.2 迭代法

时间复杂度:递归算法仍然需要遍历整个链表,因此时间复杂度为:O(n),n为链表长度。

空间复杂度:递归需要申请栈空间来保存函数调用关系,因此空间复杂度为:O(n),最多n层调用。

相关推荐
凤年徐23 分钟前
【数据结构初阶】单链表
c语言·开发语言·数据结构·c++·经验分享·笔记·链表
木子.李3474 小时前
排序算法总结(C++)
c++·算法·排序算法
闪电麦坤955 小时前
数据结构:递归的种类(Types of Recursion)
数据结构·算法
Gyoku Mint6 小时前
机器学习×第二卷:概念下篇——她不再只是模仿,而是开始决定怎么靠近你
人工智能·python·算法·机器学习·pandas·ai编程·matplotlib
纪元A梦6 小时前
分布式拜占庭容错算法——PBFT算法深度解析
java·分布式·算法
px不是xp7 小时前
山东大学算法设计与分析复习笔记
笔记·算法·贪心算法·动态规划·图搜索算法
枫景Maple7 小时前
LeetCode 2297. 跳跃游戏 VIII(中等)
算法·leetcode
鑫鑫向栄7 小时前
[蓝桥杯]修改数组
数据结构·c++·算法·蓝桥杯·动态规划
鑫鑫向栄8 小时前
[蓝桥杯]带分数
数据结构·c++·算法·职场和发展·蓝桥杯