第17章 经典数据结构类型

  1. 堆栈

堆栈接口提供三种基本的操作:push、pop 和 top。

Push:把一个新值压入到堆栈的顶部。

Pop: 只把顶部元素从堆栈中移除,它并不返回这个值。

Top: 返回顶部元素的值,但它并把顶部元素从堆栈中移除

 

(1)堆栈接口

#ifndef STACK_H

#define STACK_H

#include<stdlib.h>

#define STACK_TYPE int

//push 把一个新值压入到堆栈中,它的参数是需要被压入的值

void push( STACK_TYPE value);

//pop 从堆栈中弹出一个值,并将其丢弃

void pop();

//top 返回堆栈顶元素的值,但不对堆栈进行修改

STACK_TYPE top();

//is_empty 如果堆栈为空,返回TRUE,否则返回FALSE

int is_empty();

//is_full 如果堆栈已满,返回TRUE,否则返回FALSE

int is_full();

//create_stack 创建堆栈,参数指定堆栈可以保存多少个元素(动态才有)

void create_stack(size_t size);

//destroy_stack 销毁堆栈,它释放堆栈所使用的内存(动态才有)

void destroy_stack();

#endif

(2)数组堆栈(静态数组存储)

#include "stack.h"

#include <assert.h>

#define STACK_SIZE 100

//存储堆栈中值的数组和一个指向堆栈顶部元素的指针

static  STACK_TYPE  stack[STACK_SIZE];

static  int         top_element=-1;

void push( STACK_TYPE value )

{

assert ( ! is_full() ); //和if用法差不多,如assert参数为0,则终止

top_element += 1;

stack [top_element] = value;

}

void pop ()

{

assert ( !is_empty() );

top_elememt -= 1;

}

STACK_TYPE top ()

{

assert ( !is_empty() );

return stack[top_element];

}

int is_empty ()

{

return top_element == -1;

}

int is_full ()

{

return top_element == STACK_SIZE - 1;

}

(3)动态数组堆栈(动态数组存储)

#include "stack.h"

#include <stdio.h>

#include <stdlib.h>

#include <malloc.h>

#include <assert.h>

static  STACK_TYPE  *stack;

static  size_t      stack_size;

static  int         top_element = -1;

void create_stack ( size_t size )

{

assert ( stack_size == 0 );   //还没有分配内存空间

stack_size = size;

  stack = malloc ( stack_size * sizeof( STACK_TYPE ));  //分配内存空间

assert ( stack != NULL );

}

void destroy_stack ()

{

assert ( stack_size > 0 );

stack_size = 0;

free ( stack );

stack = NULL;

}

void push ( STACK_SIZE value )   //进栈

{

assert ( !is_full() );

top_element += 1;

stack [top_element] = value;

}

void pop ()    //移动顶端指针

{

assert ( !is_empty() );

top_element -= 1;

}

STACK_TYPE top ()     //出栈

{

assert ( !is_empty() );

return stack[top_element];

}

int is_empty ()

{

assert ( stack_size > 0 );     //有内存空间只是没有值

return top_element == -1;

}

int is_full ()

{

assert ( stack_size > 0 );

return top_element == stack_size -1;

}

(4)链式堆栈(链表存储)

#include "stack.h"

#include <stdio.h>

#include <stdlib.h>

#include <malloc.h>

#include <assert.h>

#define FALSE 0

typedef  struct  STACK_NODE {

STACK_TYPE  value;

struct  STACK_NODE  *next;

}StackNode;

static  StackNode  *stack;

void  create_stack ( size_t size )

{

}

void destroy_stack ()

{

while  ( !is_empty() );

pop ();

}

void push ( STACK_TYPE value)

{

StackNode  *new_node;

new_node = malloc (sizeof( StackNode ))

assert ( new_node ! = NULL );

new_node -> value = value;

new_node -> next = stack;

stack = new_node;

}

void pop ()    //顶部元素移除

{

StackNode  *first_node;

assert ( !is_empty() );

first_node = stack;

stack = first_node -> next;

free ( first_node );

}

STACK_TYPE top ()   //返回顶部元素值

{

assert ( !is_empty() );

return stack -> value;

}

int is_empty ()

{

return stack == NULL;

}

int is_full ()   //链式堆栈不会填满,所以is_full始终返回false

{

return FALSE;

}

2.队列

  (1)队列接口

#ifndef QUEUE_H

#define QUEUE_H

#include <stdlib.h>

#define QUEUE_TYPE

//create_queue 创建一个队列,参数指定队列可以存储的元素的最大数量(动态)

