zoukankan      html  css  js  c++  java
  • js实现双向链表

    1.概念

      上一个文章里我们已经了解到链表结构,链表的特点是长度不固定,不用担心插入新元素的时候新增位置的问题。插入一个元素的时候,只要找到插入点就可以了,不需要整体移动整个结构。

      这里我们了解一下双向链表的结构。尽管从链表中头节点遍历到尾节点很容易,但是反过来,从后向前遍历就没有那么简单。通过给Node对象增加一个属性,该属性存储指向前驱节点的链接,这样就容易多了。此时祥链表中插入一个节点需要更多的工作,我们需要指出该节点正确的前驱和后续。但是在从链表中删除节点的时候效率更高了,不需要再查找待删除节点的前驱节点了。如下图1演示了双向链表的工作原理。

    图1

    首先是要为Node类增加一个previouse属性,这个属性指向当前节点的前驱:

    function Node(element){
        this.element = element;
        this.next = null;
        this.previous = null;
    }

      双向链表的insert()方法和单项链表的类似,但是需要设置新节点的previouse属性,是其指向该节点的前驱。该方法的定义如下:

    复制代码
    function insert(newElement , item){
        var newNode = new Node(newElement);
        var current = this.find(item);
        newNode.next = current.next;
        newNode.previous = current;
        current.next = newNode;
    }
    复制代码

      双向链表的删除remove()方法币单项链表的效率更高,因为不需要查找前驱节点了。首选需要在链表中找出存储待删除数据的节点,然后设置该节点的next属性,使其指向待删除节点的后续。设置该节点的后续的previouse的属性,使其指向待删除节点的前驱。如下图2展示删除节点的过程:

    图2

    remove()方法的定义如下:

    复制代码
    function remove(item){
        var currNode = this.find(item);
        if(!(currNode.next == null)){
            currNode.previous.next = currNode.next;
            currNode.next.previous = currNode.previous;
            currNode.next = null;
            currNode.previous = null;
        }
    }
    复制代码

      为了实现反向显示链表中元素的任务,需要给链表增加一个工具方法,用来查找链表中最后一个节点。findLast()方法找出链表中最后一个节点,同时免除从前往后遍历之苦。如下:

    复制代码
    function findLast(){
        var currNode = this.head;
        while (!(currNode.next == null)){
            currNode = currNode.next;
        }
        return currNode;
    }
    复制代码

      有了这个工具方法之后就,就可以很容易的写出反向显示双向链表的元素的方法,dispReverse()方法如下所示:

    复制代码
    function dispReverse(){
        var currNode = this.head;
        currNode = this.findLast();
        while (!(currNode.previous == null)){
            document.write(currNode.element + ' ');
            currNode = currNode.previous;
        }
    }
    复制代码

    2.代码实现

      双向链表就上面一些特性,下面是完整的代码实现和测试代码:

    复制代码
    function Node(element){
        this.element = element;
        this.next = null;
        this.previous = null;
    }
    
    function LList(){
        this.head = new Node('head');
        this.find = find;
        this.insert = insert;
        this.display = display;
        this.remove = remove;
        this.findLast = findLast;
        this.dispReverse = dispReverse;
    }
    
    function dispReverse(){
        var currNode = this.head;
        currNode = this.findLast();
        while (!(currNode.previous == null)){
            document.write(currNode.element + ' ');
            currNode = currNode.previous;
        }
    }
    
    function findLast(){
        var currNode = this.head;
        while (!(currNode.next == null)){
            currNode = currNode.next;
        }
        return currNode;
    }
    
    function remove(item){
        var currNode = this.find(item);
        if(!(currNode.next == null)){
            currNode.previous.next = currNode.next;
            currNode.next.previous = currNode.previous;
            currNode.next = null;
            currNode.previous = null;
        }
    }
    
    function display(){
        var currNode = this.head;
        while (!(currNode.next == null)){
            document.write(currNode.next.element + ' ');
            currNode = currNode.next;
        }
    }
    
    function find(item){
        var currNode = this.head;
        while (currNode.element != item){
            currNode = currNode.next;
        }
        return currNode;
    }
    
    function insert(newElement , item){
        var newNode = new Node(newElement);
        var current = this.find(item);
        newNode.next = current.next;
        newNode.previous = current;
        current.next = newNode;
    }
    
    
    var cities = new LList();
    cities.insert('Conway','head');
    cities.insert('Russellville', 'Conway');
    cities.insert('Carlisle', 'Russellville');
    cities.insert('Alma' , 'Carlisle');
    cities.display();
    document.write('<br>');
    cities.remove('Carlisle');
    cities.display();
    document.write('<br>');
    cities.dispReverse();
    复制代码
  • 相关阅读:
    .NET Core前后端分离快速开发框架(Core.3.0+AntdVue)
    关于框架
    什么是 https ?这应该是全网把 https 讲的最好的一篇文章了
    NET手撸绘制TypeScript类图——下篇
    及时发现问题,力争找到更好的解决办法,而不是得过且过
    一个命令永久禁用Win10驱动程序强制签名
    表达式的结果为true或false
    Parcel 搭建浏览器自动刷新的 非 SPA项目
    Gulp 搭建前端非SPA 项目, 修改文件浏览器自动刷新
    vue 里 this.$parent 作用
  • 原文地址:https://www.cnblogs.com/manshufeier/p/9583070.html
Copyright © 2011-2022 走看看