双向链表也叫双链表,是链表的一种,它的每个数据结点中都有两个指针,分别指向直接后继和直接前驱。所以,从双向链表中的任意一个结点开始,都可以很方便地访问它的前驱结点和后继结点。一般我们都构造双向循环链表

由于双向链表可以方便地实现正序和逆序两个方向的插入、查找等功能,在很多算法中经常被使用,

这里用C++构造了一个双向链表,提供了对双向链表的插入、查找、删除节点、排序等功能,其中排序提供了插入排序和冒泡排序两种方式

  1. #include<iostream>
  2.  
  3. using namespace std;
  4.  
  5. class Node //组成双向链表的节点
  6. {
  7. public:
  8. int data;
  9. Node * pNext;
  10. Node * pLast;
  11. };
  12.  
  13. class List //构造一个双向链表
  14. {
  15. private:
  16. Node * pHead;
  17. Node * pTail;
  18. int length;
  19. public:
  20. List(int length)    //创建双向链表
  21. {
  22. this->length=length;
  23. pHead=new Node();
  24. pHead->pLast=NULL;
  25. pTail=pHead;
  26. for(int i=;i<length;i++)
  27. {
  28. Node * temp=new Node();
  29. cout<<"please enter the no"<<i+<<" Node's data:";
  30. cin>>temp->data;
  31. temp->pNext=NULL;
  32. temp->pLast=pTail;
  33. pTail->pNext=temp;
  34. pTail=temp;
  35. }
  36. }
  37.  
  38. void traverseList()    //正向遍历
  39. {
  40. Node * p=pHead->pNext;
  41. while(p!=NULL)
  42. {
  43. cout<<p->data<<endl;
  44. p=p->pNext;
  45. }
  46. }
  47.  
  48. void traverseListReturn()    //逆向遍历
  49. {
  50. Node * p=pTail;
  51. while(p->pLast!=NULL)
  52. {
  53. cout<<p->data<<endl;
  54. p=p->pLast;
  55. }
  56. }
  57.  
  58. void sortList() //冒泡排序
  59. {
  60. Node * p=new Node();
  61. Node * q=new Node();
  62. int temp;
  63. for(p=pHead->pNext;p->pNext!=NULL;p=p->pNext)
  64. {
  65. for(q=p->pNext;q!=NULL;q=q->pNext)
  66. {
  67. if(q->data<p->data)
  68. {
  69. temp=q->data;
  70. q->data=p->data;
  71. p->data=temp;
  72. }
  73. }
  74. }
  75. }
  76.  
  77. void sortListByInsertWay() //插入排序
  78. {
  79. if(pHead->pNext==NULL||pHead->pNext->pNext==NULL)
  80. {
  81. return;
  82. }
  83. Node * p2=pHead->pNext->pNext;
  84. Node * p1=pHead;
  85. pHead->pNext->pNext=NULL;
  86. while(p2)
  87. {
  88. Node * pN=p2->pNext;
  89. while(p1->pNext)
  90. {
  91. if(p2->data<p1->pNext->data)
  92. {
  93. p2->pNext=p1->pNext;
  94. p2->pLast=p1;
  95. p1->pNext->pLast=p2;
  96. p1->pNext=p2;
  97. break;
  98. }
  99. p1=p1->pNext;
  100. }
  101. if(p1->pNext==NULL)
  102. {
  103. p2->pNext=NULL;
  104. p2->pLast=p1;
  105. p1->pNext=p2;
  106. }
  107. p2=pN;
  108. }
  109.  
  110. //重新查找pTail的位置
  111. Node * pt=pHead;
  112. while(pt->pNext)
  113. {
  114. pt=pt->pNext;
  115. }
  116. pTail=pt;
  117. }
  118.  
  119. void changeList(int num,int position)    //修改链表中指定位置的节点
  120. {
  121. Node * p=pHead->pNext;
  122. if(position>length||position<=)
  123. {
  124. cout<<"over stack !"<<endl;
  125. return;
  126. }
  127. for(int i=;i<position-;i++)
  128. {
  129. p=p->pNext;
  130. }
  131. p->data=num;
  132. }
  133.  
  134. void insertList(int num,int position)    //插入数据
  135. {
  136. Node * p=pHead->pNext;
  137. if(position>length||position<=)
  138. {
  139. cout<<"over stack !"<<endl;
  140. return;
  141. }
  142. for(int i=;i<position-;i++)
  143. {
  144. p=p->pNext;
  145. }
  146. Node * temp=new Node();
  147. temp->data=num;
  148. temp->pNext=p;
  149. temp->pLast=p->pLast;
  150. p->pLast->pNext=temp;
  151. p->pLast=temp;
  152. length++;
  153. }
  154.  
  155. void clearList()      //清空
  156. {
  157. Node * q;
  158. Node * p=pHead->pNext;
  159. while(p!=NULL)
  160. {
  161. q=p;
  162. p=p->pNext;
  163. delete q;
  164. }
  165. p=NULL;
  166. q=NULL;
  167. }
  168.  
  169. void deleteList(int position)   //删除指定位置的节点
  170. {
  171. Node * p=pHead->pNext;
  172. if(position>length||position<=)
  173. {
  174. cout<<"over stack !"<<endl;
  175. return;
  176. }
  177. for(int i=;i<position-;i++)
  178. {
  179. p=p->pNext;
  180. }
  181. p->pLast->pNext=p->pNext;
  182. p->pNext->pLast=p->pLast;
  183. delete p;
  184. length--;
  185. }
  186.  
  187. int getItemInList(int position)      //查找指定位置的节点
  188. {
  189. Node * p=pHead->pNext;
  190. if(position>length||position<=)
  191. {
  192. cout<<"over stack !"<<endl;
  193. return ;
  194. }
  195. for(int i=;i<position-;i++)
  196. {
  197. p=p->pNext;
  198. }
  199. return p->data;
  200. }
  201.  
  202. ~List()
  203. {
  204. Node * q;
  205. Node * p=pHead->pNext;
  206. while(p!=NULL)
  207. {
  208. q=p;
  209. p=p->pNext;
  210. delete q;
  211. }
  212. p=NULL;
  213. q=NULL;
  214. }
  215.  
  216. };
  217.  
  218. int main()
  219. {
  220. List l();
  221. l.traverseList();
  222. cout<<"AFTER SORT------------------------------------------------------"<<endl;
  223. // l.sortList(); //冒泡排序
  224. l.sortListByInsertWay(); //插入排序
  225. l.traverseList();
  226. cout<<"AFTER INSERT-----------------------------------------------------"<<endl;
  227. l.insertList(,);
  228. l.traverseList();
  229. cout<<"AFTER DELETE-----------------------------------------------------"<<endl;
  230. l.deleteList();
  231. l.traverseList();
  232. cout<<"Return Traverse---------------------------------------------"<<endl;
  233. l.traverseListReturn();
  234. cout<<"Find the Second Node's data:"<<l.getItemInList()<<endl;
  235. return ;
  236. }

