服务器之家:专注于VPS、云服务器配置技术及软件下载分享
分类导航

PHP教程|ASP.NET教程|Java教程|ASP教程|编程技术|正则表达式|C/C++|IOS|C#|Swift|Android|VB|R语言|JavaScript|易语言|vb.net|

服务器之家 - 编程语言 - C/C++ - 数据结构 栈的操作实例详解

数据结构 栈的操作实例详解

2021-05-20 15:52xpleaf C/C++

这篇文章主要介绍了数据结构 顺序栈的定义、初始化、空栈判断、入栈、出栈操作的相关资料,需要的朋友可以参考下

数据结构 的操作实例详解

说明:

    往前学习数据结构,想运行一个完整的顺序栈的程序都运行不了,因为书上给的都是一部分一部分的算法,并没有提供一个完整可运行的程序,听了实验课,自己折腾了一下,总算可以写一个比较完整的顺序栈操作的小程序,对于栈也慢慢开始有了感觉。下面我会把整个程序拆开来做说明,只要把这些代码放在一个文件中,用编译器就可以直接编译运行了。

一、实现

1.程序功能

  关于栈操作的经典程序,首当要提及进制数转换的问题,利用栈的操作,就可以十分快速地完成数的进制转换。

2.预定义、头文件导入和类型别名

    代码如下:

?
1
2
3
4
5
6
7
8
9
10
#include<stdio.h>
#include<stdlib.h>
#define OVERFLOW -1
#define ERROR 0
#define FALSE 0
#define TRUE 1
#define OK 1
 
typedef int ElemType;
typedef int Status;

    除了两个头文件的导入是必须的之外,下面做两点说明:

(1)其余的常量定义都是可选的,为的就是在下面的代码书写过程中可以尽量使用英文来表达程序的意思,而不是在代码的实现过程中直接使用数字,依个人喜欢,也可以直接使用数字;

(2)使用typedef做类型的别名也仅仅是为了程序中代码的意思更加清晰明了而已,实际也可以不这样使用;

3.顺序栈的定义 

   代码如下:

?
1
2
3
4
5
6
typedef struct{
  ElemType *elem;   //存储空间的基址
  int top;      //栈顶元素的下一个元素,简称栈顶位标
  int size;      //当前分配的存储容量,作用看入栈操作就可以知道
  int increment;   //扩容时,增加的存储容量,作用看入栈操作就可以知道
} SqStack;         //顺序栈名称

4.栈的初始化

    代码如下:

?
1
2
3
4
5
6
7
8
Status InitStack_Sq(SqStack &S, int size, int inc){   //接受3个参数,&S是对结构体的引用
  S.elem = (ElemType*)malloc(size*sizeof(ElemType)); //分配存储空间
  if(S.elem == NULL) return OVERFLOW;   //判断上一步分配存储空间是否成功
  S.top = 0;      //置S为空栈,S.top为0即表示栈为空栈
  S.size = size;    //栈的空间初始容量值
  S.increment = inc;  //栈的空间初始增量值(如果需要扩容)
  return OK;    //上面的执行正常,返回OK
}

5.空栈的判断

    代码如下:

?
1
2
3
4
5
6
7
8
Status StackEmpty_Sq(SqStack S){
  if(S.top == 0)
    return TRUE;
  else
    return FALSE;
}
//空栈的决断是,如果栈为空就返回1,否则就返回0,当然可以不这样规定;
//至于为什么要做空栈的判断,自然是有原因的,下面再看程序的代码时就可以知道了。

6.入栈

    代码如下:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
Status Push_Sq(SqStack &S, ElemType e){  //将元素e压入栈,这里e只是一个形参而已
  ElemType *newbase;    //定义中间变量
  if(S.top>= S.size){    //S.top如果指向最后一个不存储元素的地址时,即S.top大于
    newbase = (ElemType*)realloc(S.elem, //等于S.size时,就表示栈满了
  (S.size + S.increment)*sizeof(ElemType)); //通过realloc动态扩容
   
  if(NULL == newbase) return OVERFLOW; //判断扩容是否成功
  S.elem = newbase;   //扩容成功后才将中间变量的值指向S.elem,防止扩容失败时,
  S.size = S.size + S.increment;   //S.elem指向一个不是原来的位置
  }
  S.elem[S.top] = e;  //将e元素入栈
  S.top++;       //使S.top加1,表示指向的是栈顶位标
  return OK;      //上面操作正常后返回1
}

7.出栈

    代码如下:

?
1
2
3
4
5
Status Pop_Sq(SqStack &S, ElemType &e){  //栈顶元素出栈,赋给元素e
  if(0 == S.top) return ERROR; 
  e = S.elem[--S.top];  //e出栈,并将S.top减1
  return OK;
}

8.进制转换的函数

    其实上面的步骤操作都是为了创建一个顺序栈和定义顺序栈的操作而已,并对可能出现的各种情况做一些相应的举措,完毕后,下面就要使用上面创建的顺序栈以及栈的操作接口了,即在数制转换函数(这里是十进制转八进制)中使用上面的操作接口,代码如下:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
