由于现在只学了C语言所以就写这个C语言版的栈的基本操作

这里说一下 :网上和书上都有这种写法 int InitStack(SqStack &p)

&p是取地址  但是这种用法好像C并不支持 ,C++才支持,所以用

C语言写就需要使用指针

代码如下:

 #include <stdio.h>
#include <stdlib.h>
#define STACK_INIT_SIZE 100//储存空间初始分配量
#define STACKINCREMENT 10//存储空间分配增量
#define OK 1
#define ERROR 0
typedef int StackType; //栈元素类型 typedef struct {
StackType *base; //在构造之前和销毁之后,base的值为NULL
StackType *top; //栈顶指针
int stacksize; //当前已分配的存储空间,以元素为单位
}SqStack; //顺序栈 //栈的初始化
int InitStack(SqStack *p) { p->base = (StackType*)malloc(STACK_INIT_SIZE * sizeof(StackType));
if (p->base == NULL) return ERROR; //内存分配失败
p->top = p->base; //栈顶与栈底相同表示一个空栈
p->stacksize = STACK_INIT_SIZE;
return OK; }
//判断栈是否为空
int EmptyStack(SqStack *p) {
//若为空栈 则返回OK,否则返回ERROR
if (p->top == p->base) return OK;
else return ERROR;
}
//顺序栈的压入
int Push(SqStack *p,StackType e) {
//插入元素e为新的栈顶元素
if ((p->top - p->base)>= p->stacksize) //栈满,追加储存空间
{
p->base = (StackType*)realloc(p->base, (p->stacksize + STACKINCREMENT) * sizeof(StackType));
if (p->base == NULL) return ERROR;// 储存空间分配失败
p->top = p->base + p->stacksize; //可能有人觉得这句有点多余(我当时也是这么想的 后面有解释)
p->stacksize += STACKINCREMENT;
}
*(p->top) = e;
(p->top)++;
return OK;
}
// 顺序栈的弹出
int Pop(SqStack *p,StackType *e) {
//若栈不空,则删除p的栈顶元素,用e返回其值
if (p->top == p->base) return ERROR;
--(p->top);
*e = *(p->top);
return OK; }
//顺序栈的销毁
int DestroyStack(SqStack *p) {
//释放栈底空间并置空
free(p->base);
p->base = NULL;
p->top = NULL;
p->stacksize = ; return OK;
}
//将顺序栈置空 栈还是存在的,栈中的元素也存在,如果有栈中元素的地址任然能调用
int ClearStack(SqStack *p) {
p->top= p->base;
return OK;
}
//返回顺序栈的元素个数
int StackLength(SqStack p) {
//栈顶指针减去栈底指针等于长度,因为栈顶指针指向当前栈顶元素的下一个位置
return p.top - p.base;
}
//返回顺序栈的栈顶元素
int GetTop(SqStack *p, StackType *e) {
//若栈不为空,则用e返回p的栈顶元素
if (p->top > p->base) {
*e = *(p->top - ); return OK;
}
else return ERROR;
}
//从栈顶到栈底对每个元素调用某个函数
int StackTraverse(SqStack p,void (*pfun)(StackType)/*函数指针*/){
//从栈底到栈顶依次对栈中的每个元素调用函数pfun()
while (p.top > p.base)
pfun(*(p.base)++); //先调用后递增
printf("\n");
return OK;
}
//打印栈中元素
void print(StackType stack) {
printf("%d\n", stack); }
//测试栈的各种操作
int main() {
int n,i;
StackType *e,a;
SqStack *pstack,stack;
pstack = &stack;
e=(StackType*)malloc(sizeof(StackType)); //为指针e分配内存地址
InitStack(pstack); //初始化栈 if (EmptyStack(pstack) == 1) printf("-------栈为空-------\n");
printf("请输入栈的元素个数:");
scanf("%d", &n);
for (i = ; i < n; i++)
{
scanf("%d", &a);
Push(pstack, a);
}
if (EmptyStack(pstack) == 0) printf("-------栈不为空-----\n"); printf("栈的长度为:%d\n", StackLength(stack));
printf("--------------------\n");
printf("请输入一个入栈元素:");
scanf("%d", &a);
Push(pstack, a);
printf("--------------------\n");
printf("栈中的元素个数为:%d\n", StackLength(stack));
printf("--------------------\n");
GetTop(pstack, e);
printf("栈顶元素为:%d\n", *e);
printf("--------------------\n");
printf("打印栈中的元素:\n");
StackTraverse(stack, print);
printf("---弹出栈顶元素---\n");
Pop(pstack, e);
printf("弹出的栈顶元素为:%d\n", *e);
printf("--------------------\n");
GetTop(pstack, e);
printf("栈顶元素为:%d\n", *e);
printf("--------------------\n");
printf("打印栈中的元素:\n");
StackTraverse(stack, print);
printf("--------------------\n"); printf("----------清空栈-------\n");
if (ClearStack(pstack) == ) printf("已清空栈\n"); printf("----------销毁栈-------\n");
if (DestroyStack(pstack) == ) printf("已销毁栈\n");
return ; }
    

 

