链表

作为一种数据结构,链表以其方便的增删查改功能,实现了无数经典有用的程序。

在之前的帖子里,我构建链表的方式是建立一个不储存数据的head节点,然后通过一边输入数据一边建立结点的方式构建整个链表。

与之前不同的是,这里建立的是包含数据的头结点head:

下面是代码,包含创建部分,输出部分,删除部分,和插入部分。

struct Node
{
int data;
Node *next;
}; int n = 0; // 结点数 //=============Node_Creat==============//
Node *Node_Creat()
{
Node *head = NULL; // 头指针置NULL Node *p1,*p2; // 开辟一个Node大小的空间 使p1和p2指向该空间
p1 = p2 = (Node *)malloc(sizeof(Node));
if(p1 == NULL || p2 == NULL)
{
printf("Overflow\n");
exit(1);
} scanf("%d",&p1 -> data); while(p1 -> data >= 0)
{
n++; if(head == NULL)
{
head = p1;
}
else
{
p2 -> next = p1;
} p2 = p1;
p1 = (Node *)malloc(sizeof(Node));
if(p1 == NULL)
{
printf("Overflow\n");
exit(1);
} scanf("%d",&p1 -> data);
} p1 = NULL;
p2 -> next = NULL;
return head;
} //============Node_Print==============//
void Node_Print(Node *head)
{
Node *p = head;
if(head == NULL) // 链表为空
{
printf("Empty!\n");
return ;
}
else
{
printf("Total: %d Node\n",n);
while(p != NULL)
{
printf("%d ",p -> data);
p = p -> next;
}
printf("\n");
}
} //==============Node_Delete()===============//
void Node_Delete(Node *head,int num)
{
Node *p1,*p2;
p1 = head; if(p1 == NULL) // 链表为空
{
printf("Empty\n");
return ;
} if(num > n) // 结点不存在
{
printf("Not Found!\n");
return ;
} n--; // 结点数-1 int i;
for(i = 1; i < num; i++)
{
if(p1 -> next == NULL)break; p2 = p1;
p1 = p1 -> next;
} if(p1 == head) // 头节点 即第一个
{
head = p1 -> next;
}
else
{
p2 -> next = p1 -> next;
free(p1);
} Node_Print(head);
} //============Node_Insert()============//
void Node_Insert(Node* head,int num,int i_data)
{
Node * newNode;
newNode = (Node *)malloc(sizeof(Node));
if(newNode == NULL)
{
printf("Overflow\n");
exit(1);
}
newNode -> data = i_data;
newNode -> next = NULL; Node *p1,*p2;
p1 = head; for(int i = 1 ;i < num; i++)
{
p2 = p1;
p1 = p1 -> next;
} p2 -> next = newNode;
newNode -> next = p1; n++; // 结点数+1 Node_Print(head);
}

样例:

链表的反转

我这里采用的方法是 改变两个结点间next指针的指向,原结点next指针指向下一个结点,反转后指向前一个结点。

Node* Node_Reverse(Node* head)
{
Node* pNow = head;//当前结点
Node* pPrv = NULL;//当前结点的前一个结点
Node* pReversedHead = NULL;//反转链表头结点 Node* pNext = NULL;//当前结点的下一个结点
while(pNow != NULL)
{
pNext = pNow -> next;
if(pNext == NULL)//如果当前结点的下一个结点为空,那么反转链表的头结点就是当前结点。
pReversedHead = pNow; pNow -> next = pPrv;//当前结点指向前一个结点 pPrv = pNow;//pPrv和pNow往前移动。
pNow = pNext;//这里要使用前面保存下来的pNext,不能使用pNow->next
}
return pReversedHead;//返回反转链表头指针。
}

大概的一个流程图(反转第一个结点)

注:红色箭头代表pPrev,pNext和pNow的移动,蓝色代表next指针的指向。

