Go不支持指针算术,仅允许取地址、解引用和比较;需通过unsafe.Pointer与uintptr实现内存偏移,如遍历数组或字节操作,但存在安全风险,应限于系统底层场景使用。

Go语言中指针运算受到严格限制,不像C/C++那样可以直接对指针进行加减等算术操作。Go的设计理念是安全和简洁,因此不支持传统的指针算术(如p++或p + n),但可以通过unsafe包实现底层内存操作,从而间接完成类似指针运算的功能。
1. Go原生不支持指针算术
在Go中,普通指针只能取地址、解引用和比较,不能进行加减乘除:
var arr [3]int = [3]int{10, 20, 30}
p := &arr[0] // 指向第一个元素
// 下面的操作是非法的:
// p++ // 编译错误
// p + 1 // 编译错误
2. 使用 unsafe.Pointer 实现指针偏移
如果确实需要指针运算(例如操作字节序列、实现底层数据结构),可以使用unsafe包中的unsafe.Pointer和uintptr。
示例:遍历int数组的每个字节
立即学习“go语言免费学习笔记(深入)”;
package mainimport ( "fmt" "unsafe" )
func main() { arr := [3]int{10, 20, 30} p := unsafe.Pointer(&arr[0]) // 获取首元素地址 size := unsafe.Sizeof(arr[0]) // 每个int的大小
for i := 0; i < len(arr); i++ { // 计算第i个元素的地址:p + i * size elemAddr := uintptr(p) + uintptr(i)*size val := *(*int)(unsafe.Pointer(elemAddr)) // 转回*int并解引用 fmt.Println(val) }}
3. 字节级指针操作(常见于底层编码)
处理二进制协议或内存映射时,常需逐字节访问内存:
data := [4]byte{1, 2, 3, 4}
p := unsafe.Pointer(&data[0])
for i := 0; i < len(data); i++ {
bytePtr := (byte)(unsafe.Pointer(uintptr(p) + uintptr(i)))
fmt.Printf("Byte %d: %d\n", i, bytePtr)
}
4. 注意事项与安全建议
使用unsafe绕过类型系统存在风险,需谨慎:
- 仅用于必须操作内存的场景:如与C交互、实现运行时库、序列化等。
- 避免在业务逻辑中使用:可读性差且易出错。
- 注意对齐问题:某些架构要求内存访问地址对齐。
- GC可能受影响:手动管理内存地址可能干扰垃圾回收。
基本上就这些。Go有意限制指针运算以提升安全性,大多数情况下应使用切片、索引等高级抽象代替。只有在性能关键或系统编程场景下才考虑unsafe方式,且应充分测试。










