代码随想录算法训练营第三天 | 链表理论基础、203.移除链表元素、707.设计链表、206.反转链表

链表理论基础

文章链接:https://programmercarl.com/%E9%93%BE%E8%A1%A8%E7%90%86%E8%AE%BA%E5%9F%BA%E7%A1%80.html

注意一下链表在java中的代码实现,学校居然不讲这个。


203.移除链表元素

建议: 本题最关键是要理解 虚拟头结点的使用技巧,这个对链表题目很重要。
题目链接/文章讲解/视频讲解::https://programmercarl.com/0203.%E7%A7%BB%E9%99%A4%E9%93%BE%E8%A1%A8%E5%85%83%E7%B4%A0.html

虚拟头节点:为了避免单独处理head被删除的情况

/**
 * Definition for singly-linked list.
 * public class ListNode {
 *     int val;
 *     ListNode next;
 *     ListNode() {}
 *     ListNode(int val) { this.val = val; }
 *     ListNode(int val, ListNode next) { this.val = val; this.next = next; }
 * }
 */
class Solution {
    public ListNode removeElements(ListNode head, int val) {
        // If the head is null, return null
        if (head == null) {
            return null;
        }
        // Create a dummy node that points to the head
        ListNode dummy = new ListNode(-1, head);
        ListNode pre = dummy;
        ListNode cur = head;
        // Traverse the list
        while (cur != null) {
            // If current node's value equals the target value, remove it
            if (cur.val == val) {
                pre.next = cur.next;
            } else {
                pre = cur;
            }
            // Move to the next node regardless of whether cur was deleted
            cur = cur.next;
        }
        // Return dummy.next instead of head to handle cases where head is deleted
        return dummy.next;
    }
}

  • 自己写的过程中出现的问题
    cur = cur.next; 写入了if条件语句内部,没有考虑到当 cur.val != val 的情况下,cur 并不会移动,要保证向后走。

707.设计链表

题目链接/文章讲解/视频讲解:https://programmercarl.com/0707.%E8%AE%BE%E8%AE%A1%E9%93%BE%E8%A1%A8.html

单链表
// Single linked list node class
class LinkNode {
    int val; // Value of the node
    LinkNode next; // Pointer to the next node

    public LinkNode() {}

    public LinkNode(int val) {
        this.val = val;
    }
}

class MyLinkedList {
    int size; // Length of the linked list
    LinkNode head; // Dummy head node

    // Initialize the linked list
    public MyLinkedList() {
        size = 0;
        head = new LinkNode(0); // Dummy head node with value 0
    }

    // Get the value of the index-th node
    public int get(int index) {
        // Check if the index is valid
        if (index < 0 || index >= size) {
            return -1;
        }

        LinkNode currentNode = head;
        // Traverse to the index-th node
        for (int i = 0; i <= index; i++) {
            currentNode = currentNode.next;
        }
        return currentNode.val;
    }

    // Add a node at the head
    public void addAtHead(int val) {
        addAtIndex(0, val);
    }

    // Add a node at the tail
    public void addAtTail(int val) {
        addAtIndex(size, val);
    }

    // Add a node before the index-th node
    public void addAtIndex(int index, int val) {
        // If index is greater than the length of the list, do not insert the node
        if (index > size) {
            return;
        }
        // If index is less than 0, insert the node at the head
        if (index < 0) {
            index = 0;
        }
        size++;
        LinkNode pre = head; // Create a predecessor node
        // Traverse to the node before the index-th node
        for (int i = 0; i < index; i++) {
            pre = pre.next;
        }
        LinkNode addNode = new LinkNode(val);
        addNode.next = pre.next;
        pre.next = addNode;
    }

    // Delete the index-th node
    public void deleteAtIndex(int index) {
        // Check if the index is valid
        if (index < 0 || index >= size) {
            return;
        }
        size--;
        LinkNode pre = head; // Create a predecessor node
        // Traverse to the node before the index-th node
        for (int i = 0; i < index; i++) {
            pre = pre.next;
        }
        pre.next = pre.next.next;
    }
}
  • 自己写错的地方:
    get() 的if条件判断中
    // i <= index是因为有虚拟头节点 for(int i = 0; i <= index; i++){
    一开始写成了i < index

    deleteAtIndex中if条件判断一开始写的是index > size,忽略到当index=size时也越界了。

双链表
  // 双链表
//   1. 定义链表节点类
class LinkNode{
    int val;
    LinkNode next, prev;
    public LinkNode(){}
    public LinkNode(int val){
        this.val = val;
    }
        
}
// 2. 定义链表类
class MyLinkedList {
    int size; //链表中元素的数量
    LinkNode head, tail; //虚拟头节点和尾节点
  
