新聞中心

深入理解void類型

作者: 時間:2016-11-27 來源:網(wǎng)絡(luò) 收藏
1.空指針

一般來說,程序的起始地址是從“代碼區(qū)”的0地址開始存放的(注:如果插入一個內(nèi)存分布圖,則更能說明問題,此處省略),但實際上現(xiàn)代操作系統(tǒng)并非如此,卻保留了從0開始的一塊內(nèi)存。至于這塊內(nèi)存到底有有多大,與具體的操作系統(tǒng)有關(guān)。如果程序試圖訪問這塊內(nèi)存,則系統(tǒng)提示異常。

本文引用地址:http://butianyuan.cn/article/201611/322491.htm

為什么操作系統(tǒng)不是保留一個字節(jié)呢?由于內(nèi)存管理是按頁來進(jìn)行的,因此無法做到單獨保留一個字節(jié)。盡管如此,但還是有極少數(shù)系統(tǒng)設(shè)定RAM區(qū)從0地址開始,但指向有效變量的指針不會指向0地址。即使“代碼區(qū)”從0地址開始,但在任何情況下,0地址都不是C語言中任何函數(shù)的起始地址,因此指向有效函數(shù)地址的指針也不會指向0地址。

?課外知識延伸

雖然80C51微控制器XDATA區(qū)(外部RAM)是從0地址開始的,但只要對保存在0地址中的變量不進(jìn)行取地址操作(&操作),即可有效地保證指針不會指向0地址。

與此同時,雖然32位ARM7微控制器也是從0地址開始的,但這塊內(nèi)存僅用于存放中斷向量代碼,而不是程序中的有效變量地址,因此即便用空指針來判斷指針的有效性,其仍然是可行的。

基于此,于是將空指針定義為指向0地址的指針。毫無疑問,任何一種指針類型都有一個特殊的指針值,即空指針。它既不會指向任何對象或函數(shù),也不是任何對象或函數(shù)的地址。而未初始化的指針,則完全可能指向任何地方。

由此可見,空指針與未初始化的指針是完全不同的兩個概念。那么,將如何在程序中獲得一個空指針呢?

2.空指針常量與NULL

標(biāo)準(zhǔn)C規(guī)定,在初始化、賦值或比較時,如果一邊是變量或指針類型的表達(dá)式,則編譯器可以確定另一邊的常數(shù)0為空指針,并生成正確的空指針值。即在指針上下文中“值為0的整型常量表達(dá)式”在編譯時轉(zhuǎn)換為空指針。

為了讓程序中的空指針使用更加明確,標(biāo)準(zhǔn)C專門定義了一個標(biāo)準(zhǔn)預(yù)處理宏NULL,其值為“空指針常量”,通常為0或(void *)0,即在指針上下文中NULL與0是等價的,而未加修飾的0也是完全可以接受的。由于void *指針的特殊賦值屬性,比如:

#define NULL ((void *)0)

當(dāng)NULL定義為((void *)0)時,即NULL是可以賦值給任何類型指針的值,它的類型為void*,而不是整數(shù)0,因此初始化“FILE *fp = NULL;”是完全合法的。

而為了區(qū)分整數(shù)0和空指針0,當(dāng)需要其它類型的0的時候,即使可能工作,但也不能使用NULL,如果這樣處理其格式是錯誤的,這在非指針上下文中是不能工作的。特別地,不能在需要ASCII空字符(NUL)的地方使用NULL。如果確實需要,則可以自定義為:

#define NUL

由此可見,常數(shù)0是一個空指針常量,而NULL僅僅是它的一個別名。

3.空指針的用途

一般來說,未初始化是不能使用的非法指針,因為它完全有可能指向任何地方,從而導(dǎo)致程序無法判斷它為非法指針。因此,不管指針變量是全局的還是局部的、靜態(tài)的還是非靜態(tài)的,都應(yīng)該在聲明它的同時進(jìn)行初始化,要么賦予一個有效的地址,要么賦予NULL。

標(biāo)準(zhǔn)C規(guī)定,全局指針變量的默認(rèn)值為NULL,而對于局部指針變量則必須明確地指定其初值。因此,void通常用于指針變量的初始化,用來判斷一個指針的有效性。比如:

unsigned char *pucBuf=(void *)0;//定義pucBuf為unsigned char類型指針并初始化為空指針

如果后續(xù)的代碼忘記初始化指針而直接使用的話,則可能造成程序失敗。雖然空指針也是非法指針,但可以通過程序判斷并告訴程序員代碼可能有問題。也就是說,如果一開始就將指針初始化為空指針,則可避免程序異常。比如:

if(pucBuf==0){

return error;//如果pucBuf為空指針,則返回參數(shù)錯誤

}

由于void類型指針的不確定性,因此它可以指向任意類型的數(shù)據(jù),那么只要在使用時做一個簡單的強制類型轉(zhuǎn)換就可以了。比如:

unsignned char*pcData = NULL;//定義pcData為unsigned char類型指針

void*pvData;//定義pvData為void類型指針

pvData = pcData;//無需進(jìn)行強制類型轉(zhuǎn)換

pcData = (unsigned char*) pvData;//將pvData強制轉(zhuǎn)換為unsigned char類型指針

顯然不存在void類型的對象,也就是說,當(dāng)對象為空類型時,其大小為0字節(jié);當(dāng)對象未確定類型時,那么它的大小也是未確定的,因此不能聲明void類型變量。比如:

void a;//非法聲明

既然上述聲明是非法的,那么,也就不能將sizeof運算符用于void類型。也就意味著,編譯器不知道所指對象的大小,由于指針的算術(shù)運算總是基于所指對象的大小的,因此不允許對void指針進(jìn)行算術(shù)運算。

總之,在指針聲明中,void *表示通用指針的類型。如果void作為函數(shù)的返回類型,則表示不返回任何值。如果void位于參數(shù)列表中,則表示沒有參數(shù)。

4. 用無類型指針作為函數(shù)參數(shù)

由于C語言中最小長度的變量為char類型(包括unsigned char、signed char等),其sizeof(char)的結(jié)果為1,而其它任何變量的長度都是它的整數(shù)倍。比如,如果使用SDCC51編譯器,其sizeof(int)為2。因為通用swap函數(shù)函數(shù)不知道需要交換的變量的類型,所以需要一個參數(shù)給出相應(yīng)的指示。由于C語言的變量類型多種多樣,因此不可能為每一種變量類型編號,而且swap并不關(guān)心變量的真正類型,所以可以用變量的長度代替變量類型。通用swap函數(shù)的原型為:

void swap(void *pvData1, void *pvData2, int iDataSize)

將a,b兩個變量(變量類型必須一樣)的值交換的代碼如下:

swap(&a, &b, sizeof(a));

通用swap排序函數(shù)的參考代碼見程序清單1.1。

程序清單1.1通用swap排序函數(shù)

1void swap (void *pvData1, void *pvData2, int iDataSize)

2{

3unsigned char *pcData1 = NULL;

4unsigned char *pcData2 = NULL;

5unsigned charucTmp1;

6

7pcData1 = (unsigned char *)pvData1;

8pcData2 = (unsigned char *)pvData2;

9

10do {

11ucTmp1 = *pcData1;

12*pcData1= *pcData2;

13*pcData2= ucTmp1;

14pcData1++;

15pcData2++;

16} while (--iDataSize > 0);

17}



關(guān)鍵詞: 深入理解void類

評論


技術(shù)專區(qū)

關(guān)閉