一、不能动的“地址”—void指针

1.1 void指针初探

  void *表示一个“不知道类型”的指针,也就不知道从这个指针地址开始多少字节为一个数据。和用int表示指针异曲同工,只是更明确是“指针”。

  因此void*只能表示一个地址,不能用来&取值,也不能++--移动指针,因此不知道多少字节是一个数据单位。

    int nums[] = {,,,,};
void* ptr1 = nums;
//int i = *ptr1; // 对于void指针没法直接取值
int* ptr2 = (int*)nums;
printf("%d,%d\n",ptr1,ptr2);
int i = *ptr2;
printf("%d\n",i);

  从输出结果可以看出,无论是无类型的void指针还是int类型指针,指向的地址都是一样的:

PS:void *就是一个不能动的“地址”,在进行&、移动指针之前必须转型为类型指针。

1.2 void指针的用途

  这里我们看一下我们之前了解的memset函数,其第一个参数就是一个void指针,它可以帮我们屏蔽各种不同类型指针的差异。如下面代码所示,我们既可以传入一个int类型数组的指针,也可以传入一个char类型数组的指针:

    int nums[];
memset(nums,,sizeof(nums));
char chs[];
memset(chs,,sizeof(chs));

  那么,我们也可以试着自己动手模拟一下这个memset函数,暂且命名为mymemset吧:

void mymemset(void *data,int num,int byteSize)
{
// char就是一个字节,而计算机中是以字节为单位存储的
char *ptr = (char*)data;
int i;
for(i=;i<byteSize;i++)
{
*ptr=num;
ptr++;
}
} int main(int argc, char *argv[])
{
int nums[];
mymemset(nums,,sizeof(nums));
int i,len=sizeof(nums)/sizeof(int);
for(i=;i<len;i++)
{
printf("%d ",nums[i]);
}
printf("\n"); return ;
}

  在这个mymemset函数中,我们利用void指针接收不同类型的指针,利用char类型(一个字节)逐个字节读取内存中的每一个字节,最后依次填充指定的数字。由于char类型是一个具体类型,所以可以使用++或者--进行指针的移动。

  对于结构体类型,也可以使用我们的mymemset函数:

typedef struct _Person
{
char *name;
int age;
} Person; Person p1;
mymemset(&p1,,sizeof(Person));
printf("p1.Age:%d\n",p1.age);

  最终的运行结果如下图所示:

void *的用途:在只知道内存,但是不知道是什么类型的时候。

二、函数指针

2.1 指向函数的指针—.NET中委托的原型

  我想用过.NET中的委托的童鞋,对于函数指针应该不会陌生,它是委托的原型。函数指针是一个指向函数的指针,我们可以在C中轻松地定义一个函数指针:

typedef void (*intFunc)(int i);

  这里我们定义了一个无返回值的,只有一个int类型参数的函数指针intFunc。我们可以在main函数中使用这个函数指针来指向一个具体的函数(这个具体的函数定义需要和函数指针的定义一致):

    // 声明一个intFunc类型的函数指针
intFunc f1 = test1;
// 执行f1函数指针所指向的代码区
f1();

  这里test1函数的定义如下:

void test1(int age)
{
printf("test1:%d\n",age);
}

  最终运行结果如下图所示,执行函数指针f1即执行了其所指向的具体的函数:

2.2 函数指针的基本使用

  这里我们通过一个小案例来对函数指针做一个基本的使用介绍。相信大部分的C#或Java码农都很熟悉foreach,那么我们就来模拟foreach对int数组中的值进行不同的处理。具体体现为for循环的代码是复用的,但是怎么处理这些数据不确定,因此把处理数据的逻辑由函数指针指定。

void foreachNums(int *nums,int len,intFunc func)
{
int i;
for(i=;i<len;i++)
{
int num = nums[i];
func(num);
}
} void printNum(int num)
{
printf("value=%d\n",num);
}

  在foreachNums函数中,我们定义了一个intFunc函数指针,printNum函数是满足intFunc定义的一个具体的函数。下面我们在main函数中将printNum函数作为函数指针传递给foreachNums函数。

    int nums[] = { ,,,, };
foreachNums(nums,sizeof(nums)/sizeof(int),printNum);

  最终运行的结果如下图所示:

  通过函数指针,我们可以屏蔽各种具体处理方法的不同,也就是将不确定的因素都依赖于抽象,这也是面向抽象或面向接口编程的精髓。

三、函数指针应用案例

