五个经典链表OJ题带你进阶C++链表篇

目录
  • 反转单链表
  • 返回链表中间节点的位置
  • 合并两个有序链表
  • 判断链表中是否有环
  • 判断环形链表进入的节点

反转单链表

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

示例 1:

输入:head = [1,2,3,4,5]

输出:[5,4,3,2,1]

题目来源:力扣

思路一: 翻转指针方向,首先我们要有三个指针,这个就不展示代码了,逻辑过程如下:

思路二:头插方法,我们把每个节点拿下来进行头插实现!代码实现如下:

struct ListNode* reverseList(struct ListNode* head)
{
	struct ListNode* cur = head;
	struct ListNode* newHead = NULL;
	while (cur)
	{
		struct ListNode* next = cur->next;

		//头插
		cur->next = newHead;
		newHead = cur;
		cur = next;
	}
	return newHead;
}

返回链表中间节点的位置

题目2:给定一个头结点为 head 的非空单链表,返回链表的中间结点。

如果有两个中间结点,则返回第二个中间结点。

示例 1:

输入:[1,2,3,4,5,6]

输出:此列表中的结点 4 (序列化形式:[4,5,6])

由于该列表有两个中间结点,值分别为 3 和 4,我们返回第二个结点

题目来源:力扣

思路: 我们使用快慢指针的办法,快指针fast走两步,慢指针slow走一步,这样当fast走完了,slow指针就走到了中间的位置,但是我们要注意,如果链表节点为奇数个则当fast为NULL就应该结束循环,如果链表节点为偶数个则当fast->next为NULL则结束循环!思路解析,代码实现如下:

struct ListNode* middleNode(struct ListNode* head)
{
	struct ListNode* slow = head, * fast = head;
	while (fast && fast->next)
	{
		slow = slow->next;
		fast = fast->next->next;
	}
	return slow;
}

合并两个有序链表

题目3:将两个升序链表合并为一个新的 升序 链表并返回。新链表是通过拼接给定的两个链表的所有节点组成的。

示例 1:

输入:l1 = [1,2,4], l2 = [1,3,4]

题目来源:力扣

思路:首先我们要判断两个链表是否为空,如果为空则返回另一个链表!接着我们需要定义两个指针头指针head和一个尾指针tail,接着我们比较list1->val是否大于list2->val然后进行链接链表的操作,并且当其中一个链表为空则跳出循环,我们则需要在循环外再次判断是哪个链表为空导致跳出的循环,并且最后把不为空的链表链接在后面!最后返回头指针head!

建议小伙伴们看着这个思路尝试着自己写一下,可以参考如下代码:

struct ListNode* mergeTwoLists(struct ListNode* list1, struct ListNode* list2)
{
	if (list1 == NULL)
		return list2;
	if (list2 == NULL)
		return list1;
	struct ListNode* head = NULL, * tail = NULL;
	if (list1->val < list2->val)
	{
		head = tail = list1;
		list1 = list1->next;
	}
	else
	{
		head = tail = list2;
		list2 = list2->next;
	}

	while (list1 != NULL && list2 != NULL)
	{
		if (list1->val < list2->val)
		{
			//尾插
			tail->next = list1;
			list1 = list1->next;
		}
		else
		{
			tail->next = list2;
			list2 = list2->next;
		}
		tail = tail->next;

	}
	if (list1)
		tail->next = list1;
	if (list2)
		tail->next = list2;

	return head;
}

判断链表中是否有环

给你一个链表的头节点 head ,判断链表中是否有环。

如果链表中存在环 ,则返回 true 。 否则,返回 false 。

示例 1:

输入:head = [3,2,0,-4], pos = 1

输出:true

解释:链表中有一个环,其尾部连接到第二个节点。

题目来源:力扣

思路: 我们使用快慢指针的方法,让fast指针一次走两步,slow指针一次走一步,当链表有环的时候,当slow进入环了,fast就开始追slow,假设fast跟slow的距离为N,每走一次fast跟slow的距离就会缩小一步,也就是N-1,N-2,N-3,N-4,直到N为0 fast就追上slow了!代码实现如下:

bool hasCycle(struct ListNode *head)
{
    struct ListNode* fast = head;
    struct ListNode* slow = head;
    while(fast && fast->next)
    {
        slow = slow->next;
        fast = fast->next->next;
        if(slow == fast)
            return true;
    }
    return false;
}

判断环形链表进入的节点

给定一个链表的头节点  head ,返回链表开始入环的第一个节点。 如果链表无环,则返回 null。