void create_queue ( size_t size );

//destroy_queue 销毁一个队列。

void destroy_queue ();

//insert 向队列添加一个新元素,参数就是要添加的元素

void insert ( QUEUE_TYPE value );

//delete 从队列中移除一个元素并将其丢弃

void delete ();

//first 返回队列中第一个元素的值,但不修改队列本身

QUEUE_TYPE first ();

//is_empty 如果队列为空,返回true,否则返回false

int is_empty ();

//is_full 如果队列已满,返回true,否则返回false

int is_full ();

#endif

(2)数组队列

#include "queue.h"

#include <stdio.h>

#include <assert.h>

#define  QUEUE_SIZE  100    //队列中元素最大数量

#define  ARRAY_SIZE  ( QUEUE_SIZE + 1) //数组长度

//用于存储队列元素的数组和指向队列头和尾的指针

static  QUEUE_TYPE  queue[ QUEUE_SIZE ];

static  size_t      front = 1;

static  size_t      rear = 0;

void  insert ( QUEUE_TYPE value )

{

assert ( !is_full() );

rear = ( rear + 1 ) % ARRAY_SIZE;

queue[rear] = value;

}

void  delete ()

{

assert ( !is_empty() );

front = ( front + 1 ) % ARRAY_SIZE;

}

QUEUE_TYPE  first ()

{

assert ( !is_empty() );

return queue[front];

}

int  is_empty ()

{

return  ( rear + 1 ) % ARRAY_SIZE == front;

}

int  is_full ()

{

return  ( rear + 2 ) % ARRAY_SIZE == front;

}

(3)动态数组队列

#include "queue.h"

#include <stdio.h>

#include <stdlib.h>

#include <malloc.h>

#include <assert.h>

static  QUEUE_TYPE  *queue;

static  size_t      queue_size;

static  size_t      array_size = queue_size + 1;

static  size_t      front = 1;

static  size_t      rear = 0;

void  create_queue ( size_t size )

{

assert ( queue_size == 0 );

queue_size = size;

queue = malloc ( queue_size * sizeof( QUEUE_TYPE ) );

assert ( queue != NULL );

}

void  destory_queue ()

{

assert ( queue_size > 0 );

queue_size == 0;

free ( queue );

queue == NULL;

}

void  insert ( QUEUE_TYPE value )

{

assert ( !is_full() );

rear = ( rear + 1 ) % array_size;

queue[rear] = value;

}

void  delete ()

{

assert ( !is_empty() );

front = ( front + 1 ) % array_size;

}

QUEUE_TYPE  first ()

{

assert ( !is_empty() );

return queue[front];

}

int  is_empty ()

{

assert ( queue_size > 0 );    //有内存没有存值

return  ( rear + 1 ) % ARRAY_SIZE == front;

}

int  is_full ()

{

assert ( queue_size > 0 );

return  ( rear + 2 ) % ARRAY_SIZE == front;

}

(4)链式队列

#include "queue.h"

#include <stdio.h>

#include <stdlib.h>

#include <malloc.h>

#include <assert.h>

#define  FALSE  0

typedef  struct  QUEUE_NODE {

QUEUE_TYPE  value;

struct  QUEUE_NODE  *next;

}QueueNode;

static  QueueNode  *front;

static  QueueNode  *rear;

void  destroy_queue ()

{

while ( !is_empty() );

delete ();

}

void  insert ( QUEUE_TYPE value )

{

QueueNode  *new_node;

new_node = ( QueueNode *) malloc ( sizeof( QueueNode ) );

assert ( new_node != NULL );

new_node -> value = value;

new_node -> next = NULL;

//把节点插到队列尾部

if ( rear == NULL )

front = new_node;

else

rear -> next = new_node;

rear = new_node;

}

void  delete ()

{

QueueNode  *first_node;

assert ( !is_empty() );

first_node = front -> next;

free ( front );

front = first_node;

if ( front == NULL )

rear = NULL;

}

QUEUE_TYPE  first ()

{

assert ( !is_empty() );

return  front -> value;

}

int  is_empty ()

{

return  front == NULL;

}

int  is_full ()

{

return  FALSE;

}

