C++链表相关内容温习回顾——移除链表元素

   本文主要对之前学过的C++链表相关内容进行温习回顾,并以 移除链表元素 为例,进行应用。

   关于链表的基础理论可见:链表理论基础

   应用示例:LeetCode 203 移除链表元素

   https://leetcode.cn/problems/remove-linked-list-elements/

在这里插入图片描述

   0、例题描述如下:

在这里插入图片描述

   1、带详细注释的可在本地IDE运行的C++链表实现如下:

#include<iostream>
#include<vector>

using namespace std;

// 定义链表的节点结构
struct ListNode
{
	int val;  // 节点存储的值
	ListNode* next;  // 指向下一个节点的指针
	ListNode() :val(0), next(nullptr) {}  // 默认构造函数
	ListNode(int x) :val(x), next(nullptr) {}  // 构造函数初始化值
	ListNode(int x, ListNode* ptr) :val(x), next(ptr) {}  // 构造函数初始化值和指针
};

// 定义解决问题的类
class Solution
{
public:
	// 移除链表中所有值为val的节点
	ListNode* removeElements(ListNode* head, int val)
	{  
		ListNode* vhead = new ListNode(0);  // 创建虚拟头节点以简化边界处理
		vhead->next = head;  // 将虚拟头节点指向实际的头节点
		ListNode* cur = vhead;  // 使用cur遍历链表
		while (cur->next != nullptr)  // 遍历链表直到尾部
		{
			if (cur->next->val == val)  // 如果当前节点的下一个节点的值等于val
			{
				ListNode* temp = cur->next;  // 临时保存需要删除的节点
				cur->next = cur->next->next;  // 将当前节点的指针指向下下个节点
				delete temp;  // 删除节点释放内存
			}
			else
			{
				cur = cur->next;  // 移动到下一个节点
			}
		}

		head = vhead->next;  // 更新头节点
		delete vhead;  // 删除虚拟头节点
		return head;  // 返回新的头节点
	}
};

int main()
{
	// 创建测试链表 [1,2,6,3,4,5,6]
	ListNode* head = new ListNode(1);
	head->next = new ListNode(2);
	head->next->next = new ListNode(6);
	head->next->next->next = new ListNode(3);
	head->next->next->next->next = new ListNode(4);
	head->next->next->next->next->next = new ListNode(5);
	head->next->next->next->next->next->next = new ListNode(6);

	Solution solution;
	int val = 6;
	head = solution.removeElements(head, val);  // 调用removeElements函数移除值为6的节点

	// 打印新的链表
	ListNode* cur = head;
	while (cur != nullptr)  // 遍历链表打印每个节点的值
	{
		cout << cur->val << " ";
		cur = cur->next;
	}
	cout << endl;

	// 释放链表内存
	cur = head;
	ListNode* next;
	while (cur != nullptr)
	{
		next = cur->next;  // 保存下一个节点
		delete cur;  // 删除当前节点
		cur = next;  // 移动到下一个节点
	}
}

   运行结果如下:

在这里插入图片描述

   2、程序的详细介绍

   这个程序实现了一个特定的链表操作,即删除链表中所有值等于给定整数 val 的节点。程序分为几个主要部分:

   1. ListNode 结构体:定义链表的节点,每个节点包含一个整数值 val 和一个指向下一个节点的指针 next

   2. Solution 类

   - removeElements 方法:这是类的主要方法,用于移除链表中所有值为 val 的节点。方法通过添加一个虚拟头节点来简化边界条件的处理,然后遍历链表,逐个检查每个节点的值。

   3. main 函数

   - 创建并初始化链表。

   - 调用 removeElements 方法处理链表。

   - 打印处理后的链表结果。

   - 释放链表中所有节点的内存,以防止内存泄漏。

   4. 函数语句的详情介绍见上面代码的注释

   这个程序特别适用于学习和理解链表的操作,特别是如何处理删除链表节点时的边界条件。通过使用虚拟头节点,代码更加简洁,逻辑更清晰。

   3、LeetCode 核心代码模式源码:

/**
 * Definition for singly-linked list.
 * struct ListNode {
 *     int val;
 *     ListNode *next;
 *     ListNode() : val(0), next(nullptr) {}
 *     ListNode(int x) : val(x), next(nullptr) {}
 *     ListNode(int x, ListNode *next) : val(x), next(next) {}
 * };
 */
class Solution
{
public:
	ListNode* removeElements(ListNode* head, int val)
	{  
		ListNode* vhead = new ListNode(0);
		vhead->next = head;
		ListNode* cur = vhead;
		while (cur->next != nullptr)
		{
			if (cur->next->val == val)
			{
				ListNode* temp = cur->next;
				cur->next = cur->next->next;
				delete temp;
			}
			else
			{
				cur = cur->next;
			}
		}

		head = vhead->next;
		delete vhead;
		return head;
	
	}
};

   4、一种可能的ACM模式源码:

   由于未知具体的输入给定形式,本例子中以如下所示的输入格式为例

[1,2,6,3,4,5,6] 6

   示例程序如下:

#include<iostream>
#include<vector>

using namespace std;

struct ListNode {
    int val;
    ListNode* next;
    ListNode() : val(0), next(nullptr) {}
    ListNode(int x) : val(x), next(nullptr) {}
    ListNode(int x, ListNode* ptr) : val(x), next(ptr) {}
};

class Solution {
public:
    ListNode* removeElements(ListNode* head, int val) {
        ListNode* vhead = new ListNode(0);  // 创建虚拟头节点以简化边界处理
        vhead->next = head;
        ListNode* cur = vhead;
        while (cur->next != nullptr) {
            if (cur->next->val == val) {
                ListNode* temp = cur->next;
                cur->next = cur->next->next;
                delete temp;
            } else {
                cur = cur->next;
            }
        }
        head = vhead->next;
        delete vhead;
        return head;
    }
};

int main() {
    vector<int> nums;
    int num, val;
    char ch;
    
    // 读取链表数值,以及需要移除的数值
    while (cin >> ch && ch != ']') { // 读取直到遇到 ']'
        if (cin >> num) {
            nums.push_back(num);
            cin >> ch; // 读取逗号或者结束的 ']'
        }
    }

    cin >> val; // 读取需要移除的数值

    // 构建链表
    ListNode* head = nullptr, * tail = nullptr;
    for (int number : nums) {
        ListNode* newNode = new ListNode(number);
        if (!head) {
            head = tail = newNode;
        } else {
            tail->next = newNode;
            tail = newNode;
        }
    }

    Solution solution;
    head = solution.removeElements(head, val);

    // 输出新链表
    ListNode* cur = head;
    if (cur) {
        cout << cur->val;
        cur = cur->next;
    }
    while (cur) {
        cout << "," << cur->val;
        cur = cur->next;
    }
    cout << endl;

    // 释放链表内存
    cur = head;
    ListNode* next;
    while (cur) {
        next = cur->next;
        delete cur;
        cur = next;
    }
}

   参考输入的运行结果如下:

1,2,3,4,5

在这里插入图片描述

  • 4
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

慕羽★

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值