数据结构基础-链表

作者: 蝉翅的空响 | 来源:发表于2019-01-14 17:19 被阅读10次

什么是链表

链表是用来存储数据集合的数据结构。有如下属性:

  • 相邻元素通过指针连接
  • 最后一个的后继指针为NULL
  • 链表长度可以增加和缩小
  • 空间按需分配,直至内存耗尽,但是存储指针会相对数组耗费一些额外空间
linkedlist

链表抽象数据结构

主要操作:

  • 添加元素,
  • 删除元素(移除并返回链表中指定位置的元素)

辅助操作:

  • 获取元素个数
  • 查询(寻找从链表表头开始的第n个结点),
  • 清空元素

这里给出插入和删除的java代码示例:

/* time:O(n) space:O(1) */
public ListNode insertLinkedList(ListNode headNode, ListNode nodeToInsert, int positon){
        if (headNode == null) {
            return nodeToInsert;
        }
        if (nodeToInsert == null) {
            return headNode;
        }
        //这里需要效验position是否在有效范围
        if (position == 0) {
            nodeToInsert.setNext(headNode);
            return nodeToInsert;
        }else {
            ListNode previousNode = headNode;
            for (int i = 1; i < positon; i++) {
                previousNode = previousNode.getNext();
            }
            nodeToInsert.setNext(previousNode.getNext());
            previousNode.setNext(nodeToInsert);
        }
        return headNode;
    }
    
public ListNode deleteNodeFromLinkedList(ListNode headNode, int position){
        if (position == 1) {
            ListNode currentNode = headNode.getNext();
            headNode = null;
            return currentNode;
        }else {
            ListNode previousNode = headNode;
            for (int i = 1; i < position; i++) {
                previousNode = previousNode.getNext();
            }
            ListNode curNode = previousNode.getNext();
            previousNode.setNext(curNode.getNext());
        }
        return headNode;
    }

为什么用链表

数组一样能用来存储数据集合,那为什么用多种数据结构来做一样的事情。因为后来发现数组在处理一些情况下的弊端,所以开始分使用情景用不同的工具干同样的事情。
先说说数组在一些情况下的缺点,

  • 大小是固定的,需要分配一块连续的空间块,就造成有时候无法分配能存储整个数组的内存空间(当数组规模太大时),(当然动态数组通过到达数组最大长度后再申请更大容量数组来加入新元素)大空间没有足量的元素也会造成空间的浪费。
  • 基于位置的插入操作实现复杂,考虑最糟的一种情况,插入到数组开始的位置,就需要移动原有数组的每一个元素。

对数组,链表最大的有点在于在任何位置插入元素的时间开销仅为O(1)。然而查找某个元素的开销为O(n)。

链表、数组和动态数组的对比

linedlist_compare

双向链表与循环链表

双向链表是单项链表的拓展,就是加入指向前一个结点的指针,用NULL表示指针的结束。循环指针就是头指针指向尾结点地址,形成了一个贪吃蛇的形状,没有NULL指针,需要注意无限循环遍历,因为每一个结点都有后继结点。

eg: 用链表实现栈

public class ListNodeStack{
    private ListNode headNode;
    public ListNodeStack(){
        this.headNode = new ListNode(null);
    }
    public void push(int data){
        if(headNode == null){
            headNode = new ListNode(data);
        }else if(headNode.getData() == null){
            headNode.setData(data);
        }else {
            ListNode node = new ListNode(data);
            node.setNext(headNode);
            headNode = node;
        }
    }
    public void pop(){
        if(isEmpty()){
            throw new EmptyStackException("Stack empty");
        }else {
            int data = headNode.getData();
            headNode = headNode.getNext();
            return data;
        }
    }
    public boolean isEmpty(){
        return null == headNode;
    }
    public void deleteStack(){
        headNode = null;
    }
}

eg1:知道链表倒数第n个结点

/* space:O(1) , time:O(n)*/
    public ListNode nthNodeFromEnd(ListNode head, int nthNode){
        ListNode fast = head, slow = head;
        for (int i = 1; i < nthNode; i++) {
            if (fast != null) {
                fast = fast.getNext();
            }
        }
        while(fast != null) {
            fast = fast.getNext();
            slow = slow.getNext();
        }
        return slow;
    }

eg2: 判定给定的链表是以NULL结尾,还是形成了一个环。
解决方法是经典的快慢指针法:试想一下乌龟和兔子在同一个轨道上赛跑。如果它们在同一个环上赛跑,那么跑得快的兔子将赶上跑得慢的乌龟,并在某一点相遇。


image
image
public boolean doesLinkedListContainerLoop(ListNode head){
        if (head == null) {
            return false;
        }
        LsitNode fastPtr = head, slowPtr = head;
        while (fastPtr != null && fastPtr.getNext() != ll) {
            fastPtr = fastPtr.getNext().getNext();
            slowPtr = slowPtr.getNext();
            if (fastPtr == slowPtr) {
                return true;
            }
        }
        return false;
    }

