当前位置: 首页 > article >正文

单链表算法实战:解锁数据结构核心谜题——移除链表元素

题目如下:

在这里插入图片描述

解题过程如下:

给了链表的头结点head就相当于知道了整个链表。

思路1:查找值为val的结点并返回结点位置,删除pos位置的结点。
涉及循环的嵌套,时间复杂度为O(n^2):
在这里插入图片描述
思考时间复杂度可不可以降为O(n)呢?

思路2:创建新链表,将原链表中值不为val的结点拿下来尾插。
在原链表中修改,涉及到遍历、删除,时间复杂度不太好,那就创建一个新链表,这里并没有申请一个新链表、新结点,而是创建了一个空链表,该链表里有两个指针newHeadnewTail,这两个指针还是指向原链表中的结点,通过修改两个新指针的指向来变成一个新的链表。

在这里插入图片描述

/**
 * Definition for singly-linked list.
 * struct ListNode {
 *     int val;
 *     struct ListNode *next;
 * };
 */
typedef struct ListNode ListNode;
struct ListNode* removeElements(struct ListNode* head, int val) {
    //创建空链表
    ListNode* newHead, *newTail;
    newHead = newTail = NULL;

    //查找值不为val的结点并拿下来尾插
    ListNode* pcur = head;
    while (pcur)
    {
        if (pcur->val != val)
        {
            //链表为空
            if (newHead == NULL)
            {
                newHead = newTail = pcur;
            }
            //链表不为空
            else
            {
                newTail->next = pcur;
                newTail = newTail->next;
            }
        }
        pcur = pcur->next;
    }
    return newHead;
}

点击运行发现Case1“解答错误”:
在这里插入图片描述
OJ代码有bug怎么办?VS调试技能用起来:

//test.c
struct ListNode{
     int val;
     struct ListNode *next;
 };

typedef struct ListNode ListNode;
struct ListNode* removeElements(struct ListNode* head, int val) {
	//创建空链表
	ListNode* newHead, * newTail;
	newHead = newTail = NULL;

	//查找值不为val的结点并拿下来尾插
	ListNode* pcur = head;
	while (pcur)
	{
		if (pcur->val != val)
		{
			//链表为空
			if (newHead == NULL)
			{
				newHead = newTail = pcur;
			}
			//链表不为空
			else
			{
				newTail->next = pcur;
				newTail = newTail->next;
			}
		}
		pcur = pcur->next;
	}
	return newHead;
}
int main()
{
	//手动构造一个链表
	SLTNode* node1 = (SLTNode*)malloc(sizeof(SLTNode));
	SLTNode* node2 = (SLTNode*)malloc(sizeof(SLTNode));
	SLTNode* node3 = (SLTNode*)malloc(sizeof(SLTNode));
	SLTNode* node4 = (SLTNode*)malloc(sizeof(SLTNode));
	SLTNode* node5 = (SLTNode*)malloc(sizeof(SLTNode));
	SLTNode* node6 = (SLTNode*)malloc(sizeof(SLTNode));
	SLTNode* node7 = (SLTNode*)malloc(sizeof(SLTNode));


	node1->data = 1;
	node2->data = 2;
	node3->data = 6;
	node4->data = 3;
	node5->data = 4;
	node6->data = 5;
	node7->data = 6;


	node1->next = node2;
	node2->next = node3;
	node3->next = node4;
	node4->next = node5;
	node5->next = node6;
	node6->next = node7;
	node7->next = NULL;

	SLTNode* plist = node1;
	int val = 6;
	ListNode* newHead = removeElements(plist, val);
	return 0;
}

在这里插入图片描述

最后发现问题所在:存储数据5的尾结点的指针域指向存储6的结点,应该把链表的尾结点的next指针置为空。

在这里插入图片描述

我们发现修改后点击运行还是会出现错误,是因为当这个链表是空链表时,不进入循环,newTail是空指针,newTail->next就是对空指针的解引用操作,这不符合语法规则,所以会报错:

在这里插入图片描述

完整代码如下:

/**
 * Definition for singly-linked list.
 * struct ListNode {
 *     int val;
 *     struct ListNode *next;
 * };
 */
typedef struct ListNode ListNode;
struct ListNode* removeElements(struct ListNode* head, int val) {
    //创建空链表
    ListNode* newHead, *newTail;
    newHead = newTail = NULL;

    //查找值不为val的结点并拿下来尾插
    ListNode* pcur = head;
    while (pcur)
    {
        if (pcur->val != val)
        {
            //链表为空
            if (newHead == NULL)
            {
                newHead = newTail = pcur;
            }
            //链表不为空
            else
            {
                newTail->next = pcur;
                newTail = newTail->next;
            }
        }
        pcur = pcur->next;
    }
    if (newTail)
        newTail->next = NULL;
    return newHead;
}

http://www.kler.cn/a/518269.html

相关文章:

  • 安宝特方案 | AR在供应链管理中的应用:提升效率与透明度
  • 每日一题 427. 建立四叉树
  • 第20篇:Python 开发进阶:使用Django进行Web开发详解
  • 参数是模型学会的东西,预训练是让它学习的东西
  • Redis实战(黑马点评)——涉及session、redis存储验证码,双拦截器处理请求
  • 【2024年华为OD机试】(A卷,200分)- 查找树中元素 (JavaScriptJava PythonC/C++)
  • 计算机网络 (54)系统安全:防火墙与入侵检测
  • 论文速读|Matrix-SSL:Matrix Information Theory for Self-Supervised Learning.ICML24
  • 机器学习11-学习路径推荐
  • Solon Cloud Gateway 开发:导引
  • 99.15 金融难点通俗解释:毛利率vs营业利润率vs净利率
  • AI画笔,绘就古今艺术星河(5/10)
  • 【Docker】私有Docker仓库的搭建
  • K8S中Service详解(三)
  • 食堂订餐小程序ssm+论文源码调试讲解
  • pytorch2.5实例教程
  • poi在word中打开本地文件
  • Cloudflare通过代理服务器绕过 CORS 限制:原理、实现场景解析
  • C语言数据结构:链表、栈与队列、排序算法与查找算法深度解析
  • 【C++高并发服务器WebServer】-1:Linux中父子进程fork创建及关系、GDB多进程调试
  • Redis(5,jedis和spring)
  • QModbusTCPClient 服务器断开引起的程序崩溃
  • ChirpIoT技术的优势以及局限性
  • Spring Boot - 数据库集成03 - 集成Mybatis
  • SSM框架探秘:Spring 整合 Mybatis 框架
  • Linux(Centos 7.6)命令详解:wc