第39行  p->top = p->base + p->stacksize;这句有必要加上吗?  答案是肯定的。
        这一个问题的关键在于 realloc 是怎么实现的,有两种情况:

   如果有足够空间用于扩大mem_address指向的内存块,则分配额外内存,并返回mem_address。
 这里说的是“扩大”,我们知道,realloc是从堆上分配内存的,当扩大一块内存空间时,
realloc()试图直接从堆上现存的数据后面的那些字节中获得附加的字节,如果能够满足,自然天下太平。
 也就是说,如果原先的内存大小后面还有足够的空闲空间用来分配,加上原来的空间大小= newsize。
那么就ok。得到的是一块连续的内存。
   如果原先的内存大小后面没有足够的空闲空间用来分配,那么从堆中另外找一块newsize大小的内存。
 并把原来大小内存空间中的内容复制到newsize中。返回新的mem_address指针。(数据被移动了)。老块被放回堆上。
 如果是第二种情况的话,s->top 就不是原来的 top 了     --百度百科

  写这些代码的时候还是遇到了一些问题 在这里总结一下:

  第一 对于指针的使用要慎重 因为它传递进函数会改变原始数据,所以对于不需要改变

指针指向的值的情况就不要使用指针。

  第二 对于指针的使用  如下定义 int *e=NULL; *e=3;

看着好像没有问题  编译也没问题 但是运行程序就出错

为什么?  没有为指针e分配内存地址 所以引用肯定错误啊(我这个逗逼错误-_-)

正确的用法应该是 int *e=NULL;e=(int*)malloc(sizeof(int)); *e=3;

我之前都是 int *e,a; e=&a; e=3; 这种用法 所以......

												

