void使用详解
2016-06-06 16:45
281 查看
1.空指针
任何一种指针类型都有一个特殊的指针值,即空指针。它既不会指向任何对象或函数,也不是任何对象或函数的地址。而未初始化的指针,则完全可能指向任何地方。
由此可见,空指针与未初始化的指针是完全不同的两个概念。那么,将如何在程序中获得一个空指针呢?
2. 空指针常量与NULL
为了让程序中的空指针使用更加明确,标准C专门定义了一个标准预处理宏NULL,其值为“空指针常量”,通常为0或(void *)0,即在指针上下文中NULL与0是等价的,而未加修饰的0也是完全可以接受的。由于void *指针的特殊赋值属性,比如:
#define NULL ((void *)0)
当NULL定义为((void *)0)时,即NULL是可以赋值给任何类型指针的值,它的类型为void*,而不是整数0,因此初始化“FILE *fp = NULL;”是完全合法的。
而为了区分整数0和空指针0,当需要其它类型的0的时候,即使可能工作,但也不能使用NULL,如果这样处理其格式是错误的,这在非指针上下文中是不能工作的。特别地,不能在需要ASCII空字符(NUL)的地方使用NULL。如果确实需要,则可以自定义为:
#define NUL '\0'
由此可见,常数0是一个空指针常量,而NULL仅仅是它的一个别名。
3. 空指针的用途
一般来说,未初始化是不能使用的非法指针,因为它完全有可能指向任何地方,从而导致程序无法判断它为非法指针。因此,不管指针变量是全局的还是局部的、静态的还是非静态的,都应该在声明它的同时进行初始化,要么赋予一个有效的地址,要么赋予NULL。
标准C规定,全局指针变量的默认值为NULL,而对于局部指针变量则必须明确地指定其初值。因此,void通常用于指针变量的初始化,用来判断一个指针的有效性。比如:
unsigned char *pucBuf=(void *)0; // 定义pucBuf为unsigned char类型指针并初始化为空指针
如果后续的代码忘记初始化指针而直接使用的话,则可能造成程序失败。虽然空指针也是非法指针,但可以通过程序判断并告诉程序员代码可能有问题。也就是说,如果一开始就将指针初始化为空指针,则可避免程序异常。比如:
if(pucBuf==0){
return error; // 如果pucBuf为空指针,则返回参数错误
}
由于void类型指针的不确定性,因此它可以指向任意类型的数据,那么只要在使用时做一个简单的强制类型转换就可以了。比如:
unsignned char *pcData = NULL; // 定义pcData为unsigned char类型指针
void *pvData; // 定义pvData为void类型指针
pvData = pcData; // 无需进行强制类型转换
pcData = (unsigned char*) pvData; // 将pvData强制转换为unsigned char类型指针
显然不存在void类型的对象,也就是说,当对象为空类型时,其大小为0字节;当对象未确定类型时,那么它的大小也是未确定的,因此不能声明void类型变量。比如:
void a; // 非法声明
既然上述声明是非法的,那么,也就不能将sizeof运算符用于void类型。也就意味着,编译器不知道所指对象的大小,由于指针的算术运算总是基于所指对象的大小的,因此不允许对void指针进行算术运算。
总之,在指针声明中,void *表示通用指针的类型。如果void作为函数的返回类型,则表示不返回任何值。如果void位于参数列表中,则表示没有参数。
4. 用无类型指针作为函数参数
通用swap函数的原型为:
void swap(void *pvData1, void *pvData2, int iDataSize)
将a,b两个变量(变量类型必须一样)的值交换的代码如下:
swap(&a, &b, sizeof(a));
通用swap排序函数的参考代码见程序清单1.1。
[cpp] view
plain copy
//程序清单1.1 通用swap排序函数
void swap (void *pvData1, void *pvData2, int iDataSize)
{
unsigned char *pcData1 = NULL;
unsigned char *pcData2 = NULL;
unsigned char ucTmp1;
pcData1 = (unsigned char *)pvData1;
pcData2 = (unsigned char *)pvData2;
do {
ucTmp1 = *pcData1;
*pcData1 = *pcData2;
*pcData2 = ucTmp1;
pcData1++;
pcData2++;
} while (--iDataSize > 0);
}
任何一种指针类型都有一个特殊的指针值,即空指针。它既不会指向任何对象或函数,也不是任何对象或函数的地址。而未初始化的指针,则完全可能指向任何地方。
由此可见,空指针与未初始化的指针是完全不同的两个概念。那么,将如何在程序中获得一个空指针呢?
2. 空指针常量与NULL
为了让程序中的空指针使用更加明确,标准C专门定义了一个标准预处理宏NULL,其值为“空指针常量”,通常为0或(void *)0,即在指针上下文中NULL与0是等价的,而未加修饰的0也是完全可以接受的。由于void *指针的特殊赋值属性,比如:
#define NULL ((void *)0)
当NULL定义为((void *)0)时,即NULL是可以赋值给任何类型指针的值,它的类型为void*,而不是整数0,因此初始化“FILE *fp = NULL;”是完全合法的。
而为了区分整数0和空指针0,当需要其它类型的0的时候,即使可能工作,但也不能使用NULL,如果这样处理其格式是错误的,这在非指针上下文中是不能工作的。特别地,不能在需要ASCII空字符(NUL)的地方使用NULL。如果确实需要,则可以自定义为:
#define NUL '\0'
由此可见,常数0是一个空指针常量,而NULL仅仅是它的一个别名。
3. 空指针的用途
一般来说,未初始化是不能使用的非法指针,因为它完全有可能指向任何地方,从而导致程序无法判断它为非法指针。因此,不管指针变量是全局的还是局部的、静态的还是非静态的,都应该在声明它的同时进行初始化,要么赋予一个有效的地址,要么赋予NULL。
标准C规定,全局指针变量的默认值为NULL,而对于局部指针变量则必须明确地指定其初值。因此,void通常用于指针变量的初始化,用来判断一个指针的有效性。比如:
unsigned char *pucBuf=(void *)0; // 定义pucBuf为unsigned char类型指针并初始化为空指针
如果后续的代码忘记初始化指针而直接使用的话,则可能造成程序失败。虽然空指针也是非法指针,但可以通过程序判断并告诉程序员代码可能有问题。也就是说,如果一开始就将指针初始化为空指针,则可避免程序异常。比如:
if(pucBuf==0){
return error; // 如果pucBuf为空指针,则返回参数错误
}
由于void类型指针的不确定性,因此它可以指向任意类型的数据,那么只要在使用时做一个简单的强制类型转换就可以了。比如:
unsignned char *pcData = NULL; // 定义pcData为unsigned char类型指针
void *pvData; // 定义pvData为void类型指针
pvData = pcData; // 无需进行强制类型转换
pcData = (unsigned char*) pvData; // 将pvData强制转换为unsigned char类型指针
显然不存在void类型的对象,也就是说,当对象为空类型时,其大小为0字节;当对象未确定类型时,那么它的大小也是未确定的,因此不能声明void类型变量。比如:
void a; // 非法声明
既然上述声明是非法的,那么,也就不能将sizeof运算符用于void类型。也就意味着,编译器不知道所指对象的大小,由于指针的算术运算总是基于所指对象的大小的,因此不允许对void指针进行算术运算。
总之,在指针声明中,void *表示通用指针的类型。如果void作为函数的返回类型,则表示不返回任何值。如果void位于参数列表中,则表示没有参数。
4. 用无类型指针作为函数参数
通用swap函数的原型为:
void swap(void *pvData1, void *pvData2, int iDataSize)
将a,b两个变量(变量类型必须一样)的值交换的代码如下:
swap(&a, &b, sizeof(a));
通用swap排序函数的参考代码见程序清单1.1。
[cpp] view
plain copy
//程序清单1.1 通用swap排序函数
void swap (void *pvData1, void *pvData2, int iDataSize)
{
unsigned char *pcData1 = NULL;
unsigned char *pcData2 = NULL;
unsigned char ucTmp1;
pcData1 = (unsigned char *)pvData1;
pcData2 = (unsigned char *)pvData2;
do {
ucTmp1 = *pcData1;
*pcData1 = *pcData2;
*pcData2 = ucTmp1;
pcData1++;
pcData2++;
} while (--iDataSize > 0);
}
相关文章推荐
- -[NSNull countByEnumeratingWithState:objects:count:]: unrecognized selector sent to instance 0x3e4e
- IOS之同步请求、异步请求、GET请求、POST请求
- Vim 配置
- C++独孤九剑第三式——鱼跃于渊(多态机制实现)
- ZC_汇编指令_test
- [转]高效Hash构建方法
- 【数学】计算折射光线方向
- 优化
- 点分布的离散趋势与方位
- Rails中destroy_all和delete_all的差别
- 让shell脚本中的echo输出带颜色
- 南城故事---教你一招就能让安卓手机变得飞快!
- 剑指Offer----面试题26:复杂链表的复制
- History 对象
- scrollReveal.js – 页面滚动显示动画JS
- tjut 2586
- JavaScript:HTML5跨文档消息传递
- OC学习笔记(七)--方法与函数的区别
- Request.GetOwinContext()打不到
- HTTP协议详解(真的很经典)