今日任务
- 链表理论基础
- 203.移除链表元素
- 707.设计链表
- 206.反转链表
链表理论基础
建议:了解一下链表基础,以及链表和数组的区别
文章链接:代码随想录
struct ListNode {
int val; // 节点上存储的元素
ListNode *next; // 指向下一个节点的指针
ListNode(int x) : val(x), next(NULL) {} // 节点的构造函数
};
链表的定义还是比较简单的,定义一个结构体,一个val值,next指针,再加上一个构造函数。如果不定义构造函数使用默认构造函数的话,在初始化的时候就不能直接给变量赋值!也比较好理解,相当于重新写个构造函数。
学习了点.,at,this的区别
在C++中,->、this、和.都是用于访问对象的成员的运算符或方法。
-
->
(箭头运算符):- 用于指针指向的对象来访问其成员。
- 当有一个指向对象的指针时,可以使用
->
来访问该对象的成员。 - 例如,如果有一个指向
myclass
类型的指针ptr
,可以使用ptr->myvar
来访问myvar
成员变量。
-
this
(指向当前对象的指针):- 是一个隐式的参数,存在于非静态成员函数中,指向调用该成员函数的对象。
- 它通常用于需要引用当前对象的情况,比如在成员函数内部传递当前对象的引用或指针给其他函数。
- 例如,
this->myvar
在成员函数中访问myvar
成员变量,与直接使用myvar
是等效的。
-
.
(点运算符):- 用于非指针的对象来访问其成员。
- 当对象已经被实例化,不是通过指针访问时,可以使用
.
来访问该对象的成员。 - 例如,如果有
myclass
类型的对象obj
,可以使用obj.myvar
来访问myvar
成员变量。
-
总结来说,
this
是一个指针,通常在成员函数内部使用,而->
和.
是访问成员的运算符,其中->
用于指针,.
用于非指针对象。
203.移除链表元素
建议: 本题最关键是要理解 虚拟头结点的使用技巧,这个对链表题目很重要。
题目链接/文章讲解/视频讲解::代码随想录
/**
* 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) {
while(head!=nullptr&&head->val==val)
{
ListNode* tmp = head;
head = head->next;
delete tmp;
}
ListNode* tmp1 = head;
while(tmp1!=nullptr&& tmp1->next!= NULL)
{
if(tmp1->next->val==val)
{
ListNode* tmp2 = tmp1->next;
tmp1->next = tmp2->next;
delete tmp2;
}
else {
tmp1 = tmp1->next;
}
}
return head;
}
};
还是要注意开始删除头节点的时候要用while循环而不是if,我一开始也被绕进去了,仔细想想就知道保证第一个结点给删了,万一第二个也要删怎么办,就会漏掉呢。
第一种方法,就是对头节点进行特殊处理。还是比较简单的,就是发现节点值要删掉,就把前面节点指向后一个结点,然后释放当前结点。我这里用的是tmp1,临时结点,发现这种命名方式还是不太好,超过,两个temp临时变量,就还是用具体含义命名,否则确实容易绕脑子。
/**
* 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* dummyHead = new ListNode(0);
dummyHead->next=head;
ListNode* tmp1 = dummyHead;
while(tmp1!=nullptr&& tmp1->next!= NULL)
{
if(tmp1->next->val==val)
{
ListNode* tmp2 = tmp1->next;
tmp1->next = tmp2->next;
delete tmp2;
}
else {
tmp1 = tmp1->next;
}
}
head = dummyHead->next;
delete dummyHead;
return head;
}
};
第二种虚拟头节点的方法也挺好理解的。
/**
* 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* tmp1 = head;
while(tmp1!=nullptr&& tmp1->next!= NULL)
{
if(tmp1->next->val==val)
{
ListNode* tmp2 = tmp1->next;
tmp1->next = tmp2->next;
delete tmp2;
}
else {
tmp1 = tmp1->next;
}
}
if(head!=nullptr&&head->val==val)
{
ListNode* tmp2 = head;
head = head->next;
delete tmp2;
}
return head;
}
};
我又想到一个问题,完全可以先把头节点后面的元素进行移除,最后再判断头节点。感觉可行,不知道有没有问题。
707.设计链表
class MyLinkedList {
public:
struct LinkedNode {
int val;
LinkedNode* next;
LinkedNode(int val):val(val), next(nullptr){}
};
MyLinkedList() {
dummpyHead=new LinkedNode(0);
size=0;
}
int get(int index) {
if (index >= size || index < 0)
return -1;
LinkedNode* cur = dummpyHead->next;
// Check if cur is null before accessing its members
if (cur == nullptr)
return -1;
while (index--) {
cur = cur->next;
// Check if cur is null during traversal
if (cur == nullptr)
return -1;
}
return cur->val;
}
void addAtHead(int val) {
LinkedNode* insert=new LinkedNode(val);
insert->next=dummpyHead->next;
dummpyHead->next=insert;
size++;
}
void addAtTail(int val) {
LinkedNode* insert=new LinkedNode(val);
LinkedNode* cur=dummpyHead;
while(cur->next!=nullptr)
cur=cur->next;
cur->next=insert;
size++;
}
void addAtIndex(int index, int val) {
if(index>size||index<0)
return;
LinkedNode* insert=new LinkedNode(val);
LinkedNode* cur=dummpyHead;
while(index--)
cur=cur->next;
insert->next=cur->next;
cur->next=insert;
size++;
}
void deleteAtIndex(int index) {
if(index>=size||index<0)
return ;
LinkedNode* cur=dummpyHead;
while(index--)
{
cur=cur->next;
}
LinkedNode* tmp=cur->next;
cur->next=cur->next->next;
delete tmp;
size--;
}
private :
int size;
LinkedNode* dummpyHead;
};
/**
* Your MyLinkedList object will be instantiated and called as such:
* MyLinkedList* obj = new MyLinkedList();
* int param_1 = obj->get(index);
* obj->addAtHead(val);
* obj->addAtTail(val);
* obj->addAtIndex(index,val);
* obj->deleteAtIndex(index);
*/
建议: 这是一道考察 链表综合操作的题目,不算容易,可以练一练 使用虚拟头结点
题目链接/文章讲解/视频讲解:代码随想录
其实感觉还算是不太难的,注意加上判断语句,判断是否越界,以及考虑空链表的情况。
struct结构体,总是忘掉最后大括号后面加个;另外插入和删除操作,忘了可以插在最后面,确实也可以>size。还有delete以后别忘了size--;一些小的错误,很容易被忽视掉,然后浪费时间去查找。
206.反转链表
建议先看视频讲解,视频讲解中对 反转链表需要注意的点讲的很清晰了,看完之后大家的疑惑基本都解决了。
/**
* 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* reverseList(ListNode* head) {
ListNode* temp; // 保存cur的下一个节点
ListNode* cur = head;
ListNode* pre = NULL;
while(cur!=NULL)
{
temp=cur->next;
cur->next=pre;
pre=cur;
cur=temp;
}
return pre;
}
};
稍微有点绕,但是其本质就是用两只手抓住两端,先修理左手的,修理完左手的。把右手的给左手,抓住一个。继续修理左手的,直到修理完,就彻底完了,就是把当前的结点,然后再往下移动,不过我在想,他不考虑最后加个虚结点吗?
最后可以加上这两句
dummy->next = pre; // 将虚拟节点指向反转后的头节点
return dummy->next; // 返回虚拟节点的下一个节点,即反转后的头节点
这才理解,根本没有虚结点,所以也不要考虑。
/**
* 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* reverse(ListNode* pre,ListNode* cur) {
if(cur==nullptr)return pre;
ListNode* temp = cur->next;
cur->next=pre;
return reverse(cur,temp);
}
ListNode* reverseList(ListNode* head) {
return reverse(nullptr,head);
}
};
所谓的递归方法,递归函数本身定义为reverse(ListNode* pre,ListNode* cur),本质上就是把pre=cur; cur=temp;替换变成 return reverse(cur,temp);稍微绕一点,但并不算难。
题目链接/文章讲解/视频讲解:代码随想录