又见链表 --- 另一种Creat方式与反转的更多相关文章

  1. Android中BroadcastReceiver的两种注册方式(静态和动态)详解

    今天我们一起来探讨下安卓中BroadcastReceiver组件以及详细分析下它的两种注册方式. BroadcastReceiver也就是"广播接收者"的意思,顾名思义,它就是用来 ...

  2. 通过三个DEMO学会SignalR的三种实现方式

    一.理解SignalR ASP .NET SignalR 是一个ASP .NET 下的类库,可以在ASP .NET 的Web项目中实现实时通信(即:客户端(Web页面)和服务器端可以互相实时的通知消息 ...

  3. (转)WCF开发框架形成之旅---WCF的几种寄宿方式

    WCF寄宿方式是一种非常灵活的操作,可以在IIS服务.Windows服务.Winform程序.控制台程序中进行寄宿,从而实现WCF服务的运行,为调用者方便.高效提供服务调用.本文分别对这几种方式进行详 ...

  4. Python模块常用的几种安装方式

    Python模块安装方法 一.方法1: 单文件模块直接把文件拷贝到 $python_dir/Lib 二.方法2: 多文件模块,带setup.py 下载模块包,进行解压,进入模块文件夹,执行:pytho ...

  5. php 递归函数的三种实现方式

    递归函数是我们常用到的一类函数,最基本的特点是函数自身调用自身,但必须在调用自身前有条件判断,否则无限无限调用下去.实现递归函数可以采取什么方式呢?本文列出了三种基本方式.理解其原来需要一定的基础知识 ...

  6. Android数据的四种存储方式

    作为一个完成的应用程序,数据存储操作是必不可少的.因此,Android系统一共提供了四种数据存储方式.分别是:SharePreference.SQLite.Content Provider和File. ...

  7. Android开发_Android数据的四种存储方式

    Android系统一共提供了四种数据存储方式.分别是:SharePreference.SQLite.Content Provider和File.由于Android系统中,数据基本都是私有的的,都是存放 ...

  8. MyEclipse中web服务器的三种配置方式

    初学Javaweb开发的人们都会遇到一个问题,就是服务器环境的搭建配置问题.下面介绍三种服务器的搭建方式. 直接修改server.xml文件 当你写了一个web应用程序(jsp/servlet),想通 ...

  9. Android数据的四种存储方式SharedPreferences、SQLite、Content Provider和File (二) —— SQLite

    SQLite是一种转为嵌入式设备设计的轻型数据库,其只有五种数据类型,分别是: NULL: 空值 INTEGER: 整数 REAL: 浮点数 TEXT: 字符串 BLOB: 大数据 在SQLite中, ...

随机推荐

  1. python center() 函数

    center Python center() 返回一个原字符串居中,并使用空格填充至长度 width 的新字符串.默认填充字符为空格. 语法 center()方法语法: str.center(widt ...

  2. postman返回参数的截取

    同事在使用postman接口测试的时候,遇到这么一个问题,在一个参数里面,返回了一个类似数组的参数,如下: 然后现在需要把数组里面的两个参数分别保存到环境变量里面: 个人的想法是通过截取的方式进行数组 ...

  3. 多项式函数插值:全域多项式插值(一)单项式基插值、拉格朗日插值、牛顿插值 [MATLAB]

    全域多项式插值指的是在整个插值区域内形成一个多项式函数作为插值函数.关于多项式插值的基本知识,见“计算基本理论”. 在单项式基插值和牛顿插值形成的表达式中,求该表达式在某一点处的值使用的Horner嵌 ...

  4. 有关padding的二三事~~

    浏览器支持 所有浏览器都支持 padding 属性. 注释:任何的版本的 Internet Explorer (包括 IE8)都不支持属性值 "inherit". 定义和用法 pa ...

  5. [转]使用C#开发ActiveX控件全攻略

    前言: 这段时间因为工作的需要,研究了一下ActiveX控件.总结如下: 先说说ActiveX的基本概念. 根据微软权威的软件开发指南MSDN(Microsoft Developer Network) ...

  6. Python: 用shell通配符匹配字符串,fnmatch/fnmatchcase

    问题:想使用Unix Shell 中常用的通配符(比如*.py , Dat[0-9]*.csv 等) 去匹配文本字符串 解决方案: 1. fnmatch 模块提供了两个函数—— fnmatch() 和 ...

  7. MyBatis学习笔记(三)——优化MyBatis配置文件中的配置

    转自孤傲苍狼的博客:http://www.cnblogs.com/xdp-gacl/p/4264301.html 一.连接数据库的配置单独放在一个properties文件中 之前,我们是直接将数据库的 ...

  8. 制作系统U盘,不用做任何动作直接从U盘启动装系统(非PE的)

    用U盘装系统可以用PE方式,进入PE系统,选择镜像文件,然后装,这种比较麻烦. 下面介绍一下从U盘启动,直接装系统的方法,这种方法从U盘启动后,不用做任何动作,就像用光盘装系统一样简单 首先要制作一下 ...

  9. Qt的四个常见的图像叠加模式

    Qt的QPainter::CompositionMode提供了多种图像叠加的模式.常见的有QPainter::CompositionMode_SourceOver, QPainter::Composi ...

  10. [转载]ASP.NET-----Repeater数据控件的用法总结

    一.Repeater控件的用法流程及实例: 1.首先建立一个网站,新建一个网页index.aspx. 2.添加或者建立APP_Data数据文件,然后将用到的数据库文件放到APP_Data文件夹中. 3 ...