基于双向链表的增删改查和排序(C++实现)的更多相关文章

  1. Mybatis_3.基于注解的增删改查

    1.实体类User.java public class User { private int id; private String name; private int age; //getter.se ...

  2. ASP.NET Web API基于OData的增删改查,以及处理实体间关系

    本篇体验实现ASP.NET Web API基于OData的增删改查,以及处理实体间的关系. 首先是比较典型的一对多关系,Supplier和Product. public class Product { ...

  3. [转]ASP.NET Web API基于OData的增删改查,以及处理实体间关系

    本文转自:http://www.cnblogs.com/darrenji/p/4926334.html 本篇体验实现ASP.NET Web API基于OData的增删改查,以及处理实体间的关系. 首先 ...

  4. node-express项目的搭建并通过mongoose操作MongoDB实现增删改查分页排序(四)

    最近写了一个用node来操作MongoDB完成增.删.改.查.排序.分页功能的示例,并且已经放在了服务器上地址:http://39.105.32.180:3333. Mongoose是在node.js ...

  5. java实现双向链表的增删改查

    双向链表的增删改查 和单链表的操作很像:https://blog.csdn.net/weixin_43304253/article/details/119758276 基本结构 1.增加操作 1.链接 ...

  6. Node.js、express、mongodb 入门(基于easyui datagrid增删改查)

    前言 从在本机(win8.1)环境安装相关环境到做完这个demo大概不到两周时间,刚开始只是在本机安装环境并没有敲个Demo,从周末开始断断续续的想写一个,按照惯性思维就写一个增删改查吧,一方面是体验 ...

  7. Mybatis实现部门表增删改查以及排序

    废话不说,直接开门见山! 需要在WebContent下的lib下导入两个包 mybatis-3.2.5.jar ojdbc6.jar package com.xdl.entity; import ja ...

  8. Mybatis_2.基于XML的增删改查

    1.实体类User.java public class User { private int id; private String name; private int age; //getter.se ...

  9. 基于django做增删改查组件,分页器组件

    增删改查组件 一.Djangoadmin的启发 二.基于Djangoadmin实现数据的增删改查 分页器组件 分页器组件的介绍以及源码解读 补充:源码下载,

随机推荐

  1. lua MVC框架 Orbit初探

    介绍 http://keplerproject.github.io/orbit/ Orbit是lua语言版本的MVC框架. 此框架完全抛弃CGILUA的脚本模型, 支持的应用, 每个应用可以卸载一个单 ...

  2. file_get_contents抓取远程URL内容

    /** * POST URL * @param $url * @param null $post * @return false / string */ public static function ...

  3. innerHTML

    对于innerHTML 属性,几乎所有的元素都有innerHTML属性,它是一个字符串,用来设置或获取位于对象起始和结束标签内的HTML.(获取HTML当前标签的起始和结束里面的内容) 下面的例子返回 ...

  4. ubuntu登陆出现问题

    手贱改了用户root权限结果登陆时提示system administrator is not allowed to login from this screen(郁闷勒) 这时可以按ctrl+F2进入 ...

  5. SVN服务端启动解决方案(2013-12-10 记)

     解决每一次开机都得用DOS启动SVN服务,而DOS窗口又无法关闭的情况 1.安装Setup-Subversion-1.8.5.msi搭建好SVN服务端(下载地址:http://subversion. ...

  6. python入门练习题1

    常见python入门练习题 1.执行python脚本的两种方法 第一种:给python脚本一个可执行的权限,进入到当前存放python程序的目录,给一个x可执行权限,如:有一个homework.py文 ...

  7. CentOS 7下Wireshark捕获USB数据包

    1. 软件准备 安装Wireshark # yum install wireshark wireshark-gnome .csharpcode, .csharpcode pre { font-size ...

  8. 指定页面配置https(apache/tomcat)

    apache/tomcat服务器下配置https         apache下配置https:             首先在网站根目录下,找到.htaccess文件(如果没有则新建),apache ...

  9. Android跨进程通信的四种方式

    由于android系统中应用程序之间不能共享内存.因此,在不同应用程序之间交互数据(跨进程通讯)就稍微麻烦一些.在android SDK中提供了4种用于跨进程通讯的方式.这4种方式正好对应于andro ...

  10. Normalize.css 初识

    一. 用来干嘛的 一个现代的.准备好了支持 HTML5 技术,并且要替代 CSS Reset 处理样式的理念. Normalize.css 使浏览器渲染所有元素更加一致,并且符合现代标准.它只是针对那 ...