Go指针禁止算术运算,仅支持取地址和解引用,确保内存安全;通过unsafe.Pointer可实现底层操作但不推荐;相比C,Go指针更安全、受限,提升程序稳定性。

Go语言中的指针设计比C语言更加安全和受限,目的是减少内存错误和提升程序稳定性。虽然两者都使用指针来操作内存地址,但在指针运算和使用方式上有显著区别。
Go不支持指针算术运算
在C语言中,可以对指针进行加减操作,例如 ptr++ 或 ptr + 5,这在遍历数组或操作连续内存时非常常见。Go明确禁止这类操作。
例如,以下C代码是合法的:
int arr[5] = {1,2,3,4,5};int *p = arr;
p++; // 指向下一个整数
但在Go中类似操作会编译失败:
立即学习“go语言免费学习笔记(深入)”;
var arr [5]intp := &arr[0]
p++ // 编译错误:invalid operation: p++ (non-numeric type *int)
Go不允许对指针做加减偏移,无法通过指针跳跃访问相邻内存。
Go指针只能取地址和解引用
Go中指针的基本操作仅限于:
- 取地址:&variable
- 解引用:*pointer
这些操作保证了指针只能访问它明确指向的变量,避免越界或误访问。比如:
p := &x
fmt.Println(*p) // 输出 10
你不能像C那样通过 *(p + 3) 去访问偏移位置的数据。
unsafe.Pointer 可实现有限的指针操作
如果确实需要底层内存操作,Go提供了 unsafe.Pointer,但它绕过了类型安全,需手动保证正确性。
通过 unsafe.Pointer 可以实现指针类型转换和有限的偏移计算,例如:
import "unsafe"arr := [5]int{1,2,3,4,5}
p := unsafe.Pointer(&arr[0])
offsetP := (*int)(unsafe.Pointer(uintptr(p) + 4*unsafe.Sizeof(arr[0])))
fmt.Println(*offsetP) // 访问第5个元素
这种方式接近C的指针运算,但必须导入 unsafe 包,且容易引发崩溃或内存错误,不推荐日常使用。
与C指针的核心区别总结
Go指针与C指针的主要差异体现在:
- C允许任意指针算术,Go禁止
- C指针可自由转换类型,Go类型严格
- Go默认指针操作是安全的,不会越界
- Go没有多重间接(如 **pp)的复杂层级滥用
- Go垃圾回收机制要求指针行为可追踪,限制了灵活性
这些限制让Go更适合现代安全编程,牺牲部分灵活性换取更高的可靠性和可维护性。
基本上就这些。Go的指针不是为了替代C的内存操控,而是提供一种受控的间接访问机制。不复杂但容易忽略。