void Converstion(int N){
  SqStack S;
  ElemType e;
  InitStack_Sq(S, 10, 5);  //栈S的初始容量置为10,每次扩容容量为5
   
  while(N != 0){
    Push_Sq(S, N%8);  //将N除以8的余数入栈
    N /= 8;      //N取值为其除以8的商
  }             //理论基础为除8取余法
   
  while(StackEmpty_Sq(S) == FALSE){
    Pop_Sq(S, e);  //依次输出栈中的余数,并赋给元素e
    printf("%d", e); //打印元素
  }

9.main函数

    进制转换函数调用栈操作的接口函数,以实现在数制转换过程中栈的操作;main函数调用数制转换函数,以实现数制的转换,代码如下:

?
1
2
3
4
5
int main(void){
  printf("Enter a number:");scanf("%d",&num);
  Converstion(num);
  printf("\n");
}

二、执行

    有了上面的代码后,就可以在编译器中编译执行了,这里我是用c free 5来进行程序代码的编译:

(1)输入的数为1348时的结果:

数据结构 栈的操作实例详解

(2)输入的数为2526时的结果:

数据结构 栈的操作实例详解

三、完整的代码

    下面把代码都放在一起:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
#include<stdio.h>
#include<stdlib.h>
#define OVERFLOW -1
#define ERROR 0
#define FALSE 0
#define TRUE 1
#define OK 1
 
typedef int ElemType;
typedef int Status;
 
typedef struct{
  ElemType *elem;
  int top;
  int size;
  int increment;
} SqStack;
 
Status InitStack_Sq(SqStack &S, int size, int inc){
  S.elem = (ElemType*)malloc(size*sizeof(ElemType));
  if(S.elem == NULL) return OVERFLOW;
  S.top = 0;
  S.size = size;
  S.increment = inc;
  return OK;
}
 
Status StackEmpty_Sq(SqStack S){
  if(S.top == 0)
    return TRUE;
  else
    return FALSE;
}
 
Status Push_Sq(SqStack &S, ElemType e){
  ElemType *newbase;
  if(S.top>= S.size){
    newbase = (ElemType*)realloc(S.elem,
  (S.size + S.increment)*sizeof(ElemType));
   
  if(NULL == newbase) return OVERFLOW;
  S.elem = newbase;
  S.size = S.size + S.increment;
  }
  S.elem[S.top] = e;
  S.top++;
  return OK;
}
 
Status Pop_Sq(SqStack &S, ElemType &e){
  if(0 == S.top) return ERROR;
  e = S.elem[--S.top];
  return OK;
}
 
void Converstion(int N){
  SqStack S;
  ElemType e;
  InitStack_Sq(S, 10, 5);
   
  while(N != 0){
    Push_Sq(S, N%8);
    N /= 8;
  }
   
  while(StackEmpty_Sq(S) == FALSE){
    Pop_Sq(S, e);
    printf("%d", e);
  }
}
 
int main(void){
  int num;
  printf("Enter a number:");scanf("%d",&num);
  Converstion(num);
  printf("\n");
}

原文链接:http://xpleaf.blog.51cto.com/9315560/1696524

延伸 · 阅读

精彩推荐
  • C/C++c/c++内存分配大小实例讲解

    c/c++内存分配大小实例讲解

    在本篇文章里小编给大家整理了一篇关于c/c++内存分配大小实例讲解内容,有需要的朋友们可以跟着学习参考下。...

    jihite5172022-02-22
  • C/C++C语言main函数的三种形式实例详解

    C语言main函数的三种形式实例详解

    这篇文章主要介绍了 C语言main函数的三种形式实例详解的相关资料,需要的朋友可以参考下...

    ieearth6912021-05-16
  • C/C++c/c++实现获取域名的IP地址

    c/c++实现获取域名的IP地址

    本文给大家汇总介绍了使用c/c++实现获取域名的IP地址的几种方法以及这些方法的核心函数gethostbyname的详细用法,非常的实用,有需要的小伙伴可以参考下...

    C++教程网10262021-03-16
  • C/C++深入C++拷贝构造函数的总结详解

    深入C++拷贝构造函数的总结详解

    本篇文章是对C++中拷贝构造函数进行了总结与介绍。需要的朋友参考下...

    C++教程网5182020-11-30
  • C/C++使用C++制作简单的web服务器(续)

    使用C++制作简单的web服务器(续)

    本文承接上文《使用C++制作简单的web服务器》,把web服务器做的功能稍微强大些,主要增加的功能是从文件中读取网页并返回给客户端,而不是把网页代码...

    C++教程网5492021-02-22
  • C/C++OpenCV实现拼接图像的简单方法

    OpenCV实现拼接图像的简单方法

    这篇文章主要为大家详细介绍了OpenCV实现拼接图像的简单方法,具有一定的参考价值,感兴趣的小伙伴们可以参考一下...

    iteye_183805102021-07-29
  • C/C++C语言实现双人五子棋游戏

    C语言实现双人五子棋游戏

    这篇文章主要为大家详细介绍了C语言实现双人五子棋游戏,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下...

    两片空白7312021-11-12
  • C/C++关于C语言中E-R图的详解

    关于C语言中E-R图的详解

    今天小编就为大家分享一篇关于关于C语言中E-R图的详解,小编觉得内容挺不错的,现在分享给大家,具有很好的参考价值,需要的朋友一起跟随小编来看看...

    Struggler095962021-07-12