javascript数据结构之双链表插入排序实例详解


Posted in Javascript onNovember 25, 2015

本文实例讲述了javascript数据结构之双链表插入排序实现方法。分享给大家供大家参考,具体如下:

数组存储前提下,插入排序算法,在最坏情况下,前面的元素需要不断向后移,以便在插入点留出空位,让目标元素插入。

换成链表时,显然无需做这种大量移动,根据每个节点的前驱节点“指针”,向前找到插入点后,直接把目标值从原链表上摘下,然后在插入点把链表断成二截,然后跟目标点重新接起来即可。

<!doctype html>
<html>
<head>
  <title>双链表-插入排序</title>
  <meta http-equiv="Content-Type" content="text/html; charset=gb2312" />
</head>
<script type="text/javascript">
  //节点类
  var Node = function (pData) {
    this.next = null; //后继“指针”
    this.prev = null; //前驱"指针"
    this.data = pData;
  }
  //单链表(约定:头节点不放内容,当哨兵位,有效元素从头节点后的第1个元素开始)
  var DbLinkList = function () {
    this.head = new Node(null); //头节点   
    //插入新元素
    this.insert = function (pNodeValue) {
      var newNode = new Node(pNodeValue);
      //如果只有头节点
      if (this.head.next == null) {
        this.head.next = newNode;
        newNode.prev = this.head;
        return;
      }
      //否则遍历找到尾节点
      var p = this.head;
      while (p.next != null) {
        p = p.next;
      }
      p.next = newNode;
      newNode.prev = p;
    }
    //获取第n个元素的数据值
    this.getData = function (index) {
      if (index < 1 || index > this.size) {
        return null;
      }
      var p = this.head;
      var i = 1;
      while (p.next != null && i <= index) {
        p = p.next;
        i += 1;
      }
      return p.data;
    }
    //取尾节点
    this.getTail = function () {
      if (this.head.next == null) {
        return null;
      }
      var p = this.head.next;
      while (p.next != null) {
        p = p.next;
      }
      return p;
    }
    //删除指定位置的元素
    this.removeAt = function (index) {
      if (index < 1 || index > this.size) {
        return null;
      }
      var p = this.head;
      var i = 1;
      //从头开始遍历,找到index位置的前一个元素
      while (p.next != null && i < index) {
        p = p.next;
        i += 1;
      }
      p.next = p.next.next; //修改index位置前一个元素的后继指针
      p.next.prev = p;
      return p.data; //返回删除元素的值    
    }
    //打印所有元素
    this.print = function () {
      document.write("<br/>");
      if (this.head.next == null) {
        return;
      }
      var p = this.head.next;
      while (p.next != null) {
        document.write(p.data + " ");
        p = p.next;
      }
      document.write(p.data + " "); //最后一个元素,需要单独打印
      document.write("<br/>");
    }
    //从后打印所有元素
    this.printFromBack = function () {
      document.write("该链表共有" + this.size + "个元素,从后向前分别为:<br/>");
      var tail = this.getTail();
      var p = tail;
      if (p == null) {
        return;
      }
      while (p.prev != null) {
        document.write(p.data + " ");
        p = p.prev;
      }
      document.write("<br/>");
    }
    //插入排序
    this.insertSort = function () {
      if (this.head.next == null || this.head.next.next == null) {
        return;
      }
      var p = this.head.next;
      while (true) {
        if (p == null) {
          return;
        }
        var t = p.prev;
        //向前查找p之前的插入点
        while (t.prev != null && t.data > p.data) {
          t = t.prev;
        }
        //如果插入点就是p的前驱节点,不用调整,
        //忽略,直接进入下一轮
        if (t.next == p) {
          p = p.next;
          continue;
        }
        //将p的后续节点先保护起来,以便下一轮循环时确定起始位置
        var x = p.next;
        //将p从链表上摘下
        if (p.next != null) {
          p.next.prev = p.prev;
        }
        p.prev.next = p.next;
        //p插入到t之后
        t.next.prev = p;
        p.next = t.next;
        t.next = p;
        p.prev = t;
        this.print(); //打印输出,调试用  
        //重新将p定位到下一轮循环的"正确"起始节点
        p = x;
      }
    }
  }
  var linkTest = new DbLinkList();
  linkTest.insert(10);
  linkTest.insert(9);
  linkTest.insert(8);
  linkTest.insert(7);
  linkTest.insert(6);
  linkTest.insert(5);
  linkTest.insert(4);
  linkTest.insert(3);
  linkTest.insert(2);
  linkTest.insert(1);
  document.write("--排序前---<br/>")
  linkTest.print();
  linkTest.insertSort();
  document.write("<br/>--排序后---<br/>")
  linkTest.print();
</script>
</html>

运行结果如下:

--排序前---

10 9 8 7 6 5 4 3 2 1 

9 10 8 7 6 5 4 3 2 1 

8 9 10 7 6 5 4 3 2 1 

