代码随想录训练营day3 | 链表1 | LeetCode 203.移除链表元素 、707.设计链表 、206.反转链表

本文介绍了三种链表操作的算法实现:如何移除链表中的特定元素,如何反转链表,以及如何设计一个支持多种操作的链表数据结构。删除元素可以使用原地修改的方法,但要注意处理头节点的情况;反转链表则展示了双指针法和递归法;设计链表包括了插入、删除等基本操作,使用虚拟头结点能简化处理。
摘要由CSDN通过智能技术生成

203.移除链表元素

题目链接:力扣-203.移除链表元素

思路:

这里就涉及两种链表操作的方式:

  • 直接使用原来的链表来进行删除操作。
  • 设置一个虚拟头结点在进行删除操作。

第二种方式做这到题更简单一点,第一种需要注意 head.val == val 的特殊情况,不能与其他情况放在一个循环里判断,会导致访问了已经释放的内存的编译错误。而使用虚拟头结点不需要单独判断这种情况。

第一种代码:

struct ListNode* removeElements(struct ListNode* head, int val){
    struct ListNode *s;
    while(head && head->val == val){
        s = head;
        head = head->next;
        free(s);
    }
    struct ListNode *p = head;         
    while(p && p->next){  
        if(p->next->val == val){
                s = p->next;
                p->next = s->next;
                free(s);
            }
        else{
                p = p->next;
            }
    }
    return head;
}

第二种代码(虚拟头):

struct ListNode* removeElements(struct ListNode* head, int val){
    typedef struct ListNode ListNode;
    ListNode *shead = (ListNode*)malloc(sizeof(ListNode));
    shead->next =head;
    ListNode *p = shead;
    ListNode *q;
    while(p && (q = p->next)){
        if(q->val == val){
            p->next = q->next;
            free(q);
        }
        else{
            p = p->next;
        }
    } 
    return shead->next;
}

206.反转链表 

题目链接:力扣-206.反转链表

思路:

  • 双指针法
  • 递归法(从前往后翻转指针指向、从后往前翻转指针指向)

定义两个指针,一个指向头节点,一个指向NULL,通过一个while循环进行整个链表的反转。递归法相对抽象一些,但和双指针法是一样的逻辑。

双指针法:

struct ListNode* reverseList(struct ListNode* head){
        struct ListNode*p = head;
        struct ListNode*q = NULL;
        struct ListNode*s;
        while(p){
            s = p;
            p = p->next;
            s->next = q;
            q = s;
        }
    return s;
}

递归法:

struct ListNode* reverse(struct ListNode* p, struct ListNode* q){
    if(!q){
        return p;
    }
    struct ListNode*s = q->next;
    q->next = p;
    return reverse(q, s);
}

struct ListNode* reverseList(struct ListNode* head){
    return reverse(NULL, head);
}

707.设计链表

题目链接:力扣-707.设计链表

思路:

这是一道考察链表综合操作的题目,比较容易出错,使用虚拟头结点会更方便。

这道题目设计链表的五个接口:

  1. 获取链表第index个节点的数值
  2. 在链表的最前面插入一个节点
  3. 在链表的最后面插入一个节点
  4. 在链表第index个节点前面插入一个节点
  5. 删除链表的第index个节点

链表的知识点有些遗忘了,需要尽快复习,二刷时注意。 

typedef struct MyLinkedList{
    int val;
    struct MyLinkedList *next;
} MyLinkedList;


MyLinkedList* myLinkedListCreate() {
    MyLinkedList* head = (MyLinkedList *)malloc(sizeof (MyLinkedList));
    head->next = NULL;
    return head;
}

int myLinkedListGet(MyLinkedList* obj, int index) {
    MyLinkedList *cur = obj->next;
    for (int i = 0; cur != NULL; i++){
        if (i == index){
            return cur->val;
        }
        else{
            cur = cur->next;
        }
    }
    return -1;
}

void myLinkedListAddAtHead(MyLinkedList* obj, int val) {
    MyLinkedList *nhead = (MyLinkedList *)malloc(sizeof (MyLinkedList));
    nhead->val = val;
    nhead->next = obj->next;
    obj->next = nhead;

}

void myLinkedListAddAtTail(MyLinkedList* obj, int val) {
    MyLinkedList *cur = obj;
    while(cur->next != NULL){
        cur = cur->next;
    }
    MyLinkedList *ntail = (MyLinkedList *)malloc(sizeof (MyLinkedList));
    ntail->val = val;
    ntail->next = NULL;
    cur->next = ntail;
}

void myLinkedListAddAtIndex(MyLinkedList* obj, int index, int val) {
    if (index == 0){
        myLinkedListAddAtHead(obj, val);
        return;
    }
    MyLinkedList *cur = obj->next;
    for (int i = 1 ;cur != NULL; i++){
        if (i == index){
            MyLinkedList* newnode = (MyLinkedList *)malloc(sizeof (MyLinkedList));
            newnode->val = val;
            newnode->next = cur->next;
            cur->next = newnode;
            return;
        }
        else{
            cur = cur->next;
        }
    }
}

void myLinkedListDeleteAtIndex(MyLinkedList* obj, int index) {
    if (index == 0){
        MyLinkedList *tmp = obj->next;
        if (tmp != NULL){
            obj->next = tmp->next;
            free(tmp);     
        }
        return;
    }
    MyLinkedList *cur = obj->next;
    for (int i = 1 ;cur != NULL && cur->next != NULL; i++){
        if (i == index){
            MyLinkedList *tmp = cur->next;
            if (tmp != NULL) {
                cur->next = tmp->next;
                free(tmp);
            }
            return;
        }
        else{
            cur = cur->next;
        }
    }
    
}

void myLinkedListFree(MyLinkedList* obj) {
    while(obj != NULL){
        MyLinkedList *tmp = obj;
        obj = obj->next;
        free(tmp);
    }
}

总结: 

今天的题目都不算难,链表不太熟练,很多的知识点都生疏了。

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值