文章目录

  • 前言
  • 题目
  • 题解
    • 写前注意点(2点)
    • NO1:单向链表实现
    • NO2:双向链表实现
  • 参考资料

前言

哈喽,我是长路,目前刚刚大三,方向是后端也偶尔捣鼓下前端,现在的主语言是Java。之前一大段时间都是在学习web开发的一些技术,就很久没有进行类似于数据结构、算法之类的学习与刷题,打算这段时间拾起来好好学一学、搞一搞。

这段时间也是机缘巧合看到草帽路飞的博客,加了自学群,正巧看到博主组织在群里组织了leetcode刷题打卡活动,我也就参与进来,为期一个月,打算坚持每天都花一些时间做一些题目,并通过博客的方式来进行记录。

目前跟着一个Github仓库刷题(leetcode):代码随想录leetcode刷题,当前为链表专题。


题目

题目来源leetcode

leetcode地址:707. 设计链表,难度:中等。

题目描述(摘自leetcode):

设计链表的实现。您可以选择使用单链表或双链表。单链表中的节点应该具有两个属性:val 和 next。val 是当前节点的值,next 是指向下一个节点的指针/引用。如果要使用双向链表,则还需要一个属性 prev 以指示链表中的上一个节点。假设链表中的所有节点都是 0-index 的。在链表类中实现这些功能:
get(index):获取链表中第 index 个节点的值。如果索引无效,则返回-1。
addAtHead(val):在链表的第一个元素之前添加一个值为 val 的节点。插入后,新节点将成为链表的第一个节点。
addAtTail(val):将值为 val 的节点追加到链表的最后一个元素。
addAtIndex(index,val):在链表中的第 index 个节点之前添加值为 val  的节点。如果 index 等于链表的长度,则该节点将附加到链表的末尾。如果 index 大于链表长度,则不会插入节点。如果index小于0,则在头部插入节点。
deleteAtIndex(index):如果索引 index 有效,则删除链表中的第 index 个节点。示例:
MyLinkedList linkedList = new MyLinkedList();
linkedList.addAtHead(1);
linkedList.addAtTail(3);
linkedList.addAtIndex(1,2);   //链表变为1-> 2-> 3
linkedList.get(1);            //返回2
linkedList.deleteAtIndex(1);  //现在链表是1-> 3
linkedList.get(1);            //返回3提示:
所有val值都在 [1, 1000] 之内。
操作次数将在  [1, 1000] 之内。
请不要使用内置的 LinkedList 库。

本地调试代码:

