C语言使用 void 指针实现Stack

数据结构基础课的教材使用C来实现数据结构,上学期看别的书用C++实现时还没什么感觉,一对比就发现C++的类和模板用起来比C方便多了。

在写四则运算计算器的时候,int写一遍,char还得写一遍感觉十分痛苦,百度了一下有没有别的解决办法,发现了下面这篇文章。

如何利用堆栈及逆波兰表达式进行数学四则运算:http://www.xuebuyuan.com/1602553.html

关键是在于理解 void 指针与一般指针的不同之处是它只保存数据的首地址,而不知道也不关心数据究竟有多少个字节。

因此利用 memcpy 可以无视数据类型进行赋值,这就让我们能够创建多种类型的Stack。

由于 memcpy 的参数是 void 指针,所以Push和Pop的都只接受地址作为参数,并且直接将改变应用于相应的内存单元,并不返回值。

此外,此处实现的Top指针也和一般的Stack定义不同,此处的Top指针总是指向下一个可用的内存单元。

仿照着写了一个Stack的实现如下:

 1 // stack.h
 2 // This stack is implemented by void pointer, so it is GENERIC.
 3 //
 4 // How the void pointer allows assignments of different data types:
 5 // first, a normal pointer is binded to a data type, so it knows the number of bytes of the
 6 // data it points to. second, it has an address, which is the start-address of the data.
 7 // so they can tell the compiler where to start and where to stop accessing the data.
 8 //
 9 // but void pointer is binded to no data type. it only knows the starting-address.
10 // so it can be forcely assigned to any pointers, thus allowing assignments of all data types.
11 // so what we do is: when we create the stack, we MALLOC a block of free space, and we take a
12 // integer, the typesize, which specifies the length of the data.
13 // We use the function MEMCPY, which directly access the addresses,to push the data in the stack
14 // and pop data out of the stack.
15 //
16 // The difference from the definiton:
17 // the stack pointer here is different from the definition.
18 // when the stack is empty, Top == base.
19 // when an element is pushed in, Top points to the address after the last byte of the element.
20 // so the Top always points to the next available space.
21
22 #include<stdio.h>
23 #include<stdlib.h>
24
25 struct Record
26 {
27     void *Top;
28     void *base;
29     int stacksize;
30     size_t typesize;
31 };
32
33 typedef struct Record *Stack;
34
35 Stack CreateStack(int stacksize, size_t typesize);
36 void Push(Stack S, void *data);
37 void Pop(Stack S, void *data);
38 int IsEmpty(Stack S);
39 void Clear(Stack S);
40 void Destroy(Stack S);
// stack.c
#include"stack.h"
#include<string.h>

Stack CreateStack(int stacksize, size_t typesize)
{
    Stack S = (Stack)malloc(sizeof(struct Record));
    S->base = malloc(stacksize*typesize);
    if(!S->base)
    {
        printf("Out of space.\n");
        exit(1);
    }

    S->Top = S->base;
    S->stacksize = stacksize;
    S->typesize = typesize;

    return S;
}

void Push( Stack S, void *data)
{
    if( (int)(S->Top - S->base) + S->typesize > S->stacksize )
    {
        printf("Out of space to Push.\n");
        exit(1);
    }

    memcpy(S->Top, data, S->typesize);
    S->Top = (void*)( (int)S->Top + S->typesize );
}

  // GNU defines the arithmetic of void* equals to char(which is defined as byte).
  // ANSI says we cannot do arithmetic on void* because we don‘t know the type.
  // so when we update the Top pointer, first regard it as an int, and convert it
  // back when we are finished.

void Pop( Stack S, void *data)
{
    if( S->Top == S->base )
    {
        printf("Error:Popping an empty stack.\n");
        exit(1);
    }

    S->Top = (void*)( (int)S->Top - S->typesize);
    memcpy(data, S->Top, S->typesize);
}

void Clear(Stack S)
{
    S->Top = S->base;
}

int IsEmpty(Stack S)
{
    return S->Top == S->base;
}

void Destroy(Stack S)
{
    free(S->base);
}
时间: 2024-08-01 13:17:07

C语言使用 void 指针实现Stack的相关文章

C语言:void指针

使用前必须进行强制类型转换 #include <stdio.h> void test(const void *p); int main(){ int i = 123; int *p = &i; puts("my test:"); test(p); } void test(const void *p){ const int *a = (const int *)p; printf("%d\n", *a); }

C语言学习之空指针NULL以及void指针详解