    public MyLinkedList() {
        this.size = 0;
        this.head = new LinkNode(0); 
        this.tail = new LinkNode(0); 
        head.next = tail;
        tail.prev = head;     
    }
    
    // 实现get方法获取节点值
    public int get(int index) {
        if(index < 0 || index >= size){
            return -1;
        }

        LinkNode currentNode;

        // 判断哪一边历时更短
        if(index >= size/2){
            currentNode = tail;
            for(int i = size; i > index; i--){
                currentNode = currentNode.prev;
            }
        }else{
            currentNode = head;
            for(int i = 0; i <= index; i++){
                currentNode = currentNode.next;
            }
        }
        return currentNode.val;
    }
    
    public void addAtHead(int val) {
        addAtIndex(0, val);
    }
    
    public void addAtTail(int val) {
        addAtIndex(size, val);
        
    }
    
    public void addAtIndex(int index, int val) {
        if(index > size){
            return;
        }
        if(index < 0){
            index = 0;
        }
        size++;
        LinkNode pre = head;
        for(int i = 0; i < index; i++){
            pre = pre.next;
        }
        // 新建结点
        LinkNode addNode = new LinkNode(val);

        addNode.next = pre.next;
        pre.next.prev = addNode;
        addNode.prev = pre;
        pre.next = addNode;
        
    }
    
    public void deleteAtIndex(int index) {
        if(index < 0 || index >= size){
            return;
        }
        size--;
        LinkNode pre = head;
        for(int i = 0; i < index; i++){
            pre = pre.next;
        }
        pre.next.next.prev = pre;
        pre.next = pre.next.next;
        
    }
}

/**
 * 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);
 */

问题:
依然是get方法中的if条件判断,一开始没有意识到用双链表的话,两个区间应该能接上才对

 if(index >= size/2){
            currentNode = tail;
            for(int i = size; i > index; i--){
                currentNode = currentNode.prev;
            }
        }else{
            currentNode = head;
            for(int i = 0; i <= index; i++){
                currentNode = currentNode.next;
            }
        }

看了视频的记录:

  • 获取第n个节点中对n进行赋值,还有对while中怎么操作,想清楚极端情况即可:如果链表里返回第0个节点的结果,看有没有空指针异常或者指向错误。
  • 头部插入节点的坑:正确的顺序应该是先连接后面的边,让插入的节点指向 pre.next;然后再添加前面的边pre.next = addNode;
  • 尾部插入节点:current = dummy head,想象极端情况,应该让current指向尾部,current.next=null停止while循环
  • 第n个节点前插入节点:要知道index的前节点才能插入。所以current指向index的前一个节点。
while(n--){
//想象极端情况,比如第0个节点,所以第0个节点应该是current.next
//所以current.next 是第n个节点
}

  • 删除第n个节点:current=dummy head。要删第n个节点,还是要知道前一个节点的指针。所以第n个节点一定是current.next。

  • 总结:

  • 虚拟头结点。

  • current.next 才是第n个点。

  • 先更新后面的边,再更新前面。

感觉视频更清晰些,可以跟着视频的思路自己再实现一遍。


206.反转链表

题目链接/文章讲解/视频讲解:https://programmercarl.com/0206.%E7%BF%BB%E8%BD%AC%E9%93%BE%E8%A1%A8.html

/**
 * Definition for singly-linked list.
 * public class ListNode {
 *     int val;
 *     ListNode next;
 *     ListNode() {}
 *     ListNode(int val) { this.val = val; }
 *     ListNode(int val, ListNode next) { this.val = val; this.next = next; }
 * }
 */

class Solution {
    public ListNode reverseList(ListNode head) {
        ListNode pre = null;
        ListNode cur = head;
        ListNode temp = null;
        while(cur != null){
            temp = cur.next; // Save the next node
            cur.next = pre; // Current node points to the previous node
            pre = cur; // Move the previous pointer
            cur = temp; // Move the current pointer
        }
        return pre;
    }
}

只实现了双指针法,递归等二刷再做。
看了视频,注意几个边界条件:
先移动pre,后移动current。
最后current指向null,pre就指向新列表的头结点。所以返回pre。


以后应该直接上手写一遍,不能反复看题解,否则一直看了忘忘了看……

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,233评论 6 495
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,357评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,831评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,313评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,417评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,470评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,482评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,265评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,708评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,997评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,176评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,827评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,503评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,150评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,391评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,034评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,063评论 2 352

推荐阅读更多精彩内容