C语言 复杂的栈(链表栈)
//复杂的栈--链表栈
#include<stdio.h>
#include<stdlib.h> #define datatype int//定义链表栈数据类型
//定义链表栈结构
struct stacklink{
datatype data;
struct stacklink *pnext;
}; typedef struct stacklink StackLink; //判断栈是否为空
int isempty(StackLink *phead);
//进栈
StackLink * push(StackLink *phead, datatype num);
//出栈
StackLink * pop(StackLink *phead, StackLink *pout);
//清空
StackLink * setempty(StackLink *phead);
//遍历栈中的数据
void findall(StackLink *phead); //链表栈容量无限大,但是代价是增加链表遍历成本 void main(){
//定义链表栈的头指针
StackLink * phead = NULL;
//压栈
phead = push(phead, );
phead = push(phead, );
phead = push(phead, );
phead = push(phead, );
phead = push(phead, );
//打印栈中的数据
findall(phead);
//出栈
StackLink *pout = (StackLink *)malloc(sizeof(StackLink));
phead = pop(phead, pout);
printf("\n弹出的元素是%d;\n", pout->data);
free(pout);
////全部出栈
//while (phead!=NULL){
// StackLink *pout = malloc(sizeof(StackLink));
// phead = pop(phead, pout);
// printf("\n弹出的元素是%d;\n", pout->data);
// free(pout);
//}
printf("\n================================\n");
//打印栈中的数据
findall(phead);
printf("\n================================\n");
//清空栈内元素
phead = setempty(phead);
//打印栈中的数据
findall(phead);
system("pause");
} //判断栈是否为空
int isempty(StackLink *phead){
if (phead == NULL)
{
return ;
}
else{
return ;
}
} //进栈
StackLink * push(StackLink *phead, datatype num){
//分配内存空间
StackLink *p = (StackLink *)malloc(sizeof(StackLink));
StackLink *p1 = phead;
p->data = num;
p->pnext = NULL;
//判断栈是否为空
if (isempty(phead))
{
//栈为空
phead = p;
//注意:这里是给指针副本phead赋值,但是main()函数里phead的值并没有变化
//要在函数里修改一个指针的值,应该使用二级指针,但是根据代码优化原则,尽量不使用二级指针,
//因此我们将phead当做返回值返回
}
else{
while (p1->pnext != NULL){
p1 = p1->pnext;
}
p1->pnext = p;
//这里的phead->pnext会直接影响main()函数里phead的值,因为phead->pnext本质上等于(*phead).pnext
//修改的是phead指针指向数据的值,而不是修改phead本身
}
return phead;
} //遍历栈中的数据
void findall(StackLink *phead){
StackLink *p = phead;
while (p != NULL){
printf("%d\n", p->data);
p = p->pnext;
}
} //出栈
StackLink * pop(StackLink *phead, StackLink *pout){
if (phead == NULL)//判断栈是否为空
{
return NULL;
}
else{
pout->pnext = NULL;
//不为空
//出栈,每次弹出最后一个
//分两种情况
//1.只有一个元素,只有一个元素,只需要弹出头指针本身
if (phead->pnext == NULL)
{
//pout = phead; 错误,这样只是改变pout指向的地址,而free(phead)会彻底释放phead指向地址的内存
//结果就是pout也为NULL
pout->data = phead->data; //注意:链表栈删除必须释放内存
free(phead);
phead = NULL;
//这个改变形参phead的值,所以只能返回phead,而有任何地方修改pout的值
return phead;
}
//2.多个元素,找到倒数第二个元素
else{
StackLink *p = phead;
while (phead->pnext->pnext != NULL){
phead = phead->pnext;
}
pout->data = phead->pnext->data;
//删除栈中最后一个元素
free(phead->pnext);
phead->pnext = NULL;
return p;
} }
} //清空
StackLink * setempty(StackLink *phead){
//清空的思路是,1,2,3,4,5通过头指针找到1,将2删除,把3移到2的位置上,再删除3
//不太清楚循环次数,所以用while
StackLink *p = NULL;
while (phead->pnext != NULL){//当第二个元素是NULL的时候,退出循环
//删除第二个
//p是第三个的指针
p = phead->pnext->pnext;
//开始删除第二个
free(phead->pnext);
phead->pnext = p;
}
//最后删除第一个元素
free(phead);
return NULL;
}

