指针是C语言的灵魂也是许多初学者望而生畏的拦路虎。本篇博客将从内存和地址的基本概念出发带你一步步理解指针的本质、指针变量的使用、指针类型的意义以及如何安全地使用指针最后通过实际案例strlen模拟、swap交换展示指针的应用场景。 目录内存与地址——指针的本质指针变量与解引用操作指针变量的大小指针类型的意义const修饰指针野指针与规避方法assert断言——调试利器传值调用与传址调用总结1. 内存与地址——指针的本质1.1 内存单元的编址计算机中的内存被划分为一个个内存单元每个单元大小为1个字节Byte每个字节都有一个唯一的编号这个编号就是地址。类比生活宿舍楼有100个房间房间有门牌号101、102...→ 快速找到人内存有大量存储单元每个单元有地址0x00、0x01...→ CPU快速找到数据核心理解text内存单元的编号 地址 指针1.2 硬件层面的编址CPU通过地址总线与内存通信32位机器32根地址线 → 可表示 2³² 个地址 → 指针大小4字节64位机器64根地址线 → 可表示 2⁶⁴ 个地址 → 指针大小8字节 指针就是地址地址就是内存单元的编号。2. 指针变量与解引用操作2.1 取地址操作符创建变量时系统会在内存中分配空间每个变量都有其地址。c#include stdio.h int main() { int a 10; printf(a 的地址%p\n, a); // %p 专门打印地址 return 0; }注意a取出的是变量a所占4个字节中地址较小的那个字节的地址。2.2 指针变量指针变量是专门用来存放地址的变量。cint a 10; int* pa a; // pa 是一个指针变量指向 int 类型的变量 a类型拆解*表示pa是一个指针变量int表示pa指向的是int类型的对象cchar ch w; char* pc ch; // 字符指针指向字符变量2.3 解引用操作符*通过地址找到其指向的对象这个过程叫解引用。c#include stdio.h int main() { int a 10; int* pa a; printf(a %d\n, *pa); // *pa 等价于 a输出 10 *pa 100; // 通过指针修改 a 的值 printf(a %d\n, a); // 输出 100 return 0; }✨*pa就是a的替身通过指针可以间接操作变量。3. 指针变量的大小指针变量的大小取决于平台与指针类型无关。平台地址总线指针大小32位x8632根4字节64位x6464根8字节c#include stdio.h int main() { printf(%zu\n, sizeof(char*)); // 4 (x86) 或 8 (x64) printf(%zu\n, sizeof(int*)); // 4 (x86) 或 8 (x64) printf(%zu\n, sizeof(double*)); // 4 (x86) 或 8 (x64) return 0; }结论所有指针变量在同一个平台下大小相同与指向的类型无关。4. 指针类型的意义既然指针大小都一样为什么还要有不同类型的指针指针类型决定了两个关键行为4.1 解引用时的权限一次操作几个字节c#include stdio.h int main() { int n 0x11223344; // 十六进制占4个字节 int* pi n; char* pc (char*)n; *pi 0; // 将 n 的 4 个字节全部置为 0 *pc 0; // 只将 n 的第一个字节置为 0 return 0; }指针类型解引用权限char*访问1个字节int*访问4个字节double*访问8个字节4.2 指针 ± 整数时的步长跳过几个字节c#include stdio.h int main() { int n 0x11223344; int* pi n; char* pc (char*)n; printf(pi %p\n, pi); printf(pi1 %p\n, pi 1); // 跳 4 个字节 printf(pc %p\n, pc); printf(pc1 %p\n, pc 1); // 跳 1 个字节 return 0; }指针类型1 跳过的字节数char*1字节short*2字节int*4字节double*8字节4.3 void* 指针泛型指针void*可以接收任意类型的地址但不能直接解引用或进行指针运算通常用于函数参数实现泛型效果如qsort。c#include stdio.h int main() { int a 10; void* p a; // 可以接收任意类型地址 // *p 20; // ❌ 错误void* 不能直接解引用 return 0; }5. const修饰指针5.1 const修饰变量const修饰的变量不能被直接修改但可以通过指针绕过限制。cconst int n 10; // n 20; // ❌ 错误n 是只读的 int* p n; *p 20; // ✅ 可以通过指针修改但破坏了 const 的本意5.2 const修饰指针的三种情况cint n 10, m 20; // 情况1const 在 * 左边 → 指向的内容不可修改指针本身可改 const int* p1 n; // *p1 20; // ❌ 错误不能通过 p1 修改 n p1 m; // ✅ 可以修改指针指向 // 情况2const 在 * 右边 → 指针本身不可修改指向的内容可改 int* const p2 n; *p2 20; // ✅ 可以修改指向的内容 // p2 m; // ❌ 错误不能修改指针本身 // 情况3两边都有 const → 都不能修改 const int* const p3 n; // *p3 20; // ❌ 错误 // p3 m; // ❌ 错误写法指向内容指针本身const int* p不可修改可修改int* const p可修改不可修改const int* const p不可修改不可修改6. 野指针与规避方法6.1 什么是野指针野指针是指指针指向的位置是不可知的随机的、不正确的、没有明确限制的。6.2 野指针的成因① 指针未初始化cint* p; // 局部指针未初始化值为随机值 *p 20; // 危险p 是野指针② 指针越界访问cint arr[10] {0}; int* p arr; for (int i 0; i 10; i) { *(p i) i; // i10 时越界p 变成野指针 }③ 返回局部变量的地址cint* test() { int n 100; return n; // 返回局部变量的地址——函数结束即销毁 } int main() { int* p test(); printf(%d\n, *p); // 危险访问已释放的空间 return 0; }6.3 规避野指针的方法① 初始化指针cint n 10; int* p1 n; // 明确指向 int* p2 NULL; // 不知道指向哪里时赋值为 NULL② 小心越界确保指针操作不超出所指向的内存范围。③ 不使用时置为 NULLcint* p n; // ... 使用 p p NULL; // 不再使用时及时置 NULL④ 使用前检查cif (p ! NULL) { *p 100; // 安全使用 }⑤ 不要返回局部变量的地址7. assert断言——调试利器assert.h中的assert()宏用于在运行时检查条件若不满足则终止程序并报错。c#include assert.h int my_strlen(const char* str) { assert(str ! NULL); // 如果 str 为 NULL程序终止 int count 0; while (*str) { count; str; } return count; }优点自动显示文件名和行号可通过定义NDEBUG宏禁用Release 版本c#define NDEBUG // 禁用 assert #include assert.h✅ 建议在 Debug 版本中使用assert帮助排查问题Release 版本自动优化。8. 传值调用与传址调用8.1 传值调用失败案例c#include stdio.h void Swap1(int x, int y) { int tmp x; x y; y tmp; } int main() { int a 10, b 20; printf(交换前a%d, b%d\n, a, b); Swap1(a, b); printf(交换后a%d, b%d\n, a, b); // 未交换 return 0; }原因形参是实参的临时拷贝修改形参不影响实参。8.2 传址调用成功案例c#include stdio.h void Swap2(int* px, int* py) { int tmp *px; // 通过指针访问实参 *px *py; *py tmp; } int main() { int a 10, b 20; printf(交换前a%d, b%d\n, a, b); Swap2(a, b); // 传递地址 printf(交换后a%d, b%d\n, a, b); // 成功交换 return 0; }输出text交换前a10, b20 交换后a20, b108.3 应用模拟实现 strlenc#include stdio.h #include assert.h // 方法1计数器推荐 size_t my_strlen1(const char* str) { assert(str ! NULL); size_t count 0; while (*str) { count; str; } return count; } // 方法2指针 - 指针更高效 size_t my_strlen2(const char* str) { assert(str ! NULL); const char* start str; while (*str) { str; } return str - start; // 两个指针相减得到元素个数 } // 方法3递归不推荐栈开销大 size_t my_strlen3(const char* str) { assert(str ! NULL); if (*str \0) return 0; return 1 my_strlen3(str 1); } int main() { char arr[] hello world; printf(len1 %zu\n, my_strlen1(arr)); printf(len2 %zu\n, my_strlen2(arr)); printf(len3 %zu\n, my_strlen3(arr)); return 0; }9. 总结概念要点指针的本质指针就是地址地址就是内存单元的编号指针变量存放地址的变量用类型*定义取地址获取变量的地址解引用*通过地址访问指向的对象指针大小32位→4字节64位→8字节与类型无关指针类型意义决定解引用权限和 ±1 步长const修饰指针左边→内容不可改右边→指针不可改野指针未初始化、越界、返回局部变量地址规避野指针初始化、置NULL、使用前检查assert调试时检查条件失败则终止传址调用通过指针修改变量实现真正的外界影响核心心法指针就是地址指针变量就是存放地址的盒子。取地址*取内容两者是逆运算。指针类型决定怎么解读和走多远。 所有代码均在 VS2022 / GCC 下测试通过。如有疑问欢迎在评论区交流下一篇预告C语言指针进阶——数组与指针的深度解析
C语言指针入门——内存地址与变量操作(附优化代码)
发布时间:2026/7/3 1:43:07
指针是C语言的灵魂也是许多初学者望而生畏的拦路虎。本篇博客将从内存和地址的基本概念出发带你一步步理解指针的本质、指针变量的使用、指针类型的意义以及如何安全地使用指针最后通过实际案例strlen模拟、swap交换展示指针的应用场景。 目录内存与地址——指针的本质指针变量与解引用操作指针变量的大小指针类型的意义const修饰指针野指针与规避方法assert断言——调试利器传值调用与传址调用总结1. 内存与地址——指针的本质1.1 内存单元的编址计算机中的内存被划分为一个个内存单元每个单元大小为1个字节Byte每个字节都有一个唯一的编号这个编号就是地址。类比生活宿舍楼有100个房间房间有门牌号101、102...→ 快速找到人内存有大量存储单元每个单元有地址0x00、0x01...→ CPU快速找到数据核心理解text内存单元的编号 地址 指针1.2 硬件层面的编址CPU通过地址总线与内存通信32位机器32根地址线 → 可表示 2³² 个地址 → 指针大小4字节64位机器64根地址线 → 可表示 2⁶⁴ 个地址 → 指针大小8字节 指针就是地址地址就是内存单元的编号。2. 指针变量与解引用操作2.1 取地址操作符创建变量时系统会在内存中分配空间每个变量都有其地址。c#include stdio.h int main() { int a 10; printf(a 的地址%p\n, a); // %p 专门打印地址 return 0; }注意a取出的是变量a所占4个字节中地址较小的那个字节的地址。2.2 指针变量指针变量是专门用来存放地址的变量。cint a 10; int* pa a; // pa 是一个指针变量指向 int 类型的变量 a类型拆解*表示pa是一个指针变量int表示pa指向的是int类型的对象cchar ch w; char* pc ch; // 字符指针指向字符变量2.3 解引用操作符*通过地址找到其指向的对象这个过程叫解引用。c#include stdio.h int main() { int a 10; int* pa a; printf(a %d\n, *pa); // *pa 等价于 a输出 10 *pa 100; // 通过指针修改 a 的值 printf(a %d\n, a); // 输出 100 return 0; }✨*pa就是a的替身通过指针可以间接操作变量。3. 指针变量的大小指针变量的大小取决于平台与指针类型无关。平台地址总线指针大小32位x8632根4字节64位x6464根8字节c#include stdio.h int main() { printf(%zu\n, sizeof(char*)); // 4 (x86) 或 8 (x64) printf(%zu\n, sizeof(int*)); // 4 (x86) 或 8 (x64) printf(%zu\n, sizeof(double*)); // 4 (x86) 或 8 (x64) return 0; }结论所有指针变量在同一个平台下大小相同与指向的类型无关。4. 指针类型的意义既然指针大小都一样为什么还要有不同类型的指针指针类型决定了两个关键行为4.1 解引用时的权限一次操作几个字节c#include stdio.h int main() { int n 0x11223344; // 十六进制占4个字节 int* pi n; char* pc (char*)n; *pi 0; // 将 n 的 4 个字节全部置为 0 *pc 0; // 只将 n 的第一个字节置为 0 return 0; }指针类型解引用权限char*访问1个字节int*访问4个字节double*访问8个字节4.2 指针 ± 整数时的步长跳过几个字节c#include stdio.h int main() { int n 0x11223344; int* pi n; char* pc (char*)n; printf(pi %p\n, pi); printf(pi1 %p\n, pi 1); // 跳 4 个字节 printf(pc %p\n, pc); printf(pc1 %p\n, pc 1); // 跳 1 个字节 return 0; }指针类型1 跳过的字节数char*1字节short*2字节int*4字节double*8字节4.3 void* 指针泛型指针void*可以接收任意类型的地址但不能直接解引用或进行指针运算通常用于函数参数实现泛型效果如qsort。c#include stdio.h int main() { int a 10; void* p a; // 可以接收任意类型地址 // *p 20; // ❌ 错误void* 不能直接解引用 return 0; }5. const修饰指针5.1 const修饰变量const修饰的变量不能被直接修改但可以通过指针绕过限制。cconst int n 10; // n 20; // ❌ 错误n 是只读的 int* p n; *p 20; // ✅ 可以通过指针修改但破坏了 const 的本意5.2 const修饰指针的三种情况cint n 10, m 20; // 情况1const 在 * 左边 → 指向的内容不可修改指针本身可改 const int* p1 n; // *p1 20; // ❌ 错误不能通过 p1 修改 n p1 m; // ✅ 可以修改指针指向 // 情况2const 在 * 右边 → 指针本身不可修改指向的内容可改 int* const p2 n; *p2 20; // ✅ 可以修改指向的内容 // p2 m; // ❌ 错误不能修改指针本身 // 情况3两边都有 const → 都不能修改 const int* const p3 n; // *p3 20; // ❌ 错误 // p3 m; // ❌ 错误写法指向内容指针本身const int* p不可修改可修改int* const p可修改不可修改const int* const p不可修改不可修改6. 野指针与规避方法6.1 什么是野指针野指针是指指针指向的位置是不可知的随机的、不正确的、没有明确限制的。6.2 野指针的成因① 指针未初始化cint* p; // 局部指针未初始化值为随机值 *p 20; // 危险p 是野指针② 指针越界访问cint arr[10] {0}; int* p arr; for (int i 0; i 10; i) { *(p i) i; // i10 时越界p 变成野指针 }③ 返回局部变量的地址cint* test() { int n 100; return n; // 返回局部变量的地址——函数结束即销毁 } int main() { int* p test(); printf(%d\n, *p); // 危险访问已释放的空间 return 0; }6.3 规避野指针的方法① 初始化指针cint n 10; int* p1 n; // 明确指向 int* p2 NULL; // 不知道指向哪里时赋值为 NULL② 小心越界确保指针操作不超出所指向的内存范围。③ 不使用时置为 NULLcint* p n; // ... 使用 p p NULL; // 不再使用时及时置 NULL④ 使用前检查cif (p ! NULL) { *p 100; // 安全使用 }⑤ 不要返回局部变量的地址7. assert断言——调试利器assert.h中的assert()宏用于在运行时检查条件若不满足则终止程序并报错。c#include assert.h int my_strlen(const char* str) { assert(str ! NULL); // 如果 str 为 NULL程序终止 int count 0; while (*str) { count; str; } return count; }优点自动显示文件名和行号可通过定义NDEBUG宏禁用Release 版本c#define NDEBUG // 禁用 assert #include assert.h✅ 建议在 Debug 版本中使用assert帮助排查问题Release 版本自动优化。8. 传值调用与传址调用8.1 传值调用失败案例c#include stdio.h void Swap1(int x, int y) { int tmp x; x y; y tmp; } int main() { int a 10, b 20; printf(交换前a%d, b%d\n, a, b); Swap1(a, b); printf(交换后a%d, b%d\n, a, b); // 未交换 return 0; }原因形参是实参的临时拷贝修改形参不影响实参。8.2 传址调用成功案例c#include stdio.h void Swap2(int* px, int* py) { int tmp *px; // 通过指针访问实参 *px *py; *py tmp; } int main() { int a 10, b 20; printf(交换前a%d, b%d\n, a, b); Swap2(a, b); // 传递地址 printf(交换后a%d, b%d\n, a, b); // 成功交换 return 0; }输出text交换前a10, b20 交换后a20, b108.3 应用模拟实现 strlenc#include stdio.h #include assert.h // 方法1计数器推荐 size_t my_strlen1(const char* str) { assert(str ! NULL); size_t count 0; while (*str) { count; str; } return count; } // 方法2指针 - 指针更高效 size_t my_strlen2(const char* str) { assert(str ! NULL); const char* start str; while (*str) { str; } return str - start; // 两个指针相减得到元素个数 } // 方法3递归不推荐栈开销大 size_t my_strlen3(const char* str) { assert(str ! NULL); if (*str \0) return 0; return 1 my_strlen3(str 1); } int main() { char arr[] hello world; printf(len1 %zu\n, my_strlen1(arr)); printf(len2 %zu\n, my_strlen2(arr)); printf(len3 %zu\n, my_strlen3(arr)); return 0; }9. 总结概念要点指针的本质指针就是地址地址就是内存单元的编号指针变量存放地址的变量用类型*定义取地址获取变量的地址解引用*通过地址访问指向的对象指针大小32位→4字节64位→8字节与类型无关指针类型意义决定解引用权限和 ±1 步长const修饰指针左边→内容不可改右边→指针不可改野指针未初始化、越界、返回局部变量地址规避野指针初始化、置NULL、使用前检查assert调试时检查条件失败则终止传址调用通过指针修改变量实现真正的外界影响核心心法指针就是地址指针变量就是存放地址的盒子。取地址*取内容两者是逆运算。指针类型决定怎么解读和走多远。 所有代码均在 VS2022 / GCC 下测试通过。如有疑问欢迎在评论区交流下一篇预告C语言指针进阶——数组与指针的深度解析