這篇文章主要介紹“C語(yǔ)言中void*的用法”,在日常操作中,相信很多人在C語(yǔ)言中void*的用法問(wèn)題上存在疑惑,小編查閱了各式資料,整理出簡(jiǎn)單好用的操作方法,希望對(duì)大家解答”C語(yǔ)言中void*的用法”的疑惑有所幫助!接下來(lái),請(qǐng)跟著小編一起來(lái)學(xué)習(xí)吧!
創(chuàng)新互聯(lián)-專(zhuān)業(yè)網(wǎng)站定制、快速模板網(wǎng)站建設(shè)、高性?xún)r(jià)比福清網(wǎng)站開(kāi)發(fā)、企業(yè)建站全套包干低至880元,成熟完善的模板庫(kù),直接使用。一站式福清網(wǎng)站制作公司更省心,省錢(qián),快速模板網(wǎng)站建設(shè)找我們,業(yè)務(wù)覆蓋福清地區(qū)。費(fèi)用合理售后完善,10多年實(shí)體公司更值得信賴(lài)。void在英文中作為名詞的解釋為"空虛、空間、空隙",而在C語(yǔ)言中,void被翻譯為"無(wú)類(lèi)型",相應(yīng)的void*為"無(wú)類(lèi)型指針"。
void似乎只有"注釋"和限制程序的作用,當(dāng)然,這里的"注釋"不是為我們?nèi)颂峁┳⑨專(zhuān)菫榫幾g器提供一種所謂的注釋。
1.對(duì)函數(shù)返回的限定,這種情況我們比較常見(jiàn)。
2.對(duì)函數(shù)參數(shù)的限定,這種情況也是比較常見(jiàn)的。
當(dāng)函數(shù)不需要返回值值時(shí),必須使用void限定,這就是我們所說(shuō)的第一種情況。例如:voidfunc(inta,char*b)。
當(dāng)函數(shù)不允許接受參數(shù)時(shí),必須使用void限定,這就是我們所說(shuō)的第二種情況。例如:intfunc(void)。
1.void指針可以指向任意類(lèi)型的數(shù)據(jù),就是說(shuō)可以用任意類(lèi)型的指針對(duì)void指針對(duì)void指針賦值。例如:
int*a;
void*p;
p=a;
如果要將void指針p賦給其他類(lèi)型的指針,則需要強(qiáng)制類(lèi)型轉(zhuǎn)換,就本例而言:a=(int*)p。在內(nèi)存的分配中我們可以見(jiàn)到void指針使用:內(nèi)存分配函數(shù)malloc函數(shù)返回的指針就是void*型,用戶(hù)在使用這個(gè)指針的時(shí)候,要進(jìn)行強(qiáng)制類(lèi)型轉(zhuǎn)換,也就是顯式說(shuō)明該指針指向的內(nèi)存中是存放的什么類(lèi)型的數(shù)據(jù)(int*)malloc(1024)表示強(qiáng)制規(guī)定malloc返回的void*指針指向的內(nèi)存中存放的是一個(gè)個(gè)的int型數(shù)據(jù)。
2.在ANSIC標(biāo)準(zhǔn)中,不允許對(duì)void指針進(jìn)行一些算術(shù)運(yùn)算如p++或p+=1等,因?yàn)榧热籿oid是無(wú)類(lèi)型,那么每次算術(shù)運(yùn)算我們就不知道該操作幾個(gè)字節(jié),例如char型操作sizeof(char)字節(jié),而int則要操作sizeof(int)字節(jié)。而在GNU中則允許,因?yàn)樵谀J(rèn)情況下,GNU認(rèn)為void*和char*一樣,既然是確定的,當(dāng)然可以進(jìn)行一些算術(shù)操作,在這里sizeof(*p)==sizeof(char)。
void幾乎只有"注釋"和限制程序的作用,因?yàn)閺膩?lái)沒(méi)有人會(huì)定義一個(gè)void變量,讓我們?cè)囍鴣?lái)定義:
voida;
這行語(yǔ)句編譯時(shí)會(huì)出錯(cuò),提示"illegaluseoftype'void'"。即使voida的編譯不會(huì)出錯(cuò),它也沒(méi)有任何實(shí)際意義。
眾所周知,如果指針p1和p2的類(lèi)型相同,那么我們可以直接在p1和p2間互相賦值;如果p1和p2指向不同的數(shù)據(jù)類(lèi)型,則必須使用強(qiáng)制類(lèi)型轉(zhuǎn)換運(yùn)算符把賦值運(yùn)算符右邊的指針類(lèi)型轉(zhuǎn)換為左邊指針的類(lèi)型。
float*p1;
int*p2;
p1=p2;
//其中p1=p2語(yǔ)句會(huì)編譯出錯(cuò),
//提示“'=':cannotconvertfrom'int*'to'float*'”,必須改為:
p1=(float*)p2;
而void*則不同,任何類(lèi)型的指針都可以直接賦值給它,無(wú)需進(jìn)行強(qiáng)制類(lèi)型轉(zhuǎn)換。
void*p1;
int*p2;
p1=p2;
但這并不意味著,void*也可以無(wú)需強(qiáng)制類(lèi)型轉(zhuǎn)換地賦給其它類(lèi)型的指針。因?yàn)?無(wú)類(lèi)型"可以包容"有類(lèi)型",而"有類(lèi)型"則不能包容"無(wú)類(lèi)型"。
按照ANSI(AmericanNationalStandardsInstitute)標(biāo)準(zhǔn),不能對(duì)void指針進(jìn)行算法操作,即下列操作都是不合法的:
void*pvoid;
pvoid++;//ANSI:錯(cuò)誤
pvoid+=1;//ANSI:錯(cuò)誤
//ANSI標(biāo)準(zhǔn)之所以這樣認(rèn)定,是因?yàn)樗鼒?jiān)持:進(jìn)行算法操作的指針必須是確定知道其指向數(shù)據(jù)類(lèi)型大小的。
//例如:
int*pint;
pint++;//ANSI:正確pint++的結(jié)果是使其增大sizeof(int)。
但是GNU則不這么認(rèn)定,它指定void*的算法操作與char*一致。因此下列語(yǔ)句在GNU編譯器中皆正確:
pvoid++;//GNU:正確
pvoid+=1;//GNU:正確
pvoid++的執(zhí)行結(jié)果是其增大了1。
在實(shí)際的程序設(shè)計(jì)中,為迎合ANSI標(biāo)準(zhǔn),并提高程序的可移植性,我們可以這樣編寫(xiě)實(shí)現(xiàn)同樣功能的代碼:
void*pvoid;
((char*)pvoid)++;//ANSI:錯(cuò)誤;GNU:正確
(char*)pvoid+=1;//ANSI:錯(cuò)誤;GNU:正確
GNU和ANSI還有一些區(qū)別,總體而言,GNU較ANSI更"開(kāi)放",提供了對(duì)更多語(yǔ)法的支持。但是我們?cè)谡鎸?shí)設(shè)計(jì)時(shí),還是應(yīng)該盡可能地迎合ANSI標(biāo)準(zhǔn)。如果函數(shù)的參數(shù)可以是任意類(lèi)型指針,那么應(yīng)聲明其參數(shù)為void*。
注:void指針可以任意類(lèi)型的數(shù)據(jù),可以在程序中給我們帶來(lái)一些好處,函數(shù)中形為指針類(lèi)型時(shí),我們可以將其定義為void指針,這樣函數(shù)就可以接受任意類(lèi)型的指針。如:
void*memcpy(void*dest,constvoid*src,size_tlen);
void*memset(void*buffer,intc,size_tnum);
這樣,任何類(lèi)型的指針都可以傳入memcpy和memset中,這也真實(shí)地體現(xiàn)了內(nèi)存操作函數(shù)的意義,因?yàn)樗僮鞯膶?duì)象僅僅是一片內(nèi)存,而不論這片內(nèi)存是什么類(lèi)型(參見(jiàn)C語(yǔ)言實(shí)現(xiàn)泛型編程)。如果memcpy和memset的參數(shù)類(lèi)型不是void*,而是char*,那才叫真的奇怪了!這樣的memcpy和memset明顯不是一個(gè)"純粹的,脫離低級(jí)趣味的"函數(shù)!void的出現(xiàn)只是為了一種抽象的需要,如果你正確地理解了面向?qū)ο笾?抽象基類(lèi)"的概念,也很容易理解void數(shù)據(jù)類(lèi)型。正如不能給抽象基類(lèi)定義一個(gè)實(shí)例,我們也不能定義一個(gè)void(讓我們類(lèi)比的稱(chēng)void為"抽象數(shù)據(jù)類(lèi)型")變量。
到此,關(guān)于“C語(yǔ)言中void*的用法”的學(xué)習(xí)就結(jié)束了,希望能夠解決大家的疑惑。理論與實(shí)踐的搭配能更好的幫助大家學(xué)習(xí),快去試試吧!若想繼續(xù)學(xué)習(xí)更多相關(guān)知識(shí),請(qǐng)繼續(xù)關(guān)注創(chuàng)新互聯(lián)網(wǎng)站,小編會(huì)繼續(xù)努力為大家?guī)?lái)更多實(shí)用的文章!