不允许修改 链表。

示例 1:

输入:head = [3,2,0,-4], pos = 1

输出:返回索引为 1 的链表节点

解释:链表中有一个环,其尾部连接到第二个节点。

题目来源:力扣

思路:我们还是用跟上面一样的快慢指针的方法,但是在后面,一个指针从相遇点meet开始走,一个指针从链表头head开始走,他们会在入口点相遇!图解,代码参考见下:

bool hasCycle(struct ListNode *head)
{
    struct ListNode* fast = head;
    struct ListNode* slow = head;
    while(fast && fast->next)
    {
        slow = slow->next;
        fast = fast->next->next;
        if(slow == fast)
            return true;
    }
    return false;
}

(0)

相关推荐

  • C++实现LeetCode(160.求两个链表的交点)

    [LeetCode] 160.Intersection of Two Linked Lists 求两个链表的交点 Write a program to find the node at which the intersection of two singly linked lists begins. For example, the following two linked lists: A:          a1 → a2 c1 → c2 → c3             B:     b1

  • C++实现LeetCode(138.拷贝带有随机指针的链表)

    [LeetCode] 138. Copy List with Random Pointer 拷贝带有随机指针的链表 A linked list is given such that each node contains an additional random pointer which could point to any node in the list or null. Return a deep copy of the list. Example 1: Input: {"$id"

  • C++实现LeetCode(148.链表排序)

    [LeetCode] 148. Sort List 链表排序 Sort a linked list in O(n log n) time using constant space complexity. Example 1: Input: 4->2->1->3 Output: 1->2->3->4 Example 2: Input: -1->5->3->4->0 Output: -1->0->3->4->5 常见排序方法有

  • C++实现LeetCode(237.删除链表的节点)

    [LeetCode] 237.Delete Node in a Linked List 删除链表的节点 Write a function to delete a node (except the tail) in a singly linked list, given only access to that node. Supposed the linked list is 1 -> 2 -> 3 -> 4 and you are given the third node with va

  • C++实现LeetCode(147.链表插入排序)

    [LeetCode] 147. Insertion Sort List 链表插入排序 Sort a linked list using insertion sort. A graphical example of insertion sort. The partial sorted list (black) initially contains only the first element in the list. With each iteration one element (red) is

  • C++实现LeetCode(203.移除链表元素)

    [LeetCode] 203.Remove Linked List Elements 移除链表元素 Remove all elements from a linked list of integers that have value val. Example Given: 1 --> 2 --> 6 --> 3 --> 4 --> 5 --> 6, val = 6 Return: 1 --> 2 --> 3 --> 4 --> 5 Credits

  • 五个经典链表OJ题带你进阶C++链表篇

    目录 反转单链表 返回链表中间节点的位置 合并两个有序链表 判断链表中是否有环 判断环形链表进入的节点 反转单链表 题目1:给你单链表的头节点 head ,请你反转链表,并返回反转后的链表. 示例 1: 输入:head = [1,2,3,4,5] 输出:[5,4,3,2,1] 题目来源:力扣 思路一: 翻转指针方向,首先我们要有三个指针,这个就不展示代码了,逻辑过程如下: 思路二:头插方法,我们把每个节点拿下来进行头插实现!代码实现如下: struct ListNode* reverseList

  • C语言实题讲解快速掌握单链表下

    目录 1.移除链表元素 2.反转链表 3.链表的中间节点 4.链表中倒数第k个节点 5.合并两个有序链表 6.链表分割 1.移除链表元素 链接直达: 移除链表元素 题目: 思路: 此题要综合考虑多种情况,常规情况就如同示例1,有多个节点,并且val不连续,但是非常规呢?当val连续呢?当头部就是val呢?所以要分类讨论 常规情况: 需要定义两个指针prev和cur,cur指向第一个数据,prev指向cur的前一个.依次遍历cur指向的数据是否为val,若是,则把prev的下一个节点指向cur的下

  • C语言实题讲解快速掌握单链表上

    目录 1.移除链表元素 2.反转链表 3.链表的中间节点 4.链表中倒数第k个节点 5.合并两个有序链表 6.链表分割 1.移除链表元素 链接直达: 移除链表元素 题目: 思路: 此题要综合考虑多种情况,常规情况就如同示例1,有多个节点,并且val不连续,但是非常规呢?当val连续呢?当头部就是val呢?所以要分类讨论 常规情况: 需要定义两个指针prev和cur,cur指向第一个数据,prev指向cur的前一个.依次遍历cur指向的数据是否为val,若是,则把prev的下一个节点指向cur的下

  • python反转单链表算法题

    现在算法是大厂面试的必考题,而且越来越难,已经不是简单的列表,字符串操作了,会涉及到各种数据结结构.单链表的反转也是经常考的一道题,里面故在此记录一下. 1.链表的特点: 顺序存储元素,但是元素在空间上是不连续的,所以在链表每个元素中除了存储元素的值,还会存储下一个元素的地址,单链表的话就只有指向下一个元素的指针,双向链表的话还会有指向前一个元素的指针.正是由于链表以上的存储特点,在做插入和删除操作时只需要断开指针的连接,不需要移动后面的数据,所以对链表修改的效率会很高,但是查找的效率会很低,这

  • Java数据结构之链表实现(单向、双向链表及链表反转)

    前言 之前学习的顺序表查询非常快,时间复杂度为O(1),但是增删改效率非常低,因为每一次增删改都会元素的移动.可以使用另一种存储方式-链式存储结构. 链表是一种物理存储单元上非连续.非顺序的存储结构.链表由一序列的结点(链表中的每一个元素成为结点)组成. 结点API设计: 类名 Node 构造方法 Node(T t,Node next) 创建Node对象 成员变量 T item:存储数据 Node next :指向下一个结点 结点类: public class Node<T>{ Node ne

  • Java几个实例带你进阶升华上篇

    目录 前言 一.案例1:水仙花 二.案例2:猜数字 三.案例3:减肥计划 四.案例4:不死神兔 五.案例5:评委打分 总结 前言 本期Java基础案例: 水仙花.猜数字.减肥计划.不死神兔.评委打分 以下是本篇文章正文内容,仅供参考 一.案例1:水仙花 题目: 在控制台输出所有的水仙花数 什么是水仙花数? 水仙花是一个三位数: 水仙花数的个位.十位.百位的数字立方和等于原数. 分析: 使用循环遍历所有的三位数(100开始到999结束): 计算之前获取三位数中的每个位上的值: 将三位数中的每个数值

  • Java几个实例带你进阶升华下篇

    目录 前言 一.案例1:两只老虎 二.案例2:三个和尚 三.案例3:考试奖励 总结 前言 以下为本文要记录的大概内容: Java基础案例: 两只老虎.三个和尚.考试奖励 以下是本篇文章正文内容,仅供参考 一.案例1:两只老虎 1.题目: 动物园里有两只老虎,已知两只老虎的体重分别为180kg.200kg,请用程序实现判断两只老虎的体重是否相同. 2.分析: 定义两个变量用于保存老虎的体重(单位为kg,这里只体现数值即可) 用三元运算符实现老虎体重的判断,体重相同,返回true,否则返回false

  • 关于JAVA经典算法40题(超实用版)

    [程序1]题目:古典问题:有一对兔子,从出生后第3个月起每个月都生一对兔子,小兔子长到第四个月后每个月又生一对兔子,假如兔子都不死,问每个月的兔子总数为多少?1.程序分析: 兔子的规律为数列1,1,2,3,5,8,13,21....public class exp2{ public static void main(String args[]){ int i=0; for(i=1;i<=20;i++)System.out.println(f(i));}public static int f(in

  • JavaScript中数据结构与算法(五):经典KMP算法

    KMP算法和BM算法 KMP是前缀匹配和BM后缀匹配的经典算法,看得出来前缀匹配和后缀匹配的区别就仅仅在于比较的顺序不同 前缀匹配是指:模式串和母串的比较从左到右,模式串的移动也是从 左到右 后缀匹配是指:模式串和母串的的比较从右到左,模式串的移动从左到右. 通过上一章显而易见BF算法也是属于前缀的算法,不过就非常霸蛮的逐个匹配的效率自然不用提了O(mn),网上蛋疼的KMP是讲解很多,基本都是走的高大上路线看的你也是一头雾水,我试图用自己的理解用最接地气的方式描述 KMP KMP也是一种优化版的

  • 探讨:将两个链表非降序合并为一个链表并依然有序的实现方法

    已知两个链表list1和list,2,各自非降序排列,将它们合并成另外一个链表list3,并且依然有序,要求保留所有节点.实现过程中,list1中的节点和list2中的节点都转移到了list3中,注意泛型的友元函数的用法.程序如有不足之处,还望指正!!!定义List类 复制代码 代码如下: #include "stdafx.h"#include <iostream> using namespace std;template<class T>struct Node

随机推荐