深入理解C語言的void*
一、void* 的類型任意性
void* 是一種通用指針類型。它可以指向任意類型的數(shù)據(jù)。例如,它可以指向一個整數(shù)(int)、一個浮點(diǎn)數(shù)(float)、一個字符(char)或者一個結(jié)構(gòu)體等。在C語言中,當(dāng)你使用void*
指針時(shí),你不需要在編譯時(shí)指定它將指向的數(shù)據(jù)類型。這使得void*
在一些需要通用指針的場景下非常有用,比如在內(nèi)存分配函數(shù)malloc中返回的就是void*
類型的指針。因?yàn)閙alloc函數(shù)不知道用戶將要分配的內(nèi)存用于存儲哪種類型的數(shù)據(jù),所以它返回一個void*
指針,用戶可以根據(jù)自己的需要將其轉(zhuǎn)換為特定類型的指針。
例如:
void* ptr = malloc(10 * sizeof(int)); int* intPtr = (int*)ptr; // 將void*指針轉(zhuǎn)換為int*指針
在這個例子中,void*
指針 ptr 可以指向分配的內(nèi)存區(qū)域,然后通過類型轉(zhuǎn)換將其轉(zhuǎn)換為int*指針,用于存儲整數(shù)數(shù)組。
二、編譯器對 void* 的類型檢查
編譯時(shí)不做類型檢查(針對void*本身)編譯器在編譯時(shí)不會對void*
指針本身進(jìn)行類型檢查。因?yàn)?nbsp;void*
表示“未知類型”的指針,編譯器無法知道它實(shí)際指向的數(shù)據(jù)類型。所以,當(dāng)你對void指針進(jìn)行操作(如賦值等)時(shí),編譯器不會檢查其指向的數(shù)據(jù)類型是否正確。例如,你可以將一個指向整數(shù)的指針賦值給void指針,也可以將一個指向字符的指針賦值給void*指針,編譯器都不會報(bào)錯。
int a = 10; char b = 'k'; void* vp1 = &a; void* vp2 = &b;
在這個例子中,vp1和vp2都是void*指針,分別指向了不同類型的變量a和b,編譯器不會對這種賦值操作進(jìn)行類型檢查。
三、需要顯式類型轉(zhuǎn)換
當(dāng)你想要使用 void*
指針訪問其中的某個值時(shí),通常需要先將其轉(zhuǎn)換為特定類型的指針,然后通過轉(zhuǎn)換后的指針來訪問值。在轉(zhuǎn)換時(shí),你需要明確指定目標(biāo)類型,編譯器會對轉(zhuǎn)換后的指針類型進(jìn)行檢查。例如,如果你想通過 void*
指針訪問一個整數(shù)的值,你需要先將其轉(zhuǎn)換為 int*
指針。
例如:
void* vp = malloc(sizeof(int)); *(int*)vp = 20; // 先將void*轉(zhuǎn)換為int*,然后通過int*指針賦值
在這個例子中,vp是一個void*
指針,指向分配的內(nèi)存。在給這塊內(nèi)存賦值之前,需要先將其轉(zhuǎn)換為int*
指針。如果轉(zhuǎn)換的目標(biāo)類型和實(shí)際存儲的數(shù)據(jù)類型不匹配,可能會導(dǎo)致運(yùn)行時(shí)錯誤。 比如,如果這塊內(nèi)存實(shí)際上存儲的是一個浮點(diǎn)數(shù),而你將其轉(zhuǎn)換為int*指針并訪問,可能會得到錯誤的結(jié)果或者引發(fā)程序異常。編譯器在轉(zhuǎn)換時(shí)會檢查語法是否正確(如是否有合適的類型轉(zhuǎn)換操作),但對于類型轉(zhuǎn)換的正確性(即是否符合程序的實(shí)際邏輯)主要依賴于程序員的正確使用。
占用的字節(jié)
一、32位系統(tǒng)
在32位系統(tǒng)中,void*
指針通常占據(jù)4個字節(jié)。這是因?yàn)?2位系統(tǒng)中的內(nèi)存地址空間是2的32次方(即4GB),用4個字節(jié)(32位)就可以表示一個內(nèi)存地址。例如,在一個32位的Windows系統(tǒng)或者32位的Linux系統(tǒng)上,無論是void指針,還是其他類型的指針(如int、char*等),它們都占據(jù)4個字節(jié)。這4個字節(jié)存儲的是一個內(nèi)存地址,這個地址可以指向進(jìn)程地址空間內(nèi)的任意位置。
二、64位系統(tǒng)
在64位系統(tǒng)中,void*
指針通常占據(jù)8個字節(jié)。64位系統(tǒng)有更大的內(nèi)存地址空間,理論上可以達(dá)到2的64次方字節(jié)。因此,需要用8個字節(jié)(64位)來表示一個完整的內(nèi)存地址。在64位系統(tǒng)上,無論是 void*
指針,還是其他類型的指針,它們的大小都是8個字節(jié)。這使得64位系統(tǒng)能夠訪問更大的內(nèi)存空間,支持更大的數(shù)據(jù)處理和更復(fù)雜的程序運(yùn)行。
四、總結(jié)
通過我們上面的介紹, 我們發(fā)現(xiàn) void*
在 C 語言中是經(jīng)常使用的, 它主要有下面這幾個方案:
- 第一個方案就是作為結(jié)構(gòu)體的字段, 這樣我們可以去表示對應(yīng)的范型字段
- 函數(shù)的參數(shù), 或者返回值, 但是我覺得這種最好少用, 因?yàn)閷?dǎo)致對應(yīng)的接口不夠明確
- 各種內(nèi)存相關(guān)的函數(shù), 這其實(shí)就是一個很奇妙的東西, 因?yàn)樵诓僮飨到y(tǒng)級別, 我們并不知道這塊內(nèi)存到底是什么類型的變量
約定: 當(dāng)我們在使用 void*
的時(shí)候, 我們最好弄清楚當(dāng)前這個指針指向的信息, 可以通過明確的變量名來完成
到此這篇關(guān)于深入理解C語言的void*的文章就介紹到這了,更多相關(guān)C語言 void*內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Dijkstra算法最短路徑的C++實(shí)現(xiàn)與輸出路徑
今天小編就為大家分享一篇關(guān)于Dijkstra算法最短路徑的C++實(shí)現(xiàn)與輸出路徑,小編覺得內(nèi)容挺不錯的,現(xiàn)在分享給大家,具有很好的參考價(jià)值,需要的朋友一起跟隨小編來看看吧2019-02-02C語言實(shí)現(xiàn)帶頭雙向循環(huán)鏈表
本文主要介紹了C語言實(shí)現(xiàn)帶頭雙向循環(huán)鏈表,文中通過示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-03-03C語言程序設(shè)計(jì)50例(經(jīng)典收藏)
本篇文章是對C語言程序設(shè)計(jì)的50個小案例進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下2013-06-06