【c语言】指针小结

一、指针是什么?

可以通过运算符&来取得变量实际保存的 起始地址

(这个地址是虚拟地址,并不是真正物理内存上的地址。)

数据类型 *标识符 = &变量;

int *pa = &a; int *pa = NULL;

(NULL表示地址为0的内存空间,这个地址是无法使用的,读写该地址会报错)

pa 中存储的就是变量 a 的地址,也叫做指向 a 的指针。

int *a,b; 等价于 int *a;int b;而不等价于 int *a; int *b;

💡 变量的初始化!int *pa;

声明指针变量之后,编译器会为指针变量本身分配一个内存空间,但是这个内存空间里面的值是随机的,也就是说, 未初始化的指针变量指向的地址是随机的。 这时一定不能去读写指针变量指向的地址,因为那个地址是随机地址,很可能会导致严重后果。

  • 错误案例:*p = 1; 错误的,p指向的内存是随机的,不可以赋值
  • 正确案例: p = &a; *p = 1 正确的,此时p指向a的地址,可以为a赋值

💡 pa中存储的是a变量的内存地址,那如何通过地址去获取a的值呢?
这个操作就叫做解引用 比如*pa就能获得a的值。

二、指针的运算

指针仅在同一段内存中比较才有意义。指针本质上就是一个无符号整数,代表了内存地址。它可以进行运算,但是规则并不是整数运算的规则。

  1. 指针与整数值的加减运算

    指针与整数值的运算,表示指针的移动。

    c 复制代码
    int* i;
    i = (int *)0x0034;
    i = i + 1; // 0x0038

    i是一个指针,指向内存地址0x0034, 不要误认为i + 1等于0x0035,正确答案是0x0038(也就是sizeof(int)的大小 )。

    原因是i + 1表示指针向内存地址的高位移动一个单位,而一个单位的short类型占据两个字节的宽度,所以相当于向高位移动两个字节。

    同样的,i - 1得到的结果是0x0032。指针移动的单位,与指针指向的数据类型有关。数据类型占据多少个字节,每单位就移动多少个字节。

  2. 指针与指针的加法运算

    两个指针进行加法是非法的。

  3. 指针与指针的减法

    相同类型的指针允许进行减法运算,返回它们之间的距离,即相隔多少个数据单位。

    高位地址减去低位地址,返回的是正值;低位地址减去高位地址,返回的是负值。

    这时,减法返回的值属于ptrdiff_t类型,这是一个带符号的整数类型别名,具体类型根据系统不同而不同。这个类型的原型定义在头文件stddef.h里面。

    c 复制代码
    int* i1;
    int* i2;
    
    i1 = (int *)0x0034;
    i2 = (int *)0x0038; // i2 与 i1之间相差4个字节,就是一个int的距离
    
    ptrdiff_t dist = i2 - i1;
    printf("%td\n", dist);  // 1
  4. 指针与指针的比较运算

    指针之间的比较运算,比较的是各自的内存地址哪一个更大,返回值是整数1(true)或0(false)。

三、多级指针

这里的 03 号格子就叫二级指针,05 号格子就叫指针,而 07 号就是我们平常用的变量。

不管几级指针有两个最核心的东西:

  • 指针本身也是一个变量,需要内存去存储,指针也有自己的地址
  • 指针内存存储的是它所指向变量的地址

这就是我为什么多级指针是逻辑上的概念,实际上一块内存要么放实际内容,要么放其它变量地址,就这么简单。

怎么去解读int **a这种表达呢?

int **a 可以把它分为两部分看,即int**a,后面 *a 中的*表示 a 是一个指针变量,前面的 int* 表示指针变量a

只能存放 int* 型变量的地址。

相关推荐
小宇子2B3 天前
一个 7 行的 C 函数,是怎么一路变成 CPU 上的电信号
c·汇编语言
handler013 天前
【算法】并查集(普通/扩展/带权)模板与例题
数据结构·c++·笔记·算法·c·图论·查并集
蓝宝石的傻话4 天前
给MibeeNvr 0.6调试的Esp32和树莓派的三个摄像头项目的技术更新细节
c
handler016 天前
【C++11 】Lambda 表达式、std::function 与 std::bind 解析
c++·c·c++11·bind·解耦·function·lamda
handler0111 天前
【C++】二叉搜索树详解及其模拟实现(代码)
开发语言·c++·算法·c··二叉搜索树·搜索树
爱学习的程序媛12 天前
C 语言全景指南:从底层原理到工业级实战
c++·c#·c
dozenyaoyida13 天前
RISC-V嵌入式开发:彻底解决“undefined reference to isatty“错误全攻略
经验分享·c·cmake·嵌入式开发·isatty·没有定义问题
Shadow(⊙o⊙)14 天前
模拟实现:glibc_1.0-文件操作函数fopen fclose fwrite fflush实现。
开发语言·c++·学习·c
liulilittle16 天前
TCP UCP:基于卡尔曼滤波的BBR增强型拥塞控制算法
linux·网络·c++·tcp/ip·算法·c·通讯
weixin_4217252617 天前
C语言、C++与C#深度研究报告:从底层控制到现代企业级开发的演进
c语言·c++·c·内存管理·编译模型