class MyLinkedList {public MyLinkedList() {}public int get(int index) {}public void addAtHead(int val) {}public void addAtTail(int val) {}public void addAtIndex(int index, int val) {}public void deleteAtIndex(int index) {}public static void main(String[] args) {MyLinkedList linkedList = new MyLinkedList();linkedList.addAtHead(1); printLinked(linkedList);linkedList.addAtTail(3); printLinked(linkedList);linkedList.addAtIndex(1,2); printLinked(linkedList);   //链表变为1-> 2-> 3System.out.println(linkedList.get(1));  //返回2linkedList.deleteAtIndex(1); printLinked(linkedList);  //现在链表是1-> 3System.out.println(linkedList.get(1)); //返回3}public static void printLinked(MyLinkedList linkedList){ListNode node =  linkedList.head;if(node == null){return;}System.out.print("[");while(node!=null){System.out.print(node.val);node = node.next;}System.out.print("]");System.out.println();}
}

题解

写前注意点(2点)

①注意下面这句话,第 index 个节点之前添加值为 val 的节点实际上与后面等于链表的长度不冲突,只需要把等于条件放在前面即可。

addAtIndex(index,val):在链表中的第 index 个节点之前添加值为 val  的节点。如果 index 等于链表的长度,则该节点将附加到链表的末尾。如果 index 大于链表长度,则不会插入节点。如果index小于0,则在头部插入节点。

②注意点(坑点):红框勾选的一个是获取,一个是删除,注意题目写的是获取或删除第index个节点,从字面上来看传入1,就是删除掉链表中的第一个元素,而实际上则是需要你删除或获取到索引为index的元素,传入1实际上是删除或获取索引为1的元素。


NO1:单向链表实现

思路:这里的话是单向链表实现,每个节点只有一个后置节点,特别的注意点就是我上面提到的两点,其他的我觉得没什么需要提的,主要看下面代码实现。

代码

class MyLinkedList {private ListNode head;//头节点private ListNode tail;//尾结点private int size;//节点数量class ListNode{private int val;private ListNode next;public ListNode(int val,ListNode next) {this.val = val;this.next = next;}}public MyLinkedList() {this.head = null;this.tail = null;this.size = 0;}public int get(int index) {if(!checkExists(index)){return -1;}ListNode cur = this.head;for (int i = 0; i < index; i++) {cur = cur.next;}return cur.val;}public void addAtHead(int val) {ListNode newHead = new ListNode(val, this.head);//更新尾结点if(this.tail == null){this.tail = newHead;}//更新头节点this.head = newHead;this.size++;}public void addAtTail(int val) {ListNode newTail = new ListNode(val, null);//更新尾结点this.tail = newTail;//处理无头节点情况if(this.head == null){this.head = newTail;}else{//遍历到尾节点插入ListNode cur = this.head;for (int i = 0; i < this.size-1; i++) {cur = cur.next;}cur.next = newTail;}this.size++;}public void addAtIndex(int index, int val) {//情况1:index>size,不插入节点if(index > this.size){return;}//情况2:index<0,在头部插入节点(包含index=0情况)if(index <= 0){this.addAtHead(val);return;}//情况3:index=size,添加到链表末尾if(index == this.size){this.addAtTail(val);return;}//最终情况:在指定index位置前插入ListNode newNode = new ListNode(val, null);ListNode pre = this.head;//当前数量>1,index>1情况,找到前置节点for (int i = 1; i < index; i++) {pre = pre.next;}//插入节点操作newNode.next = pre.next;pre.next = newNode;this.size++;}public void deleteAtIndex(int index) {if(!checkExists(index)){return;}//情况1:index=0情况if(index == 0){this.head = this.head.next;if(size == 1){this.tail = null; //若是当前数量也为1,尾节点也要删除}this.size--;return;}//定位到要删除的指定节点前一个ListNode cur = head;for (int i = 1; i < index; i++) {cur = cur.next;}//判断是否是尾节点,是的话更新尾节点if(cur.next == this.tail){this.tail = cur;}else{//删除指定节点cur.next = cur.next.next;}this.size--;}//检查当前链表索引是否存在public boolean checkExists(int index){if(index < 0 || index >= size){return false;}return true;}}


NO2:双向链表实现

案例参考leetcode题解的某个评论:我是理解了jdk-8 LinkedList源码写出来的 有不对的地方,希望指正

使用双向链表的话其实就更加轻松一点,并且在本案例中对于方法重用的设计也更加巧妙!

思路:主要每个节点就是设置了一个前置节点和一个后置节点,相对于前面的单向链表多了一个前置节点。

代码

class MyLinkedList {private class Node{public int val;public Node next;public Node prev;Node(int val){this.val = val;}Node(Node prev,int val,Node next){this.prev = prev;this.val = val;this.next = next;}}//头节点private Node first;//尾结点private Node last;//链表长度private int size;public MyLinkedList() {this.first = null;this.last = null;this.size = 0;}public boolean checkIndex(int index){if(index<0 || index>=size){return false;}return true;}public int get(int index) {if(!checkIndex(index)){return -1;}if(index == 0){return this.first.val;}else if(index == this.size-1){return this.last.val;}else{return indexOf(index).val;}}private Node indexOf(int index) {//判断当前索引在链表中的大致位置,若是在左边优先从头节点遍历if(index < size>>1 ){Node cur = this.first;for (int i = 0; i < index; i++) {cur = cur.next;}return cur;}else{//从后向前遍历Node cur = this.last;for (int i = 1 ;i< this.size-index; i++) {cur = cur.prev;}return cur;}}public void addAtHead(int val) {//记录保存一下当前的第一个节点Node f = this.first;Node newNode = new Node(null, val, f);this.first = newNode;//更新一下头节点//如果原本第一个节点为null,说明原本链表为空,此时尾结点也肯定为空if(f == null){this.last = newNode;}else{f.prev = newNode;//原本的前置节点进行更新}this.size++;}public void addAtTail(int val) {//记录保存一下当前的尾节点Node l = this.last;Node newNode = new Node(l, val, null);this.last = newNode;//更新一下尾节点if(l == null){this.first = newNode;}else{l.next = newNode;//更新一下当前头节点}this.size++;}public void addAtIndex(int index, int val) {if(index == 0){this.addAtHead(val);}else if(index == this.size){this.addAtTail(val);}else if(index>this.size){return;}else{//此时index是在当前链表必存在的情况add(index,val);}}public void add(int index, int val) {Node cur = indexOf(index);Node preNode =  cur.prev;Node newNode = new Node(preNode, val, cur);//更新索引为index节点的后置节点为新节点preNode.next = newNode;cur.prev = newNode;this.size++;}public void deleteAtIndex(int index) {if(!checkIndex(index)){return;}//当前节点数量=1情况if(this.size == 1){this.size--;this.first = null;this.last = null;return;}//当前节点数量>1情况:找到指定要删除的节点Node delNode = indexOf(index);//删除节点为头节点情况if(index == 0){this.first = delNode.next;this.first.prev = null;}else if(index == this.size-1){//为尾结点情况this.last = this.last.prev;this.last.next = null;}else{//中间情况,前后节点都要进行更新delNode.prev.next = delNode.next;delNode.next.prev = delNode.prev;}this.size--;}
}


参考资料

[1]. leetcode评论

[2]. 代码随想录—707.设计链表


我是长路,感谢你的耐心阅读。如有问题请指出,我会积极采纳!
欢迎关注我的公众号【长路Java】,分享Java学习文章及相关资料
Q群:851968786 我们可以一起探讨学习
注明:转载可,需要附带上文章链接

leetcode【链表—中等】707.设计链表相关推荐

