首页 > 其他分享 >Go 中普通指针、unsafe.Pointer 与 uintptr 之间的关系和指针运算

Go 中普通指针、unsafe.Pointer 与 uintptr 之间的关系和指针运算

时间:2023-09-09 16:25:33浏览次数:39  
标签:people int uintptr unsafe height Pointer 指针

C 语言指针运算

指针运算就是对指针类型的变量做常规数学运算,例如加减操作,实现地址的偏移。指针运算在 C 语言中是原生支持的,可以直接在指针变量上做加减,例如:

#include <stdio.h>
 
const int MAX = 3;
 
int main ()
{
   int  var[] = {10, 100, 200};
   int  i, *ptr;
 
   /* 指针中的数组地址 */
   ptr = var;
   for ( i = 0; i < MAX; i++)
   {
 
      printf("存储地址:var[%d] = %p\n", i, ptr );
      printf("存储值:var[%d] = %d\n", i, *ptr );
 
      /* 直接对指针做++操作,指向下一个位置 */
      ptr++;
   }
   return 0;
}

结果

存储地址:var[0] = e4a298cc
存储值:var[0] = 10
存储地址:var[1] = e4a298d0
存储值:var[1] = 100
存储地址:var[2] = e4a298d4
存储值:var[2] = 200

C 语言指针运算犹如一把双刃剑,使用得当会起到事半功倍,有神之一手的效果,反之则会产生意想不到的 bug 而且很难排查。因为在做指针运算时是比较抽象的,具体偏移了多少之后指向到了哪里是非常不直观的,可能已经偏离了设想中的位置而没有发现,运行起来就会出现错误。

例如这段 C 代码,找出数组中最小的元素:

#include <stdio.h>

int findMin(int *arr, int length) {
    int min = *arr;
    for (int i = 0; i <= length; i++) {  // 注意这里是 i <= length,而不是 i < length
        printf("i=%d v=%d\n", i, *(arr+i));
        if (*(arr + i) < min) {
            min = *(arr + i);
        }
    }
    return min;
}

int main() {
    int arr[] = {1, 2, 3, 4, 5};
    int length = sizeof(arr) / sizeof(arr[0]);
    printf("Min value is: %d\n", findMin(arr, length));
    return 0;
}

数组中最小的是 1,可结果却是 0:

i=0 v=1
i=1 v=2
i=2 v=3
i=3 v=4
i=4 v=5
i=5 v=0
Min value is: 0

这是由于在 findMin 函数中循环条件是 i ≤ length ,超出数组大小多循环了一次,实际上数组已经越界,而 C 语言的数组实际上就是指针,C 运行时认为这是在指针运算,所以不会报错,导致数组访问到了其他内存地址,最终得到了一个错误结果。

事实上有很多病毒和外挂的原理就是利用指针来访问并修改程序运行时内存数据来达到目的。例如游戏外挂可能会搜索和修改内存中的特定值,以改变玩家的生命值、金钱或其他游戏属性。通过指针运算,外挂可以直接访问这些内存位置并对其进行修改。而病毒可能使用指针运算来插入其自己的代码到一个运行中的程序,或者篡改程序的正常控制流,以达到其恶意目的。

在 C 语言之后的很多语言多多少少都对指针做了限制,例如 PHP 中的引用就可以看做是指针的简化版,而 Java 甚至干脆移除了指针。

Go 指针运算

在 Go 中默认的普通指针也是指代的是一个内存地址,值类似 0x140000ac008,但 Go 的普通指针不支持指针运算的,例如对指针做加法会报错:

a := 10
var p *int = &a
p = p + 1

报错

invalid operation: p + 1 (mismatched types *int and untyped int)

但 Go 还是提供了一种直接操作指针的方式,就是 unsafe.Pointer 和 uintptr。

uintptr 是一个整型,可理解为是将内存地址转换成了一个整数,既然是一个整数,就可以对其做数值计算,实现指针地址的加减,也就是地址偏移,类似跟 C 语言中一样的效果。

而 unsafe.Pointer 是普通指针和 uintptr 之间的桥梁,通过 unsafe.Pointer 实现三者的相互转换。

*T <-> unsafe.Pointer <-> uintptr

先看看这三位都长什么样:

func main() {
	a := 10
	var b *int
	b = &a

	fmt.Printf("a is %T, a=%v\n", a, a)
	fmt.Printf("b is %T, b=%v\n", b, b)

	p := unsafe.Pointer(b)
	fmt.Printf("p is %T, p=%v\n", p, p)
	
	uptr := uintptr(p)
	fmt.Printf("uptr is %T, uptr=%v\n", uptr, uptr)
}

输出

a is int, a=10
b is *int, b=0x140000ae008
p is unsafe.Pointer, p=0x140000ae008
uptr is uintptr, uptr=1374390247432

举一个通过指针运算修改结构体的例子

type People struct {
	age    int32
	height int64
	name   string
}
people := &People{}
fmt.Println(people)

// 将 people 普通指针转成 unsafe.Pointer 再转为 uintptr
// 后面再加上 height 字段相对于结构体本身的偏移量,就得到了 height 的地址的 uintptr 值
// 再将 height 的 uintptr 值转成 unsafe.Pointer 赋值给 height 变量
// 所以现在 height 的类型是 unsafe.Pointer
height := unsafe.Pointer(uintptr(unsafe.Pointer(people)) + unsafe.Offsetof(people.height))
fmt.Printf("people addr is %v\n", unsafe.Pointer(people))
fmt.Printf("height is %T\n", height)
fmt.Printf("height addr is %v\n", height)

println("---")

// 使用类型转换,将 unsafe.Pointer 类型的 height 转换成 *int 指针
// 再通过最前面的 * 解引用,修改其值 身高2米26
*((*int)(height)) = 226
fmt.Println(people)