本文和大家分享的主要是c 语言 空指针NULL 以及 void 指针相关内容,一起来看看吧,希望对大家 学习c语言有所帮助. 空指针 NULL 一个指针变量可以指向计算机中的任何一块内存,不管该内存有没有被分配,也不管该内存有没有使用权限,只要把地址给它,它就可以指向,C 语言没有一种机制来保证指向的内存的正确性,程序员必须自己提高警惕. 很多初学者会在无意间对没有初始化的指针进行操作,这是非常危险的,请看下面的例子: 1. #include 2. int main () { 3. char *

程序猿之---C语言细节27(函数无参数时细节、函数默认返回int型证明、return默认还回值、void指针++操作)

主要内容:函数无参数时细节.函数默认返回int型证明.return默认还回值.void指针++操作 一.函数无参数时细节 函数无参数时应该加上void 在c语言中一个函数 void f(); 在使用时传递参数f(2);没有报错,而在c++中则会报错 最好加上void来明确函数是无参数的 二.函数默认返回类型为int型 见下面程序 三.return默认返回1 细节:return不可返回执行栈内存中的指针,因为该内存在函数体结束时自动销毁 四.void 指针++操作 void *p; p++; //

C/C++语言void及void指针深层探索(转)

转自:http://www.lanou3g.com/blog/sort/SelfiOS/page/78 1.概述 许多初学者对C/C++语言中的void及void指针类型不甚理解,因此在使用上出现了一些错误.本文将对void关键字的深刻含义进行解说,并详述void及void指针类型的使用方法与技巧. 2.void的含义 void的字面意思是“无类型”,void *则为“无类型指针”,void *可以指向任何类型的数据. void几乎只有“注释”和限制程序的作用,因为从来没有人会定义一个void变

C语言空指针NULL以及void指针

空指针NULL 在C语言中,如果一个指针不指向任何数据,我们就称之为空指针,用NULL表示.例如: int *p = NULL; 注意区分大小写,null 没有任何特殊含义,只是一个普通的标识符. NULL 是一个宏定义,在stdio.h被定义为: #define NULL ((void *)0) (void *)0表示把数值 0 强制转换为void *类型,最外层的( )把宏定义的内容括起来,我们自己进行宏定义时也推荐这么做,防止发生歧义. 总之,p 的值为 0.你可以输出 p 的值: #in

【转载】C/C++语言void及void指针深层探索

C/C++语言void及void指针深层探索 1.概述许多初学者对C/C++语言中的void及void指针类型不甚理解,因此在使用上出现了一些错误.本文将对void关键字的深刻含义进行解说,并详述void及void指针类型的使用方法与技巧. 2.void的含义void的字面意思是“无类型”,void *则为“无类型指针”,void *可以指向任何类型的数据.void几乎只有“注释”和限制程序的作用,因为从来没有人会定义一个void变量,让我们试着来定义:void a;这行语句编译时会出错,提示“

void指针

1回顶部 热门文章:C++中extern “C”含义深层探索 编程实现盗2005版QQ源码 1.概述 许多初学者对C/C++语言中的void及void指针类型不甚理解,因此在使用上出现了一些错误.本文将对void关键字的深刻含义进行解说,并详述void及void指针类型的使用方法与技巧. 2.void的含义 void的字面意思是“无类型”,void *则为“无类型指针”,void *可以指向任何类型的数据. void几乎只有“注释”和限制程序的作用,因为从来没有人会定义一个void变量,让我们试

void* 指针有什么用

void*其实可以理解为一个context,利用它可以很好的实现C语言面向对象编程.我觉得使用void*,不是为了看起来合适,而是一个约定,使用C/C++编程的人员看到void*,习惯性的就会认为--这是个私有数据,只有定义的一方才有权解释:这也是我们在回调函数中传递void*作为userdata的基础共识. void*指针其实可以用作泛型,你想象一下你在C中要交换两个变量,如果是两个整数的话,就像: void swap_int(int* lhs, int* rhs) { int tmp = *

void及void指针类型

1.概述 许多初学者对C/C++语言中的void及void指针类型不甚理解,因此在使用上出现了一些错误.本文将对void关键字的深刻含义进行解说,并 详述void及void指针类型的使用方法与技巧. 2.void的含义 void的字面意思是“无类型”,void *则为“无类型指针”,void *可以指向任何类型的数据. void几乎只有“注释”和限制程序的作用,因为从来没有人会定义一个void变量,让我们试着来定义: void a; 这行语句编译时会出错,提示“illegal use of ty