This blog starts from scratch to demonstrate how to implement a linked list data structure using structs, covering the basic operations including insertion, deletion, traversal, and searching.
structLinkedList { // By default, construct a sentinel node that does not store any value, and its reference points to the first element of the list // Initially, it points to nullptr ListNode* sentinel; int size = 0; //size represents the number of nodes in the Linkedlist
// Constructor: Initialize the sentinel node, allocate memory on the heap for it LinkedList(): sentinel(newListNode()), size(0) { cout << "Created successfully" << endl; };
// Destructor: Iteratively delete all nodes (including the memory pointed to by the sentinel node initially) // Once the sentinel points to a null pointer, the destruction process is complete ~LinkedList() { while (sentinel != nullptr) { ListNode* temp = sentinel; sentinel = sentinel->next; delete temp; } cout << "Destroyed successfully" << endl; }
voidinsertAfter(ListNode* current, int value){ // If the current node is null, return or throw an exception if (current == nullptr) { throw std::invalid_argument("Current node is null"); return; }
// Create a new node ListNode* newNode = new ListNode{value, current->next}; // Insert the new node after the current node current->next = newNode; size++; }
Modify①插入到第一个节点
这个问题非常好解决,相当于更换了链表的头结点,只需要更改插入节点的引用即可
1 2 3 4 5 6 7
voidinsertAtHead(int value){ // Set the reference of the new node to the reference of the sentinel node ListNode* newNode = new ListNode{value, sentinel->next}; // Change the reference of the sentinel node because the first element of the list has changed sentinel->next = newNode; size++; }
voidinsertAtPosition(int value, int position){ // If the position is 0 or the list is empty, insert at the head if (position == 0 || sentinel->next == nullptr) { insertAtHead(value); return; }
ListNode* current = sentinel->next; int currentPosition = 0; // Set a pointer to start the traversal
// Find the node before the insertion position while (current->next != nullptr && currentPosition < position - 1) { current = current->next; currentPosition++; }
// If the position exceeds the end of the list, insert at the end if (currentPosition < position - 1) { ListNode* newNode = new ListNode{value, nullptr}; current->next = newNode; size++; } else { // Insert the new node // current represents the node at index n-1, inserting after it achieves the desired index insertAfter(current, value); } }
// Insert directly at the end voidinsertAtEnd(int value){ if (sentinel->next == nullptr) { ListNode* newNode = new ListNode{value, nullptr}; sentinel->next = newNode; size++; return; } ListNode* current = sentinel->next; while (current->next != nullptr) { current = current->next; } // Traverse to the last node // Create a new node ListNode* newNode = new ListNode{value, nullptr}; current->next = newNode; size++; }
voidinsertAtPosition(int value, int position){ // 如果位置是0或链表为空,则在头部插入 if (position == 0 || head == nullptr) { insertAtHead(value); return; }
Node* current = head; int currentPosition = 0; //设置一个指针,开始执行遍历
// 找到插入位置的前一个节点 while (current->next != nullptr && currentPosition < position - 1) { current = current->next; currentPosition++; }
// 如果位置超出了链表的末尾,则在末尾插入 if (currentPosition < position - 1) { Node* newNode = new Node{value, nullptr}; current->next = newNode; } else { // 插入新节点 //current代表的是n-1索引上的节点,插入到他的后面就是需要达到的目标索引 insertAfter(current, value); } }
单链表的删除
原理和单链表的插入大同小异:
将delPtr前面的一个节点p的引用更改,使其直接指向后面的一个节点即可
但是在删除时,要额外关注头和尾的特殊情况
如果是第一个元素,则需要更改头结点的对应的值
如果是尾部元素,直接将上一个元素的应用设置为空指针即可
一定要注意delete!!!
我们先来看一下比较基础的功能实现:删除末尾的元素,思路如下:
首先遍历数组走到最后一个元素
由于链表非连续的物理结构的特点,其遍历的过程的复杂度会高于数组,达到了O(n)的时间复杂度。
修改倒数第二个元素的引用为空指针
delete掉最后一个元素的内存
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16
// Delete the last element of the list voiddeletetheend(){ if (sentinel->next == nullptr) return; // Empty list
ListNode* current = sentinel->next; ListNode* prev = sentinel;
while (current->next != nullptr) { prev = current; current = current->next; } // At this point, current is the last element of the list prev->next = nullptr; delete current; size--; }
voiddeletetheplace(int index){ if (sentinel->next == nullptr) return; // Empty list
ListNode* current = sentinel->next; ListNode* prev = sentinel; int count = 0;
while (current != nullptr && count < index) { prev = current; current = current->next; count++; } if (count < index) { // Index is too large, no corresponding index return; } else { // Then current is the node at the target index prev->next = current->next; delete current; size--; } }
其他操作
遍历,插入和删除是链表基本操作中最基本的三个操作,现在我们可以轻而易举地实现其他对链表的操作了。
索引
返回特定索引值的val值
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
intat(int index){ ListNode* current = sentinel->next; // At this point, current points to the address of the first element of the list int count = 0; if (count >= size) { throwout_of_range("Out of range"); } else { while (count < index) { current = current->next; count++; } // At this point, current points to the element at the target index return current->val; } }
修改
修改特定索引值的val值
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
voidmodify(int index,int replacement){ ListNode* current = sentinel->next; // At this point, current points to the address of the first element of the list int count = 0; if (count >= size) { throwout_of_range("Out of range"); } else { while (count < index) { current = current->next; count++; } // At this point, current points to the element at the target index current->val=replacement; } }
intsearch(int target){ int count = 0; ListNode* current = sentinel->next; // At this point, current points to the address of the first element of the list while (current != nullptr) { if (current->val == target) { return count; } count++; current = current->next; } return-1; }
打印列表(可视化)
1 2 3 4 5 6 7 8
voidprintList(){ ListNode* current = sentinel->next; while (current != nullptr) { std::cout << current->val << " -> "; current = current->next; } std::cout << "NULL" << std::endl; }
置空检查
判断链表是否有元素
1 2 3 4 5 6 7 8
// Check if the list is empty boolisempty(){ if (sentinel->next == nullptr) { returntrue; } else { returnfalse; } }
/* * @Author: Xiyuan Yang xiyuan_yang@outlook.com * @Date: 2024-12-08 10:28:07 * @LastEditors: Xiyuan Yang xiyuan_yang@outlook.com * @LastEditTime: 2024-12-08 13:48:56 * @FilePath: \CODE_for_Vscode\C++_project\Linked_list_copy.cpp * @Description: * Do you code and make progress today? * Copyright (c) 2024 by Xiyuan Yang, All Rights Reserved. */ #include<iostream> usingnamespace std;
structListNode { int val; ListNode *next; ListNode(int value = 0, ListNode *next = nullptr) : val(value), next(next) {} };
structLinkedList { // By default, construct a sentinel node that does not store any value, and its reference points to the first element of the list // Initially, it points to nullptr ListNode* sentinel; int size = 0;
// Constructor: Initialize the sentinel node, allocate memory on the heap for it LinkedList(): sentinel(newListNode()), size(0) { cout << "Created successfully" << endl; };
// Destructor: Iteratively delete all nodes (including the memory pointed to by the sentinel node initially) // Once the sentinel points to a null pointer, the destruction process is complete ~LinkedList() { while (sentinel != nullptr) { ListNode* temp = sentinel; sentinel = sentinel->next; delete temp; } cout << "Destroyed successfully" << endl; }
// Four insertion functions // Insert a node at the head of the list voidinsertAtHead(int value){ // Set the reference of the new node to the reference of the sentinel node ListNode* newNode = new ListNode{value, sentinel->next}; // Change the reference of the sentinel node because the first element of the list has changed sentinel->next = newNode; size++; }
// Insert a new node after the specified node // current represents the node after which you want to insert voidinsertAfter(ListNode* current, int value){ // If the current node is null, return or throw an exception if (current == nullptr) { throw std::invalid_argument("Current node is null"); return; }
// Create a new node ListNode* newNode = new ListNode{value, current->next}; // Insert the new node after the current node current->next = newNode; size++; }
voidinsertAtPosition(int value, int position){ // If the position is 0 or the list is empty, insert at the head if (position == 0 || sentinel->next == nullptr) { insertAtHead(value); return; }
ListNode* current = sentinel->next; int currentPosition = 0; // Set a pointer to start the traversal
// Find the node before the insertion position while (current->next != nullptr && currentPosition < position - 1) { current = current->next; currentPosition++; }
// If the position exceeds the end of the list, insert at the end if (currentPosition < position - 1) { ListNode* newNode = new ListNode{value, nullptr}; current->next = newNode; size++; } else { // Insert the new node // current represents the node at index n-1, inserting after it achieves the desired index insertAfter(current, value); } }
// Insert directly at the end voidinsertAtEnd(int value){ if (sentinel->next == nullptr) { ListNode* newNode = new ListNode{value, nullptr}; sentinel->next = newNode; size++; return; } ListNode* current = sentinel->next; while (current->next != nullptr) { current = current->next; } // Traverse to the last node // Create a new node ListNode* newNode = new ListNode{value, nullptr}; current->next = newNode; size++; }
// Delete operation // Delete the node with the specified value voiddeleteNode(int key){ if (sentinel->next == nullptr) return; // Empty list
ListNode* current = sentinel->next; ListNode* prev = sentinel;
while (current != nullptr) { if (current->val == key) { break; } prev = current; current = current->next; }
if (current == nullptr) return; // Node does not exist
// Delete the last element of the list voiddeletetheend(){ if (sentinel->next == nullptr) return; // Empty list
ListNode* current = sentinel->next; ListNode* prev = sentinel;
while (current->next != nullptr) { prev = current; current = current->next; } // At this point, current is the last element of the list prev->next = nullptr; delete current; size--; }
// Delete the element at a specific index in the list voiddeletetheplace(int index){ if (sentinel->next == nullptr) return; // Empty list
ListNode* current = sentinel->next; ListNode* prev = sentinel; int count = 0;
while (current != nullptr && count < index) { prev = current; current = current->next; count++; } if (count < index) { // Index is too large, no corresponding index return; } else { // Then current is the node at the target index prev->next = current->next; delete current; size--; } } voidmodify(int index,int replacement){ ListNode* current = sentinel->next; // At this point, current points to the address of the first element of the list int count = 0; if (count >= size) { throwout_of_range("Out of range"); } else { while (count < index) { current = current->next; count++; } // At this point, current points to the element at the target index current->val=replacement; } } // List search intat(int index){ ListNode* current = sentinel->next; // At this point, current points to the address of the first element of the list int count = 0; if (count >= size) { throwout_of_range("Out of range"); } else { while (count < index) { current = current->next; count++; } // At this point, current points to the element at the target index return current->val; } }
intsearch(int target){ int count = 0; ListNode* current = sentinel->next; // At this point, current points to the address of the first element of the list while (current != nullptr) { if (current->val == target) { return count; } count++; current = current->next; } return-1; }
// Print list content voidprintList(){ ListNode* current = sentinel->next; while (current != nullptr) { std::cout << current->val << " -> "; current = current->next; } std::cout << "NULL" << std::endl; }
// Check if the list is empty boolisempty(){ if (sentinel->next == nullptr) { returntrue; } else { returnfalse; } } };