在golang中,可以利用unsafe包來實現指針轉換,實現內存地址在不同指針類型間的轉換,進而更靈活地操作內存。例如unsafe包下的Pointer()函數可以將任意變量的地址轉換成Pointer類型,也可以將Pointer類型轉換成任意的指針類型,Pointer類型是不同指針類型之間互轉的中間類型。
本教程操作環境:windows7系統、GO 1.18版本、Dell G3電腦。
在golang中,可以利用unsafe包來實現指針轉換。
golang的指針轉換
Golang 提供了 unsafe 包,讓我們能夠直接操作指定內存地址的內存。
unsafe包下,有定義type Pointer *ArbitraryType(任意類型指針),可繞過GO的類型限制,type ArbitraryType int
-
任何類型的指針值都可以轉換為Pointer。
-
Pointer可以轉換為任何類型的指針值。
-
uintptr可以轉換為Pointer。
-
Pointer可以轉換為uintptr。
通過 unsafe.Pointer() 函數,我們能夠獲取變量的內存地址表示,本質上這是個整數。可以將任意變量的地址轉換成 Pointer 類型,也可以將 Pointer 類型轉換成任意的指針類型,它是不同指針類型之間互轉的中間類型。
但 Pointer 不支持運算,如果要在內存地址上進行加減運算,需要將其轉為 uintptr 類型。
下面我們嘗試讀取切片地址,并通過內存操作遍歷其內容:
package main import "fmt" import "unsafe" func main() { // head = {address, 10, 10} // body = [1,2,3,4,5,6,7,8,9,10] var s = []int{1, 2, 3, 4, 5, 6, 7, 8, 9, 10} var address = (**[10]int)(unsafe.Pointer(&s)) var len = (*int)(unsafe.Pointer(uintptr(unsafe.Pointer(&s)) + uintptr(8))) var cap = (*int)(unsafe.Pointer(uintptr(unsafe.Pointer(&s)) + uintptr(16))) fmt.Println(address, *len, *cap) var body = **address for i := 0; i < 10; i++ { fmt.Printf("%d ", body[i]) } } ---------- 0xc000004460 10 10 1 2 3 4 5 6 7 8 9 10
上述代碼中:
-
unsafe.Pointer(&s) 獲取切片 s 底層表示的第一個位置的內存地址,也即底層數組的地址存放地址,
通過 (**[10]int)(unsafe.Pointer(&s)) 將其轉為 **[10]int 類型指針,又通過 **addrss 還原為數組;
-
unsafe.Pointer(uintptr(unsafe.Pointer(&s)) + uintptr(8)) 通過地址運算,獲得 length 的存放地址,
進而通過 (*int)(unsafe.Pointer(uintptr(unsafe.Pointer(&s)) + uintptr(8))) 將 length 內存轉為 int 指針
最后通過 *len 獲取切片長度;
對于 cap 的操作與 len 類似,不再贅述;
總之:
通過 unsafe,我們能夠實現內存地址在不同指針類型間的轉換,進而更靈活地操作內存;
本實驗也進一步驗證了切片的底層存儲結構;
unsafe 在不是必須的條件下應該少使用,直接操作內存畢竟是風險較大的;
【