7 8 9 10 6 5 4 3 2 1 

6 7 8 9 10 5 4 3 2 1 

5 6 7 8 9 10 4 3 2 1 

4 5 6 7 8 9 10 3 2 1 

3 4 5 6 7 8 9 10 2 1 

2 3 4 5 6 7 8 9 10 1 

1 2 3 4 5 6 7 8 9 10 

--排序后---

1 2 3 4 5 6 7 8 9 10

希望本文所述对大家JavaScript程序设计有所帮助。

Javascript 相关文章推荐
js 数组操作代码集锦
Apr 28 Javascript
javascript 面向对象编程  function是方法(函数)
Sep 17 Javascript
jQuery操作Select的Option上下移动及移除添加等等
Nov 18 Javascript
鼠标移到图片上变大显示而不是放大镜效果
Jun 15 Javascript
浅谈类似于(function(){}).call()的js语句
Mar 30 Javascript
在JS中操作时间之getUTCMilliseconds()方法的使用
Jun 10 Javascript
jQuery1.9+中删除了live以后的替代方法
Jun 17 Javascript
Node.js Mongodb 密码特殊字符 @的解决方法
Apr 11 Javascript
vue中使用refs定位dom出现undefined的解决方法
Dec 21 Javascript
Vue 实时监听窗口变化 windowresize的两种方法
Nov 06 Javascript
JS实现电话号码的字母组合算法示例
Feb 26 Javascript
Vue+Element实现动态生成新表单并添加验证功能
May 23 Javascript
js获取图片宽高的方法
Nov 25 #Javascript
javascript数据结构之二叉搜索树实现方法
Nov 25 #Javascript
javascript常用经典算法实例详解
Nov 25 #Javascript
javascript实现很浪漫的气泡冒出特效
Sep 05 #Javascript
jQuery插件jquery-barcode实现条码打印的方法
Nov 25 #Javascript
JavaScript编写简单的计算器
Nov 25 #Javascript
HTML5 Shiv完美解决IE(IE6/IE7/IE8)不兼容HTML5标签的方法
Nov 25 #Javascript
You might like
PHP实现微信公众平台音乐点播
2014/03/20 PHP
php正则提取html图片(img)src地址与任意属性的方法
2017/02/08 PHP
PHP实现在windows下配置sendmail并通过mail()函数发送邮件的方法
2017/06/20 PHP
Javascript模板技术
2007/04/27 Javascript
菜鸟javascript基础资料整理3 正则
2010/12/06 Javascript
在VS2008中使用jQuery智能感应的方法
2010/12/30 Javascript
基于jquery循环map功能的代码
2011/02/26 Javascript
自己动手开发jQuery插件教程
2011/08/25 Javascript
下拉菜单点击实现连接跳转功能的js代码
2013/05/19 Javascript
jQuery横向擦除焦点图特效代码分享
2015/09/06 Javascript
window.open打开窗口被拦截的快速解决方法
2016/08/04 Javascript
jquery删除table当前行的实例代码
2016/10/07 Javascript
vue-dialog的弹出层组件
2020/05/25 Javascript
原生JS实现圆环拖拽效果
2017/04/07 Javascript
详解vue嵌套路由-params传递参数
2017/05/23 Javascript
在vue里面设置全局变量或数据的方法
2018/03/09 Javascript
vue-cli构建项目下使用微信分享功能
2018/05/28 Javascript
vue后台管理之动态加载路由的方法
2018/08/13 Javascript
详解适配器在JavaScript中的体现
2018/09/28 Javascript
vue单页面实现当前页面刷新或跳转时提示保存
2018/11/02 Javascript
解决vue2 在mounted函数无法获取prop中的变量问题
2018/11/15 Javascript
浅谈Vue3.0新版API之composition-api入坑指南
2020/04/30 Javascript
Python操作MySQL简单实现方法
2015/01/26 Python
CentOS中升级Python版本的方法详解
2017/07/10 Python
Python数据结构之单链表详解
2017/09/12 Python
Python使用Selenium模块实现模拟浏览器抓取淘宝商品美食信息功能示例
2018/07/18 Python
python微信聊天机器人改进版(定时或触发抓取天气预报、励志语录等,向好友推送)
2019/04/25 Python
python实现小世界网络生成
2019/11/21 Python
python3连接MySQL8.0的两种方式
2020/02/17 Python
python读取图片颜色值并生成excel像素画的方法实例
2021/02/19 Python
CSS3实现银灰色动画效果的导航菜单代码
2015/09/01 HTML / CSS
Gap加拿大官网:Gap Canada
2017/08/24 全球购物
体育纪念品、亲笔签名的体育收藏品:Steiner Sports
2020/07/31 全球购物
运动会口号8字
2014/06/07 职场文书
离婚协议书范本及离婚须知
2014/10/15 职场文书
在Windows下安装配置CPU版的PyTorch的方法
2021/04/02 Python