C和指针之学习笔记(6)的更多相关文章

  1. 02.C语言关于指针的学习笔记

    指针是一个特殊的变量,它里面存储的数值被解释成为内存里的一个地址. 要搞清一个指针需要搞清指针的四方面的内容: 指针的类型,指针所指向的 类型,指针的值或者叫指针所指向的内存区,还有指针本身所占据的内 ...

  2. 《征服c指针》学习笔记-----统计文本单词数目的程序word_count

    1.程序的要求:对用户指定的英文文本文件(包括标准输入),将英文单词按照字母顺序输出到用户指定的文本文件中(包括标准输出),并且在各单词后面显示单词的出现次数. 2.模块设计: 主要分为:1.从输入流 ...

  3. &lt;深入理解C指针&gt;学习笔记和总结 第四章 指针和数组

    数组是一个什么玩意: 数组和指针我的理解,有同样之处也有不同之处.因有同样之处,因此一些资料上说,数组和指针本质是同样的.因有不同之处,因此也有一些资料上说,数组和指针是不一样的. 同样之处: 数组名 ...

  4. C和指针之学习笔记(5)

    第10章 使用结构和指针 单链表 typedef struct NODE { struct NODE *link; int value; } Node; 插入到一个有序单链表: #include< ...

  5. C和指针之学习笔记(4)

    第9章 字符串 字符串的输入与输出 int  ch;  char strings[80];  FILE *input; (1)scanf(“%c”,&ch);   printf(“%c \n” ...

  6. C和指针之学习笔记(3)

    第8章 数组 1.数组与指针 数组名是一个个元素的地址. int  a[10];  int  b[10];  int  *c; (1) c = & a[0]; &a[0]表示一个指向数 ...

  7. C和指针之学习笔记(2)

    第6章 指针 1.在一组字符串中查找字符: #include<stdio.h> #include<assert.h> #include<stdlib.h> #def ...

  8. C和指针之学习笔记(1)

    第1章 1.输入字符串 while((ch=getchar())!=EOF  &&  ch!=’\n’) ; ch=getchar() while(ch!=EOF  && ...

  9. C语言学习笔记之成员数组和指针

    成员数组和指针是我们c语言中一个非常重要的知识点,记得以前在大学时老师一直要我们做这类的练习了,但是最的还是忘记了,今天来恶补一下.     单看这文章的标题,你可能会觉得好像没什么意思.你先别下这个 ...

随机推荐

  1. LintCode 532: Reverse Pairs

    LintCode 35: Reverse Linked List 题目描述 翻转一个链表. 样例 给出一个链表1->2->3->null,这个翻转后的链表为3->2->1 ...

  2. 【leetcode 简单】第四十九题 颠倒二进制位

    颠倒给定的 32 位无符号整数的二进制位. 示例: 输入: 43261596 输出: 964176192 解释: 43261596 的二进制表示形式为 000000101001010000011110 ...

  3. 1、编写第一个java程序--Hello—World

    1.下载JDK8.0文件 下载网址:https://www.oracle.com/technetwork/java/javase/downloads/jdk8-downloads-2133151.ht ...

  4. Django之原生Ajax操作

    Ajax主要就是使用 [XmlHttpRequest]对象来完成请求的操作,该对象在主流浏览器中均存在(除早起的IE),Ajax首次出现IE5.5中存在(ActiveX控件). 先通过代码来看看Aja ...

  5. php文件上传——php经典实例

     php文件上传——php经典实例 表单页 <html> <head> <title>文件上传</title> <meta charset='ut ...

  6. win8扁平风格的物流公司网站后台管理模板——后台

    链接:http://pan.baidu.com/s/1o79Zp2M 密码:tqrz

  7. GitBash、EGit、SourceTree三个Git管理工具对比

    Git管理工具对比(GitBash.EGit.SourceTree) GitBash是采用命令行的方式对版本进行管理,功能最为灵活强大,但是由于需要手动输入希望修改的文件名,所以相对繁琐. EGit是 ...

  8. Geoserver WFS跨域设置

    测试版本为geoserver2.11.0. 两种方法都可以实现跨域设置: 第一种: 下载跨域jar包jetty-servlets.jar(下载geoserver使用的对应jetty版本——可以查看&l ...

  9. scrollreveal(页面滚动显示动画插件支持手机)

    scrollreveal.js是一款可以轻易实现桌面和移动浏览器元素随页面滚动产生动画的js插件.该插件通过配置可以在页面滚动,元素进入视口时产生炫酷的动画效果,同时还支持元素的3D效果,非常的实用. ...

  10. java基础12 抽象类(及关键字:abstract)

    抽象类:abstract 1.应用的场景 我们描述一类事物时,存在着某种行为,但这种行为目前不具体,那么我们就可以抽取这种行为的声明,但是不去实现这种行为,我们就需要使用抽象类. 2.抽象的好处 强制 ...