C语言 复杂的栈(链表栈)的更多相关文章
- Python与数据结构[1] -> 栈/Stack[0] -> 链表栈与数组栈的 Python 实现
栈 / Stack 目录 链表栈 数组栈 栈是一种基本的线性数据结构(先入后出FILO),在 C 语言中有链表和数组两种实现方式,下面用 Python 对这两种栈进行实现. 1 链表栈 链表栈是以单链 ...
- c语言编程之栈(链表实现)
用链表实现栈,完成了出栈入栈功能. #include"stdio.h" typedef int element; //define a struct descirbe a stac ...
- C语言中的堆与栈20160604
首先声明这里说的是C语言中的堆与栈,并不是数据结构中的!一.前言介绍:C语言程序经过编译连接后形成编译.连接后形成的二进制映像文件是静态区域由代码段和数据段(由二部分部分组成:只读数据 段,未初始化数 ...
- C语言实现顺序栈以及栈的特点
什么是栈? 同顺序表和链表一样,栈也是用来存储逻辑关系为 "一对一" 数据的线性存储结构,如下图所示. 从上图我们看到,栈存储结构与之前所学的线性存储结构有所差异,这缘于栈对数据 ...
- C语言学习笔记——堆和栈——未整理
C语言笔记 栈区 栈stack是一种先进后出的内存结构,所有的自动变量,函数的形参都是由编译器自动放出栈中,当一个自动变量超出其作用域时,自动从栈中弹出.出入栈是由C语言编译器自动分配 ...
- 【转】C语言堆栈入门——堆和栈的区别
@2019-04-25 [小记] C语言堆栈入门——堆和栈的区别
- C语言实现链栈的初始化&进栈&出栈&读取栈顶元素
/*链表实现栈的一系列操作*/ #include<stdio.h> #include<stdlib.h> #define OK 1 #define ERROR 0 typede ...
- C语言数据结构基础学习笔记——栈和队列
之前我们学过了普通的线性表,接下来我们来了解一下两种特殊的线性表——栈和队列. 栈是只允许在一端进行插入或删除的线性表. 栈的顺序存储结构也叫作顺序栈,对于栈顶指针top,当栈为空栈时,top=-1: ...
- java——链表、链表栈 LinkedListStack、链表队列 LinkedListQueue
LikedList: package Date_pacage; public class LinkedList<E> { public static void main(String[] ...
随机推荐
- Purchase购物车实例分析
源代码下载地址:http://code4app.com/ios/55655def933bf09d328b5141 此源代码从中学到以下四个知识点 第一:关于右边只有一个被选中的效果展现,左边部分代码内 ...
- 深入理解Activity-任务,回退栈,启动模式
一.任务.回退栈的概念 一个任务是多个能够和用户进行交互并且能完成某种功能的activities的集合. 这些Activity被安排回退栈中,能在合适的时候被打开显示出来. 在我们使用Android手 ...
- CSS 子选择器(六)
一.子选择器 子选择器中前后部分之间用一个大于号隔开,前后两部分选择符在结构上属于父子关系. 子选择器是根据左侧选择符指定的父元素,然后在该父元素下寻找匹配右侧选择符的子元素. 二.简单例子 < ...
- Android源码分析之Message
准备开始写点东西,算是对自己阅读源码的一个记录/笔记,也希望能对同样感兴趣的人有所帮助,希望能坚持下去,加油. 在Android的开发中,我们经常用到Handler.postXXX方法,或者View. ...
- Web应用程序系统的多用户权限控制设计及实现-首页模块【5】
首页模块就是展示不同权限的最终结果了,在阅读这章之前若有些不明白,可看看后续的单独的权限模块,用户模块,目录模块后从整体上再看首页模块. 阅读该模块需要一定或者是比较熟练的js知识,EasyUI Ta ...
- 怎样在VS2010中打开VS2012的项目
VS2012中对C#的支持度非常好,不管是编写方便程度(不需要插件就能高亮代码及代码自动提示功能),还对MFC的一些功能优化很多. 我们可以修改两个工程文件来把VS2012的工程文件一直到VS2010 ...
- Jmeter组件执行顺序与作用域
一.Jmeter重要组件: 1)配置元件---Config Element: 用于初始化默认值和变量,以便后续采样器使用.配置元件大其作用域的初始阶段处理,配置元件仅对其所在的测试树分支有效,如,在同 ...
- MySQL数据库初识(二)
8. 向数据表中插入数据记录(INSERT): 向数据表中插入数据记录有两种方法: 基本语法1:INSERT INTO 数据表 (字段名1,字段名2,字段名3……字段名n) VALUES (数据值1, ...
- HTTP 协议中的 Content-Encoding 和 Transfer-Encoding(内容编码和传输编码)
转自:http://network.51cto.com/art/201509/491335.htm Transfer-Encoding,是一个 HTTP 头部字段,字面意思是「传输编码」.实际上,HT ...
- nyoj 203 三国志 dijkstra+01背包
题目链接:http://acm.nyist.net/JudgeOnline/problem.php?pid=203 思路:先求点0到每个点的最短距离,dijkstra算法,然后就是01背包了 我奇怪的 ...