// 同样的操作可以修改年龄和名字
age := unsafe.Pointer(uintptr(unsafe.Pointer(people)) + unsafe.Offsetof(people.age))
*((*int)(age)) = 18

name := unsafe.Pointer(uintptr(unsafe.Pointer(people)) + unsafe.Offsetof(people.name))
*((*string)(name)) = "小明"

fmt.Println(people)

输出

people: &{0 0 }
people addr is 0x1400005e020
height is unsafe.Pointer
height addr is 0x1400005e028
---
people: &{0 226 }
people: &{18 226 }
people: &{18 226 小明}

再看一个操作,通过指针转换,将一个字节切片转成浮点数组:

package main

import (
	"fmt"
	"unsafe"
)

func main() {
	// 假设我们有一个字节切片,并且我们知道它是由浮点数表示的
	byteSlice := []byte{0, 0, 0, 0, 0, 0, 240, 63} // 1.0 的 IEEE-754 表示

	// 使用 unsafe 把字节切片转换为浮点数切片
	floatSlice := (*[1]float64)(unsafe.Pointer(&byteSlice[0]))

	fmt.Println(floatSlice)
}

输出

&[1]

这个过程不需要 Go 的类型检查,绕过了很多流程,相对来说性能会更高。

所以大体上通过 unsafe.Pointer 的指针运算会应用在如下几个方面:

  1. 性能优化: 当性能是关键因素时,unsafe 可以用来避免一些开销。例如,通过直接操作内存,可以避免切片或数组的额外分配和复制。
  2. C 语言交互: 当使用 cgo 与 C 语言库交互时,unsafe 包通常用于转换类型和指针。
  3. 自定义序列化/反序列化: 在自定义的序列化或反序列化逻辑中,unsafe 可以用于直接访问结构的内存布局,可以提高性能。
  4. 实现非标准的数据结构: 有时,特定的问题需要非标准的数据结构。unsafe 允许你直接操作内存,可以用来实现一些 Go 的标准库中没有的数据结构。
  5. 反射: 与反射结合时,unsafe 可以用于访问结构体的私有字段。

本文由mdnice多平台发布

标签:people,int,uintptr,unsafe,height,Pointer,指针
From: https://www.cnblogs.com/caipi/p/17689638.html

相关文章

  • 空指针产生的条件 null对象调用属性
    null对象调用属性 e.gpublicclassStudent{Integerage;Stringname;Stringaddress;Useruser;}Studentstudent=newStudent();12:student.getUser().getName()Exceptioninthread"main"java.lang.NullPointerException atpatter......
  • 画圆盘的指针
    效果图 privateBitmapGetPointerImg(BitmappointerImg,floatangle,intw,inth,intcenterX,intcenterY){//创建新的位图作为旋转后的图片BitmaprotatedImage=newBitmap(w,h);//创建一个Graphics......
  • 通过指针变量存取一维数组元素
    通过指针变量存取一维数组元素下面展示一下。#include<stdio.h>intmain(){ inta[10],*p; for(p=a;p<a+10;p++) { scanf("%d",p); }for(p=a;p<a+10;p++) { printf("%d",*p); } printf("\n"); return0;}测试输入......
  • 单词搜索 II(字典树、数组)、合并两个有序数组(数组、双指针)、验证回文串(双指针、字
    单词搜索II(字典树、数组)给定一个mxn二维字符网格board****和一个单词(字符串)列表words,找出所有同时在二维网格和字典中出现的单词。单词必须按照字母顺序,通过相邻的单元格内的字母构成,其中“相邻”单元格是那些水平相邻或垂直相邻的单元格。同一个单元格内的字母在一......
  • 学习使用双指针(leetcode)
    一、K和数对的最大数目(JAVA)给你一个整数数组nums和一个整数k。每一步操作中,你需要从数组中选出和为k的两个整数,并将它们移出数组。返回你可以对数组执行的最大操作数。示例1:输入:nums=[1,2,3,4],k=5输出:2解释:开始时nums=[1,2,3,4]:-移出1和4,......
  • 双指针法删除数组里面的值
    你一个数组 nums 和一个值 val,你需要 原地 移除所有数值等于 val 的元素,并返回移除后数组的新长度。不要使用额外的数组空间,你必须仅使用 O(1) 额外空间并 原地 修改输入数组。元素的顺序可以改变。你不需要考虑数组中超出新长度后面的元素。 说明:为什么返回数......
  • 代码随想录刷题记录——双指针篇
    27.移除元素题目链接快慢指针,最终返回index值为移除元素后的数组末尾元素下标+1.#include<vector>usingnamespacestd;classSolution{public:intremoveElement(vector<int>&nums,intval){//快慢指针intnums_length=nums.size();......
  • 【C++】C++ 引用详解 ⑦ ( 指针的引用 )
    文章目录一、二级指针可实现的效果二、指针的引用1、指针的引用等同于二级指针(重点概念)2、引用本质-函数间接赋值简化版本3、代码示例-指针的引用一、二级指针可实现的效果指针的引用效果等同于二级指针,因此这里先介绍二级指针;使用二级指针作为参数,可......
  • 【C++】C++ 引用详解 ④ ( 函数返回 静态变量 / 全局变量 的 引用 / 指针 )
    文章目录一、函数返回静态变量/全局变量的引用/指针1、函数返回局部变量引用或指针无意义2、函数返回静态变量/全局变量的引用或指针3、代码示例-函数返回静态变量/全局变量的引用或指针一、函数返回静态变量/全局变量的引用/指针1、函数返回局部变量引用或指针......
  • 【C语言进阶】指针数组 —— 数组指针
    (文章目录)......