3.1 计算任意类型的最大值-getMax

  (1)定义函数指针及getMax主体:

typedef int (*compareFunc)(void *data1,void *data2);
// getMax 函数参数说明:
// data 待比较数据数组的首地址,uniteSize单元字节个数
// length:数据的长度。{1,3,5,6}:length=4
// 比较data1和data2指向的数据做比较,
// 如果data1>data2,则返回正数
void *getMax(void *data,int unitSize,int length,compareFunc func)
{
int i;
char *ptr = (char*)data;
char *max = ptr; for(i=;i<length;i++)
{
char *item = ptr+i*unitSize;
//到底取几个字节进行比较是func内部的事情
if(func(item,max)>)
{
max = item;
}
} return max;
}

  这里可以看到,在getMax中到底取几个字节去比较都是由compareFunc所指向的函数去做,getMax根本不用关心。

  (2)定义符合函数指针定义的不同类型的函数:

int intDataCompare(void *data1,void *data2)
{
int *ptr1 = (int*)data1;
int *ptr2 = (int*)data2; int i1=*ptr1;
int i2=*ptr2; return i1-i2;
} typedef struct _Dog
{
char *name;
int age;
} Dog; int dogDataCompare(void *data1,void *data2)
{
Dog *dog1 = (Dog*)data1;
Dog *dog2 = (Dog*)data2; return (dog1->age)-(dog2->age);
}

  (3)在main函数中针对int类型和结构体类型进行调用:

int main(int argc, char *argv[])
{
// test1:int类型求最大值
int nums[] = { ,,,, };
int *pMax = (int *)getMax(nums,sizeof(int),sizeof(nums)/sizeof(int),
intDataCompare);
int max = *pMax;
printf("%d\n",max);
// test2:结构体类型求最大值
Dog dogs[] ={{"沙皮",},{"腊肠",},{"哈士奇",},
{"京巴",},{"大狗",}};
Dog *pDog = (Dog *)getMax(dogs,sizeof(Dog),
sizeof(dogs)/sizeof(Dog),dogDataCompare);
printf("%s=%d",pDog->name,pDog->age); return ;
}

  最终运行结果如下图所示:

3.2 C中自带的qsort函数—自定义排序

  qsort包含在<stdlib.h>头文件中,此函数根据你给的比较条件进行快速排序,通过指针移动实现排序。排序之后的结果仍然放在原数组中。使用qsort函数必须自己写一个比较函数。我们可以看看qsort函数的原型:

 void qsort ( void * base, size_t num, size_t size, int ( * comparator ) ( const void *, const void * ) );

  可以看出,qsort的第四个参数就是一个函数指针!其所指向的函数应该是一个返回值为int类型的,参数为两个void指针。那么,我们可以使用上面我们已经写好的两个compare方法作为参数传入qsort来对上面的int数组和结构体数组进行快速排序。

    int nums[] = { ,,,, };
qsort(nums,sizeof(nums)/sizeof(int),sizeof(int),intDataCompare);
int i;
for(i=;i<sizeof(nums)/sizeof(int);i++)
{
printf("%d ",nums[i]);
}
printf("\n"); Dog dogs[] ={{"沙皮",},{"腊肠",},{"哈士奇",},
{"京巴",},{"大狗",}};
qsort(dogs,sizeof(dogs)/sizeof(Dog),sizeof(Dog),dogDataCompare);
for(i=;i<sizeof(dogs)/sizeof(Dog);i++)
{
printf("%s %d ",dogs[i].name,dogs[i].age);
}

  那么,快速排序后是否有结果呢?答案是肯定的,我们可以传入各种比较方法,可以升序排序也可以降序排序。

参考资料

  如鹏网,《C语言也能干大事(第三版)》

作者:周旭龙

出处:http://edisonchou.cnblogs.com

本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文链接。