eg3:逆置单向列表

public ListNode reverseListUsingRecursion(ListNode head){
        if (head == null || head.getNext() == null) {
            return head;
        }
        ListNode newHead = reverseListUsingRecursion(head.getNext());
        head.getNext().setNext(head);
        head.setNext(null);
        return newHead;
    }

迭代版本

public ListNode reverseList(ListNode head){
        ListNode tmp = null, nextNode = null;
        while (head != null){
            nextNode = head.getNext();
            head.setNext(tmp);
            tmp = head;
            head = nextNode;
        }
        return tmp;
    }

eg4: 逆置链表,两个为单位进行逆置,eg:1->2->3->4->x => 2->1->4->3->x

//space:o(n), time:O(n)
public ListNode reversePairRecursive(ListNode head) {
        ListNode temp;
        if (head == null || head.getNext() == null) {
            return head;
        }
        temp = head.getNext();
        head.setNext(temp.getNext());
        temp.setNext(head);
        head = temp;
        head.getNext().setNext(reversePairRecursive(head.getNext().getNext()));
        return head;
    }

迭代版本代码:

/* space:o(1), time:O(n) */
    public ListNode reversePairIterative(ListNode head) {
        ListNode temp1 = null;
        ListNode temp2 = null;
        while (head != null && head.getNext() != null) {
            if (temp1 != null) {
                temp1.next().setNext(head.getNext());
            }
            temp1 = head.getNext();
            head.setNext(temp1.getNext());
            temp1.setNext(head);
            if (temp2 == null) {
                temp2 = temp1;
            }
            head = head.getNext();
        }
        return temp2;
    }

eg5: 逆置链表,k个为单位进行逆置,eg:1 2 3 4 5 6 7 8 9 10 , k=3: 3 2 1 6 5 4 9 8 7 10

public ListNode getKPlusOneThNode(int k, ListNode head) {
        ListNode kNode = head;
        if (head == null) {
            return head;
        }
        for (int i = 1; kNode != null && (i <= k); i++, kNode = kNode.getNext()) {
            if (k == i) {
                return kNode;
            }
        }
        return head.getNext();
    }

public boolean hasKNode(ListNode head, int k) {
        if (head == null) {
            return head;
        }
        for (int i = 1; head != null && (i <= k); i++, head = head.getNext()) {
            if (i == k) {
                return true;
            }
        }
        return false;
    }

public ListNode reverseBolckOf(ListNode head, int k) {
        ListNode newHead, curNode, temp, next;
        curNode = head;
        if (k ==0 || k == 1) {
            return head;
        }
        if (hasKNode(head, k)) {
            newHead = getKPlusOneThNode(k, head);
        }else {
            newHead = head;
        }
        while (curNode != null && hasKNode(curNode, k)) {
            temp = getKPlusOneThNode(k, curNode);
            int i = 0;
            while (i < k) {
                next = curNode.getNext();
                curNode.setNext(temp);
                temp = curNode;
                curNode = next;
                i++;
            }
        }
        return newHead;
    }

看动画理解「链表」实现LRU缓存淘汰算法

相关文章

  • 2020年最新整理的java学习路线

    阶段一:数据结构 一、基础 1、基本的数据结构 [](1)基础概念 [](2)数组 [](3)链表 [](4)栈:...

  • 最新整理的java学习路线

    阶段一:数据结构 一、基础 1、基本的数据结构 [](1)基础概念 [](2)数组 [](3)链表 [](4)栈:...

  • 数据结构-链表

    链表与数组是个相对互补,数组不足的地方恰好用链表可以满足,它也算基础数据结构,可用来表示其他逻辑数据结构。 链表在...

  • java 集合精华一页纸

    从最基础的数据结构 数组|链表|树 开始,基于这些基础数据结构通过各种设计组合成具备特定功能的数据结构,这些结构是...

  • 用Java写单向链表

    数据结构—单向链表 为了巩固自己的基础知识,这次就用 Java 来写一个单向链表。问:什么是单向链表?首先链表是数...

  • 链表、递归、堆、Hashmap、归并排序算法

    java数据结构——链表 链表(Linked list)是一种常见的基础数据结构,是一种线性表,但是并不会按线性的...

  • 数据结构&算法

    数据结构的简单分类 一维数据结构基础: 数组 array(String),链表 linked list高级:栈(s...

  • 数据结构---链表

    1.链表基础 链表是数据结构中另一种最基础的数据结构,数组需要开辟一段连续的存储空间,所以在初始化的时候需要指定大...

  • Java链表

    一、链表介绍 数组和链表都是最基础的线性数据结构,可以用来实现栈,队列等非线性,有特定应用场景的数据结构。数组作为...

  • C++ 数据结构之链表

    链表 一、 何为链表 引用维基百科的介绍, 链表(Linked list)是一种常见的基础数据结构, 是一种线性...

网友评论

    本文标题:数据结构基础-链表

    本文链接:https://www.haomeiwen.com/subject/nisjdqtx.html