简介:

  在用C/C++开发系统中,我们知道用数组或者单链表来开发,如果是数据比较大的话,性能很不好,效率也不高。因此常常需要考虑系统的实用性,常常采用双向链表来开发。

示例:

1.数据

typedef struct node{

  int data;                // 数据

  struct node *last;  // 前一个数据节点 

  struct node *next;  // 后一个数据节点

}Node;

typedef struct {

  Node *head;

  Node *tail;

}LinkList;

2.创建链表

LinkList *createList()
{
LinkList *list = malloc(sizeof(LinkList));
list -> head = NULL;
lsit ->tail = NULL;
return list;
}

3.插入

  3.1头插法

void addNodeBeHead(LinkList *list,int num)
{
Node *p = malloc(sizeof(Node));
p -> data = num;
p -> last = NULL;
P -> next = NULL; if(list - > head != NULL)
{
// 1.指向新的节点
// 2.新节点的next指向原来的节点
// 3.新节点变成头节点
list ->head->last = p;
node ->next = list ->head;
list -> head = p;
}
else{
list -> head = P;
list -> tail = p;
}
}

  3.2尾插法

void addNodeAfterTail(LinkList *list,int num)
{
Node *p = malloc(sizeof(Node));
p -> data = num;
p -> last = NULL;
P -> next = NULL; if(list - > head != NULL)
{
// 1.指向新的节点
// 2.新节点的last指向原来的节点
// 3.新节点变成尾节点
list ->tail->next = p;
node ->last = list -> tail;
list -> tail = p;
}
else{
list -> head = P;
list -> tail = p;
}
}

4.打印链表

void printList(LinkList *list)
{
Node *p = list -> head;
if(p == NULL)
{
printf("当前为空链表");
}else
{
while(p != NULL)
{
printf("%s",p-> data);
p = p-> next;
} }
}

5.把链表的头结点删除,并返回头结点的数值

int popHeadNode(LinkList *list)
{
if(list -> head == NULL)
{
printf("the list is empty!");
return -;
}
else if (list -> head == list -> tail)
{
Node *node = list - > head;
int value = node ->data;
list -> head = NULL;
list -> tail = NULL;
free(node);
return value;
}
else
{
Node *node = list -> head;
int value = node -> data;
// 1.head指向下一个节点
// 2.并将下一个的last置空
list -> head = list -> head ->next;
list -> head -> last = NULL;
free(node);
return value; } }

6.统计节点总数

int countNodes(LinkList *list)
{
int count = ;
Node *p = list -> head;
if(list -> head != NULL)
{
while(p != NULL)
{
count++;
p = p -> next;
}
return count; }
return ;
}

7.链表转成数组(注意:数组一定要malloc初始化,否则无法返回)

int *listToArray(LinkList *list)
{
if(list -> head == NULL)
{
return NULL;
} int i = ;
int n = countNodes(list);
int *arr = malloc(sizeof(int) * n);
Node *p = list -> head;
while(p != NULL)
{
arr[i] = p -> data;
i++;
p = p -> next;
}
return arr;
}

8.把链表所有的节点头尾对调

1->2->3->NULL 对调后 NULL->3->2->1

void reverseList(LinkList *list)
{
if(list -> head != NULL)
{
int *arr = listToArray(list);
int i = countNodes(list) - ;
Node *node = list -> head;
while(p != NULL)
{
P -> data = arr[i];
i --;
p = p -> next;
}
}
}

第二种做法:(把原来的链表重新连接)

void reverseList(LinkList *list)
{ Node *prev = NULL;
Node *p = list -> head;
Node *temp;
list -> tail = p; // 尾部变头部
while(p != NULL)
{
temp = p -> next;
p -> next = prev;
prev = p;
p = temp;
}
list ->head = prev;
}

