05-指针的汇编

711 阅读2分钟

Xcode编译器,真机ARM64环境下讨论

1. 指针的基础知识

  • 1.1 指针的宽度&指向数据的宽度

    • 指针变量的宽度
      ARM64下,指针变量的宽度为 8 字节
    printf("%lu\n", sizeof(int *));
    printf("%lu\n", sizeof(char *));
    
    • 指针指向数据的宽度
    int a = 0
    int *pa = &a; // 指向的数据为int类型,int的宽度为4字节
    
    char b = '1';
    char *pb = &b; // 指向的数据为char类型,char的宽度为1字节
    

    在指针变量的运算的时候,是根据指针指向数据的宽度来决定结果的

  • 1.2 指针的运算

    指针的运算只有加法和减法
    指针运算的目的是移动指针的位置,取出对应的数据

    示例

    int *pa = (int *)0x100;
    printf("%d", pa+2);  // 打印pa指针指向的地址为0x108
    
    char *pb = (char *)0x100;
    print("%d", pb+2);  // 0x102
    
    int **pc = (int **)0x100;
    printf("%d", pc+2); // 0x110, 一个int*占8字节,
    
    

    指针运算的特点:它的运算依赖指向数据类型的宽度!
    (数据类型 *) pointer = 地址值x;
    pointer + a = x + a * sizeof(数据类型)
    减法同理

    示例2(了解)

    int *a = (int *)100;
    int *b = (int *)200;
    
    int x = a - b;
    printf("%d", x); // -25:-100/4
    
    int x2 = a - 199;
    printf("%d", x2); // 100-199*4 = -696
    

    一般很少利用指针进行这样的计算,主要是用于类似 pa+2 这样的偏移多少去取第几个数据

    示例3: 指针的比较(了解)

    int *a = (int *)100;
    int *b = (int *)200;
    if (a>b) {
        printf("a>b");
    } else {
        printf("a<b");
    }
    
    // print:  a<b , 地址值的比较
    
  • 1.3 指针与数组

    int arr[5] = {1,2,3,4,5};
    int *pa = arr;
    for (int i = 0; i < 5; i++) {
        printf("%d--%d", arr[i], *(pa+i));
    }
    
    // int *pa == arr == &arr[0]
    // arr编译器认定为不允许更改其所指向的地址值,所以arr++会报错
    

2 指针的汇编代码

示例1

void fun2(void){
    int b = 10;
    int *a;
    a = &b;
    printf("hello");
    return;
}

指针汇编
栈内存分析

示例2

void func3(void){
    int *p1;
    int a = *p1;
    int b = *(p1+2);
}

汇编分析

示例3

void func4(void){
    char **p;
    char c = **p;
}