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