你必须知道的指针基础-7.void指针与函数指针的更多相关文章

  1. STL函数static void (* set_malloc_handler(void (*f)()))()与函数指针解析

    在C++ STL的SGI实现版本中,一级空间配置器class __malloc_alloc_template中有一个静态函数的实现如下: static void (*set_malloc_handle ...

  2. typdef void(*fun)(void)笔记【函数指针/typdef函数指针】

    1. 函数指针:返回类型(*函数名)(参数表) 2. 使用typdef void(*fun)(void) typedef的功能是定义新的类型.第一句就是定义了一个MyFun的类型,并定义这种类型为 指 ...

  3. typedef void (*Fun) (void) 的理解——函数指针——typedef函数指针

    首先介绍大家比较熟悉的typedef int i;//定义一个整型变量i typedef myInt int: myInt j;//定义一个整型变量j 上面介绍得是我们常用的比较简单的typedef的 ...

  4. C++成员函数指针错误用法警示(成员函数指针与高性能的C++委托,三篇),附好多评论

    今天做一个成绩管理系统的并发引擎,用Qt做的,仿照QtConcurrent搞了个模板基类.这里为了隐藏细节,隔离变化,把并发的东西全部包含在模板基类中.子类只需注册需要并发执行的入口函数即可在单独线程 ...

  5. C语言函数指针基础

    本文写的非常详细,因为我想为初学者建立一个意识模型,来帮助他们理解函数指针的语法和基础.如果你不讨厌事无巨细,请尽情阅读吧. 函数指针虽然在语法上让人有些迷惑,但不失为一种有趣而强大的工具.本文将从C ...

  6. C基础--函数指针的使用

    之前在看代码的时候,看了函数指针的使用,大体分为如下几类: 做一个function list,通过指针索引调用,使得处理功能类似的函数看起来更加清晰: 函数指针作为另一个函数的参数,用作回调: lin ...

  7. 虚函数指针sizeof不为sizeof(void*)

    ref:http://bbs.csdn.net/topics/360249561 一个继承了两个虚基类又增加了自己的一个虚函数pif的类,sizeof(指向pif的指针)竟然是8(X86).我是从这里 ...

  8. C语言基础:函数指针 分类: iOS学习 c语言基础 2015-06-10 21:55 15人阅读 评论(0) 收藏

    函数指针:指向函数的指针变量. 函数名相当于首地址. 函数指针定义:返回值类型  (*函数指针变量名)(参数类型1,参数类型2,....)=初始值 函数指针类型:返回值类型  (*)(参数类型1,参数 ...

  9. 指针专题6-空指针NULL和void指针

    1 NULL指针 一个指针变量可以指向计算机中任何一块内存,不管该内存有没有被分配,也不管该内存有没有使用权限,只要把地址给他,他就可以指向.C语言没有一种机制保证指向内存的正确性,程序员必须自己提高 ...

随机推荐

  1. linux环境中使用转义字符使命令行字符颜色高亮

    而通过转义序列设置终端显示属性的格式为: \033[Param {;Param;...}m 其中转义序列以 \033[ 为开头,m 为设置属性结束,中间部分的 Param 为属性值,{} 表示可以设置 ...

  2. BitHacks

    备份文件时看到的.我以前居然下过这东西. 2016-12-4 12:05:52更新 纯文本格式真棒.假如使用word写的我能拷过来格式还不乱?? Markdown真好. Bit Hacks By Se ...

  3. 虚拟机主机能互相ping通,但是无法远程连接

    首先将虚拟机关机,找到Edit virtual machine Settings(编辑虚拟机设置)

  4. SpringMVC(三) RequestMapping修饰类

    SpringMVC使用@RequestMapping 注解为控制器指定可以处理哪些URL请求. 可以用于类定义以及方法定义: 类定义:提供初步的请求映射信息.相对于WEB应用的根目录. 方法处:提供进 ...

  5. selenium使用笔记(二)——Tesseract OCR

    在自动化测试过程中我们经常会遇到需要输入验证码的情况,而现在一般以图片验证码居多.通常我们处理这种情况应该用最简单的方式,让开发给个万能验证码或者直接将验证码这个环节跳过.之前在技术交流群里也跟朋友讨 ...

  6. http错误代码含义中英文对照

    Http错误代码含义中文 概要当用户试图通过 HTTP 或文件传输协议 (FTP) 访问一台正在运行 Internet 信息服务 (IIS) 的服务器上的内容时,IIS 返回一个表示该请求的状态的数字 ...

  7. IT

    http://www.cnblogs.com/TomXu/archive/2011/12/19/2291448.html " 经常从Recruiter那里得到抱怨:“汤姆,为什么面试者每次回 ...

  8. 网站常用UI整理

    1.UI插件包使用   Bootstrap 地址:http://v3.bootcss.com/getting-started/     Bootstrap buttons 按钮 地址:http://w ...

  9. PS特效精粹

    冲击有力的广告效果 神奇画笔功能 + 强大的图层样式 Photoshop中的三维特效

  10. C#输出文本树形层次,前或者后自定义空格位数

    Indent String with Spaces This example shows how to indent strings using method for padding in C#. T ...