顺序栈的基本操作(C语言)的更多相关文章

  1. D_S 顺序栈的基本操作

    //  main.cpp #include <iostream> using namespace std; #include "Status.h" typedef in ...

  2. 链栈的基本操作(C语言)

    栈的链式储存结构称为链栈.链栈的节点类型与链式线性表的节点类型 定义相同,不同的是它是仅在表头进行操作的单链表.链栈通常用不带头节 点的单链表来实现,栈顶指针就是链表的头指针 ,如图所示: 代码如下: ...

  3. php模拟顺序栈基本操作

    php模拟顺序栈基本操作 一.总结 写函数什么永远记住边界情况:比如 echo "栈已满!<br/>" ;  栈已空这种,那就能多考虑几种情况,代码就很正宗了 1.对象 ...

  4. C++语言实现顺序栈

    C++语言实现顺序栈 在写C语言实现顺序栈的时候,我已经向大家介绍了栈的特点以及介绍了栈的相关操作,并利用C语言实现了相关算法.在这里小编就不在继续给大家介绍了,需要温习的可以去我的博客看看.在这篇博 ...

  5. 把十进制整数转换为r(r=2)进制输出(顺序栈实现)

    上周的第二个作业补上~~ 上周的要求: 1.给出顺序栈的存储结构定义. 2.完成顺序栈的基本操作函数. 1)      初始化顺序栈 2)      实现入栈和出栈操作 3)      实现取栈顶元素 ...

  6. 数据结构之顺序栈SqStack

    顺序栈SqStack 基本操作 Status InitStack()//构造一个空栈S Status DestroyStack()//销毁栈S,S不再存在 Status ClearStack()//把 ...

  7. 数据结构——Java实现顺序栈

    一.分析 栈是限定仅在表的一端进行插入或删除操作的线性表,对于栈来说,操作端称为栈顶,另一端则称为栈底,栈的修改是按照后进先出的原则进行的,因此又称为后进先出的线性表. 顺序栈是指利用顺序存储结构实现 ...

  8. 数据结构(C实现)------- 顺序栈

    栈是限定仅在表的一端进行插入或删除的纯属表,通常称同意插入.删除的一端为栈顶(Top),对应在的.则称还有一端为栈底(Bottom). 不含元素的栈则称为空栈. 所设栈S={a1,a2,a3,..., ...

  9. C语言实现顺序栈以及栈的特点

    什么是栈? 同顺序表和链表一样,栈也是用来存储逻辑关系为 "一对一" 数据的线性存储结构,如下图所示. 从上图我们看到,栈存储结构与之前所学的线性存储结构有所差异,这缘于栈对数据 ...

随机推荐

  1. 廖雪峰Java6 IO编程-2input和output-7序列化

    1.序列化 序列化是指把一个Java对象变成二进制内容byte[] 序列化后可以把byte[]保存到文件中 序列化后可以把byte[]通过网络传输 一个Java对象要能序列化,必须实现Serializ ...

  2. Python之Eric安装注意事项

    处理该问题: http://www.knowsky.com/950133.html 注意缺乏的是qscintilla 双击install.py进行安装

  3. python之路——14

    王二学习python的笔记以及记录,如有雷同,那也没事,欢迎交流,wx:wyb199594 复习 1.迭代器 1.双下方法:不常直接调用,是通过其他语法触发的 2.可迭代的:可迭代协议——含有__it ...

  4. 知识点:Mysql 基本用法之视图

    视图 视图是一个虚拟表(非真实存在),其本质是[根据SQL语句获取动态的数据集,并为其命名],用户使用时只需使用[名称]即可获取结果集,可以将该结果集当做表来使用. 使用视图我们可以把查询过程中的临时 ...

  5. 涂抹mysql笔记-mysql复制特性

    <>mysql复制特性:既可以实现整个服务(all databases)级别的复制,也可以只复制某个数据库或某个数据库中的某个指定的表对象.即可以实现A复制到B(主从单向复制),B再复制到 ...

  6. 62.纯 CSS 创作一只蒸锅(感觉不好看呀)

    原文地址:https://segmentfault.com/a/1190000015389338 HTML code: <!-- steamer: 蒸锅: lid: 盖子: pot: 锅 --& ...

  7. Mac 系统下 mysql 的安装与配置

    1.mysql 的安装 1)官网下载 mysql 安装包:http://www.mysql.com/downloads/ 2)下载后解压打开安装包,点击 pkg 文件进行安装 3)注意:最后一步弹窗会 ...

  8. 查找单链表中倒数第K个位置上的结点,若查找成功返回该节点的data域,若不成功只返回0

    算法的基本设计思想:定义两个指针变量p和q  初始时均指向头结点的下一个结点(即链表的第一个结点)p沿链表移动,当p移动到第k个结点时,q指针开始与p指针同时移动,当p指针移动到最后一个结点时,q指针 ...

  9. 十七、 Observer 观察者设计模式

    设计: 代码清单: Observer public interface Observer { void update(NumberGenerator generator); } DigitObserv ...

  10. 【Django】RROR 2003 (HY000): Can't connect to MySQL server on 'localhost' (10061)

    刚刚启动项目的时候,突然报了这个错误.之前一直正常 后来百度一下,让我在window的host文件下,把被注释的127.0.0.1   localhost这个的注释取消 然鹅并木有用 直接用cmd连接 ...