第三篇、C_双向链表(循环链表)的更多相关文章

  1. LWJGL3的内存管理,第三篇,剩下的两种策略

    LWJGL3的内存管理,第三篇,剩下的两种策略 上一篇讨论的基于 MemoryStack 类的栈上分配方式,是效率最高的,但是有些情况下无法使用.比如需要分配的内存较大,又或许生命周期较长.这时候就可 ...

  2. 从0开始搭建SQL Server AlwaysOn 第三篇(配置AlwaysOn)

    从0开始搭建SQL Server AlwaysOn 第三篇(配置AlwaysOn) 第一篇http://www.cnblogs.com/lyhabc/p/4678330.html第二篇http://w ...

  3. (转) 从0开始搭建SQL Server AlwaysOn 第三篇(配置AlwaysOn)

    原文地址: http://www.cnblogs.com/lyhabc/p/4682986.html 这一篇是从0开始搭建SQL Server AlwaysOn 的第三篇,这一篇才真正开始搭建Alwa ...

  4. 【第三篇】ASP.NET MVC快速入门之安全策略(MVC5+EF6)

    目录 [第一篇]ASP.NET MVC快速入门之数据库操作(MVC5+EF6) [第二篇]ASP.NET MVC快速入门之数据注解(MVC5+EF6) [第三篇]ASP.NET MVC快速入门之安全策 ...

  5. 第三篇 Entity Framework Plus 之 Query Cache

    离上一篇博客,快一周,工作太忙,只能利用休息日来写一些跟大家分享,Entity Framework Plus 组件系列文章,之前已经写过两篇 第一篇 Entity Framework Plus 之 A ...

  6. 我的屌丝giser成长记-研三篇

    进入研三以来,基本都是自己的自由时间了,从导师的项目抽离出来,慢慢的都交给师弟他们来负责.研三的核心任务就是找工作以及写毕业论文,因为有导师科研基金项目成果作为支撑,所以自己的论文没什么可担心,一切都 ...

  7. javascript面向对象系列第三篇——实现继承的3种形式

    × 目录 [1]原型继承 [2]伪类继承 [3]组合继承 前面的话 学习如何创建对象是理解面向对象编程的第一步,第二步是理解继承.本文是javascript面向对象系列第三篇——实现继承的3种形式 [ ...

  8. 前端工程师技能之photoshop巧用系列第三篇——切图篇

    × 目录 [1]切图信息 [2]切图步骤 [3]实战 前面的话 前端工程师除了使用photoshop进行测量之外,更重要的是要使用该软件进行切图.本文是photoshop巧用系列的第三篇——切图篇 切 ...

  9. 解剖SQLSERVER 第三篇 数据类型的实现(译)

    解剖SQLSERVER 第三篇  数据类型的实现(译) http://improve.dk/implementing-data-types-in-orcamdf/ 实现对SQLSERVER数据类型的解 ...

  10. 深入理解javascript函数系列第三篇——属性和方法

    × 目录 [1]属性 [2]方法 前面的话 函数是javascript中的特殊的对象,可以拥有属性和方法,就像普通的对象拥有属性和方法一样.甚至可以用Function()构造函数来创建新的函数对象.本 ...

随机推荐

  1. 松下蓄电池与UPS使用和维护

      使用条件及环境1.充电电流(浮充使用):0.15CA以下2.放电电流范围:0.05CA-3CA3.环境温度:0℃-40℃ (适宜的温度是25℃) 4.充电电压:(12V电池推荐值)   周围温度 ...

  2. Altium Designer PCB 常用功能键

    altium designer 5种走线模式的切换 : shift+space 方格与格点的切换:View-Grids-ToggleVisible Grid Kind源点:Edit-Origin-Se ...

  3. [iOS基础控件 - 3.2] transform的使用

    A.概念 可以控制UIView的位置.尺寸.旋转 transform是一种状态,每次都是相对于原始状态作出的形变     1.位置移动 // 向上移动 - (IBAction)top:(UIButto ...

  4. ALM11测试计划页面图解1

    在 ALM 侧栏上的测试下方,选择测试计划. 在查看菜单中,选择测试网格或测试计划树. 在 ALM 侧栏上的测试下方,选择测试计划. 右键单击测试,并选择测试详细信息. 在测试计划树中选择主题文件夹, ...

  5. like用法

    SQL:btitle like '%"+keyword+"%' 存储过程:keyword like ''%'+@keyword+'%'' 直接查找:name like '%wang ...

  6. JQuery事件处理的注意事项

    1.jQuery 名称冲突 jQuery 使用 $ 符号作为 jQuery 的简介方式. 某些其他 JavaScript 库中的函数(比如 Prototype)同样使用 $ 符号. jQuery 使用 ...

  7. Linux 下 expect 脚本语言中交互处理常用命令

    Linux 下 expect 脚本语言中交互处理常用命令 1. #!/usr/bin/expect 告诉操作系统脚本里的代码使用那一个 shell 来执行.这里的 expect 其实和 Linux 下 ...

  8. smarty3--registerPlugin()函数报错问题

    smarty版本:smarty3.1.30 registerPlugin错误信息: Notice: Trying to get property of non-object in E:\Joomla\ ...

  9. Uploadify 3.2 参数属性、事件、方法函数详解

    一.属性 属性名称 默认值 说明 auto true 设置为true当选择文件后就直接上传了,为false需要点击上传按钮才上传 . buttonClass ” 按钮样式 buttonCursor ‘ ...

  10. 英语阅读积累<一>

    Passage 1.Woodpecker    惹treat or trick There are many apple trees in a garden.  They’re good friend ...