  1. Suzy找到实习了吗Day 3 | 链表开始啦 203移除链表元素 707设计链表 206 反转链表

    定义链表的结构 class ListNode:def __init__(self, val, next=None): #构造函数self.val = valself.next = next 尾部nod ...

  2. LeetCode刷题---707. 设计链表(双向链表-带头尾双结点)

    文章目录 一.编程题:707. 设计链表(双向链表-带头尾双结点) 1.题目描述 2.示例1: 3.提示: 二.解题思路 1.思路 2.复杂度分析: 3.算法图解(双向链表) 三.代码实现 三.单向链 ...

  3. 力扣707设计链表(单链表,JavaScript)

    1,头指针指向列表第一个元素,尾指针指向链表最后一个元素,链表末尾的index=this.size-1 var MyLinkedList = function() {this.size=0this.h ...

  4. day03链表基础_移除链表元素_设计链表_反转链表

    链表理论基础 链表是一种通过指针串联在一起的线性结构,每一个节点由两部分组成,一个是数据域一个是指针域(存放指向下一个节点的指针),最后一个节点的指针域指向null(空指针的意思). 链表的入口节点称 ...

  5. leetcode 707 设计链表

    设计链表的实现.您可以选择使用单链表或双链表.单链表中的节点应该具有两个属性:val 和 next.val 是当前节点的值,next 是指向下一个节点的指针/引用.如果要使用双向链表,则还需要一个属性 ...

  6. 链表-1(链表理论基础、移除链表元素、设计链表翻转链表)

    链表理论基础 链表是一种通过指针串联在一起的线性结构,每一个节点由两部分组成,一个是数据域一个是指针域(存放指向下一个节点的指针),最后一个节点的指针域指向null(空指针的意思). 单链表 双链表 ...

  7. LeetCode 707. 设计链表(List)

    文章目录 1. 设计一个单链表 2. 双向链表 1. 设计一个单链表 在链表类中实现这些功能: get(index):获取链表中第 index 个节点的值.如果索引无效,则返回-1. addAtHea ...

  8. Leetcode 707.设计链表

    传送门:力扣 答案在文后/ 在链表类中实现这些功能: get(index):获取链表中第 index 个节点的值.如果索引无效,则返回-1. addAtHead(val):在链表的第一个元素之前添加一 ...

  9. My Seventh Page - 设计链表 - By Nicolas

    这一篇page对应的是leetcode上面707.设计链表这个题目,首先这个题目的描述还是比较简单的,就是我们可以选择使用单链表或者双链表实现链表对应的增删改查的许多功能.小尼这个题目大部分都是看的题 ...

  10. 设计链表(单链表、双链表)

    707. 设计链表 设计链表的实现.您可以选择使用单链表或双链表.单链表中的节点应该具有两个属性:val 和next.val 是当前节点的值,next 是指向下一个节点的指针/引用.如果要使用双向链表 ...

最新文章

  1. python第五章_Python数据分析-第5章Series(下)
  2. 解决Visual SVN在linux下不能访问的问题
  3. 使用Denyhost来阻止恶意连接SSH的IP
  4. sudo apt-get install 安装软件总是出现“404 NOT FOUND” 错误的解决方案 !
  5. android 添加广告用proguard混淆后不显示问题解决方法
  6. 最新综述 | 强化学习中从仿真器到现实环境的迁移
  7. xml文件 卷积神经网络_理解卷积神经网络中的输入与输出形状(Keras实现)
  8. adoquery.parameters流化
  9. Touch Driver介绍
  10. 为什么有的人有心事就容易失眠?
  11. php筛选怎么做,thinkphp条件筛选 例子
  12. plsql以及instantclient下载安装配置使用
  13. 从多个文档在Word 2010中创建主文档
  14. 多线程处理大量数据 java
  15. Java学习笔记(五):一张图总结完JVM8基础概念
  16. WPF-实现TextBox和PasswordBox显示背景文字
  17. 基于C+++FLTK实现(WinForm)超市收银系统【100010032】
  18. jQuery migrate 介绍
  19. 短视频是屌丝逆袭的一个绝好的机会
  20. Jzoj5410 小型耀斑

热门文章

  1. Git之(一)Git是什么
  2. 【Scratch】青少年蓝桥杯_每日一题_12.01_角色装扮
  3. Golang Go语言 安装包 下载 官方包 与 Golang 中文网
  4. 数据分析三大神器之一:Numpy
  5. book mac pro怎么重装系统_Macbook Pro 2011完全重装系统
  6. 判断43是不是质数用c语言,1是素数吗(c语言判断一个数为素数)
  7. 分销系统的用户关系数据库设计~
  8. python断言语句的语法_Python断言教程
  9. Ubuntu(Linux) 磁盘分区方案
  10. linux中测试环境的搭建