C语言 —— 指针
目录
1. 指针是什么?
2. 指针和指针类型的关系
2.1 指针的解引用
2.2 指针+-整数
3. 野指针
3.1 野指针成因
1. 指针未初始化
2. 指针越界访问
3. 指针指向的空间释放
3.2 如何规避野指针
4. 指针运算
4.1 指针+-整数
4.2 指针-指针
指针-指针的使用
4.3 指针的关系运算
5. 指针和数组
6. 二级指针
7. 指针数组
1. 指针是什么?
指针是什么?
指针理解的2个要点:
1. 指针是内存中一个最小单元的编号,也就是地址
2. 平时口语中说的指针,通常指的是指针变量,是用来存放内存地址的变量
总结:指针就是地址,口语中说的指针通常指的是指针变量。
我们在初识C语言中知道, 内存是一块很大的空间, 这么大的内存空间如何管理呢?
其实我们是把内存切割成一个个的内存单元, 一个内存单元大小是1byte
.
对于要点的理解:
- 第一点
- 第二点
当写下以下代码时:
int a = 10;//a是整型变量,占用4个字节的内存空间
意味着要在内存中找4个空间存放a.
指针=指针变量, 指针变量中存放地址, 通过这个地址可以找到一个内存单元.
总结:
- 指针变量是用来存放地址的,地址是唯一标示一块地址空间的。
- 指针的大小在32位平台是4个字节,在64位平台是8个字节。
注意下面所使用的环境是x86的环境, 也就是32位的环境. (如果改成x64就是64位的环境)
我们来看一下指针的大小.
int main()
{char* pc = NULL;short* ps = NULL;int* pi = NULL;double* pd = NULL;//sizeof 返回的值的类型是无符号整型 unsigned intprintf("%zu\n", sizeof(pc));printf("%zu\n", sizeof(ps));printf("%zu\n", sizeof(pi));printf("%zu\n", sizeof(pd));return 0;
}
执行结果:
可以看到, 不同类型指针的大小都是4个字节.
既然这样, 不管是创建什么类型的指针, 在32位平台是4个字节,在64位平台是8个字节, 那么为什么还要区分不同类型的指针出来? 而不是直接使用一个统一的指针类型?
C语言并没有使用统一的指针类型, 说明每一种类型的指针都是有意义的, 并不多余.
接下来我们来探讨指针类型的意义.
2. 指针和指针类型的关系
2.1 指针的解引用
我们来看这段代码的效果:
int a = 0x11223344;
int* pa = &a;
*pa = 0;
通过调试可以看到:
如果把代码改为:
int a = 0x11223344;
char* pc = (char*)&a;//a原本应该是int*, 而非要赋给char*, 此时进行强转
*pc = 0;
pc是否有能力存下a的地址? 有, pc也是指针变量, 在x86环境下的大小也为4个字节.
我们来调试查看:
所以可以看到, a的地址是可以存到pc中.
但是, 使用的时候会有差异, 即*pc
.
注: 上图为重新调试的截图, 所以会和前面的图有地址差异.
可以看到, *pc
所修改的内容只有一个字节, 因为pc
的类型是char*
指针, 所以解引用时只访问了一个字节.
而前面使用int*
的pc
是访问了4个字节.
综上, 我们有以下结论.
指针类型决定了指针在被解引用的时候访问几个字节
如果是int*的指针,解引用访问4个字节
如果是char*的指针,解引用访问1个字节
(还可以推广到其他类型)
这便是指针类型的第一个意义.再来看第二个意义.
2.2 指针+-整数
int main()
{int a = 0x11223344;int* pa = &a;char* pc = (char*)&a;printf("pa = %p\n", pa);printf("pa+1 = %p\n", pa+1);printf("pc = %p\n", pc);printf("pc+1 = %p\n", pc+1);return 0;
}
执行结果:
所以我们得出以下结论:
指针的类型决定了指针+-1操作的时候,跳过几个字节, 它决定了指针的步长.
3. 野指针
概念: 野指针就是指针指向的位置是不可知的(随机的、不正确的、没有明确限制的)
3.1 野指针成因
1. 指针未初始化
int main()
{int* p;//p没有初始化,就意味着没有明确的指向//一个局部变量不初始化的话,放的是随机值:0xcccccccc*p = 10;//非法访问内存了,这里的p就是野指针return 0;
}
2. 指针越界访问
int main()
{int arr[10] = { 0 };int* p = arr;//&arr[0]int i = 0;for (i = 0; i <= 10; i++){//当指针指向的范围超出数组arr的范围时,p就是野指针*p = i;p++;}return 0;
}
3. 指针指向的空间释放
int* test()
{int a = 10;return &a;
}int main()
{int*p = test();printf("haha\n");printf("abcdef\n");if (p != NULL){printf("%d\n", *p);}return 0;
}
3.2 如何规避野指针
1. 明确的给指针初始化
2. 小心指针越界
3. 指针指向空间释放即使置NULL
4. 避免返回局部变量的地址
5. 指针使用之前检查有效性
4. 指针运算
- 指针+- 整数
- 指针-指针
- 指针的关系运算
4.1 指针+-整数
#define N_VALUES 5
float values[N_VALUES];
float *vp;
//指针+-整数;指针的关系运算
for (vp = &values[0]; vp < &values[N_VALUES];)
{*vp++ = 0;
}
4.2 指针-指针
指针-指针得到的结果的绝对值是指针和指针之间元素的个数.
int arr[10] = { 0 };
printf("%d\n", &arr[0] - &arr[9]);//9
注意: 不是所有的指针都能相减, 指向同一块空间的2个指针才能相减.
下面写法是错误的:
int arr[10] = { 0 };
char ch[5] = {0};
printf("%d\n", &ch[0] - &arr[5]);//err
指针-指针的使用
使用指针-指针的方式求字符串长度.
显然的, 只需要知道第一个字符的地址和\0的地址就可以知道字符串长度, 两个指针相减即可.
#include <stdio.h>
#include <string.h>int my_strlen(char* str)
{char* start = str;while (*str != '\0'){str++;}return (str - start);
}int main()
{int len = my_strlen("abcdef");printf("%d\n", len);return 0;
}
4.3 指针的关系运算
#define N_VALUES 5
float values[N_VALUES];
float *vp;for(vp = &values[N_VALUES]; vp > &values[0];)
{*--vp = 0;
}
代码简化:
for(vp = &values[N_VALUES-1]; vp >= &values[0];vp--)
{*vp = 0;
}
实际在绝大部分的编译器上是可以顺利完成任务的,然而我们还是应该避免这样写,因为标准并不保证它可行。
标准规定:
允许指向数组元素的指针与指向数组最后一个元素后面的那个内存位置的指针比较,但是不允许与指向第一个元素之前的那个内存位置的指针进行比较。
5. 指针和数组
我们知道:
数组:一组相同类型元素的集合
指针变量:是一个变量,存放的是地址
数组名表示的是数组首元素的地址(两种情况除外)
int main()
{int arr[10] = {0};//arr 是首元素的地址//&arr[0]int* p = arr;//我们可以通过指针来访问数组, 这就是两者的联系int sz = sizeof(arr) / sizeof(arr[0]);int i = 0;for (i = 0; i < sz; i++){printf("%d ", *(p + i));}for (i = 0; i < sz; i++){printf("%p ----- %p\n", &arr[i], p + i);}return 0;
}
6. 二级指针
#include <stdio.h>int main()
{int a = 10;int* pa = &a;//pa是一个指针变量,一级指针变量//*pa = 20;printf("%d\n", a);//pa是变量, 它在内存中也会有自己的内存空间, 那么就能&pa.int** ppa = &pa;//ppa是一个二级指针变量**ppa = 20;return 0;
}
一级指针变量就是如果通过pa找a的话, 找一次就可以了.
二级指针变量是用来存放级指针变量的地址的.
7. 指针数组
指针数组是数组, 是存放指针的数组。
我们知道, 创建比较多的相同类型的变量可以使用数组, 那么假设对于这些变量都有对应的指针去存放它们的地址, 是否也能用类似的方式, 去创建一个类型都为指针的数组? 显然是可以的, 也就是说C语言是有这种语法存在的.
int a = 10;
int b = 20;
int c = 30;int* pa = &a;
int* pb = &b;
int* pc = &c;
int arr[10];int* parr[10] = {&a, &b, &c};
int i = 0;
for (i = 0; i < 3; i++)//0 1 2
{printf("%d ", *(parr[i]));
}
int arr1[4] = { 1,2,3,4 };
int arr2[4] = { 2,3,4,5 };
int arr3[4] = { 3,4,5,6 };int* parr[3] = {arr1, arr2, arr3};
int i = 0;
for (i = 0; i < 3; i++)
{int j = 0;for (j = 0; j < 4; j++){printf("%d ", parr[i][j]);}printf("\n");
}
相关文章:

C语言 —— 指针
目录 1. 指针是什么? 2. 指针和指针类型的关系 2.1 指针的解引用 2.2 指针-整数 3. 野指针 3.1 野指针成因 1. 指针未初始化 2. 指针越界访问 3. 指针指向的空间释放 3.2 如何规避野指针 4. 指针运算 4.1 指针-整数 4.2 指针-指针 指针-指针的使用 4.3 指针的关系运…...

淘宝店铺所有商品数据接口,淘宝整店所有商品数据接口,淘宝店铺商品接口,淘宝API接口
淘宝店铺所有商品数据接口可以通过淘宝开放平台获取。以下是具体步骤: 在开放平台注册成为开发者并创建一个应用,获取到所需的 App Key 和 App Secret 等信息。使用获取到的 App Key 和 App Secret 进行签名和认证,获取 Access Token。调用开…...

【Redis】Java客户端使用zset命令
zadd/zrange zcard zrem zscore zrank...

记录一个@Transaction注解引发的bug
记录一个Transactional(readOnly true)注解引发的bug 一、问题代码和报错 1-1 问题代码模拟 引发这个问题的三大要素分别是: 事务注解任意数据库操作数据库操作后执行耗时业务(耗时超过数据库配置的超时时间) //1.这里是问题的核心之一…...

解决docker使用pandarallel报错OSError: [Errno 28] No space left on device
参考:https://github.com/nalepae/pandarallel/issues/127 在使用pandarallel报错OSError: [Errno 28] No space left on device,根据上述issue发现确实默认使用的MEMORY_FS_ROOT为 /dev/shm,而在docker环境下这个目录大小只有64M࿰…...
Javascript自定义页面复制事件
Javascript自定义页面复制事件 – WhiteNights Site 2023年10月13日 文章访问量:90 标签:Javascript 监听copy事件以达到自定义页面复制功能的效果。 写者注 需要注意的是,浏览器的部分拓展插件(如迅雷)会导致本文…...

Nginx:反向代理(示意图+配置)
示意图: 反向代理 反向代理(Reverse Proxy)是代理服务器的一种,它代表服务器接收客户端的请求,并将这些请求转发到适当的服务器。当请求在后端服务器完成之后,反向代理搜集请求的响应并将其传输给客户端。…...

macbook笔记本电脑内存怎么清理才能干净流畅?
假如你还在为“你的系统内存不足”的提示所困扰,或者你的Mac电脑突然运行缓慢和卡顿,那么你一般需要认真了解一下macbook内存怎么清理了? MacBook是功能强大的电脑,这点毫无疑问,但是它仍旧会随着时间推移变得运行缓慢。值得庆幸…...
spark 与 mapreduce 对比
Spark 为什么比 MapReduce 快总结 首先澄清几个误区: 1)两者都是基于内存计算的,任何计算框架都肯定是基于内存的,所以说网上所说的 Spark 是基于内存计算所以快,显然是错误的。 2)DAG 计算模型减少的是磁…...

kafka 相关概念
1 kafka 生产者 kafka 用push的方式把消息推送到topic 每个topic下可以有多个分区, 可以用hash 也可以用轮询的方式指定分区 每个分区内部是可以保证顺序的,但是整体无法保证顺序,除非设置成一个topic只有一个分区。 kafka这种多分区的设置 带…...
Ubuntu下vscode配置OpenCV以及Libtorch
opencv安装 sudo apt-get updatesudo apt-get install libopencv-dev 该方式安装的版本可能比较旧。 测试代码 #include <opencv2/opencv.hpp>#include <iostream>int main() {cv::Mat image cv::imread("t.png");cv::imshow("Image", ima…...
关于共识算法Raft的常见误解
关于共识算法Raft的常见误解 Raft 共识算法最终一致性与线性一致性日志的覆盖与删除Remove节点时需要skip 总结参考文档 Raft 共识算法 最近翻了翻Raft相关的资料,同时也总结了日常工作的一些积累,就当做Raft技术笔记吧。 由于工作的关系,Ra…...

Python学习基础笔记七十——模块和库1
模块和库: 一个python代码文件就实现了功能。功能比较单一。 在企业中,项目开发的文件,可能有成百上千个。 不同的代码文件,实现了不同的功能模块,就像一块块积木一样。这些功能文件整合起来,实现一个完…...

SystemVerilog Assertions应用指南 第一章(1.28章节 内建的系统函数)
SVA提供了几个内建的函数来检查一些最常用的设计条件。 $onehot(expression)—检验表达式满足“one-hot”,换句话说,就是在任意给定的时钟沿,表达式只有一位为高。 $onehot0( expression)—检验表达式满足“ zero one-hot”,换句话说,就是在任意给定…...
正则表达式(自用)
正则表达式 符号概述 分类符号用法示例元字符^以 ***开头$以 ***结尾d匹配数字s匹配任意的空白符.匹配除换行符以外的任意字符w匹配字母或数字或下划线或汉字\转义重复限定符*次数,至少一次至少1次?0次 或者 1次{n}{n,}{n,m}重复n次;n活更多次&#x…...
大厂真题:【模拟】OPPO2023秋招提前批-小欧数组求和
题目描述与示例 题目描述 小欧拿到了一个数组,她有q次操作,每次操作修改一个元素。小欧希望每次修改后得到当前数组所有元素之和。你能帮帮她吗? 输入描述 第一行输入两个正整数n和q,代表数组的大小和操作次数。 第二行输入n…...
Python括号匹配问题
给定一个只包含小写字母的字符串,判断该字符串中的括号是否闭合,如果每个左括号都有对应的右括号,并且括号的嵌套顺序正确,那么括号就能正确闭合。 否则,括号不能正确闭合,字符串中括号仅限于 "("…...

微信小程序备案内容常见问题汇总
一、备案时间点 自2023年09月01日起,新的微信小程序,必须备案后才能上架; 在2024年03月31日前,所有小程序都必须完成备案; 于2024年04月01日起,对未备案小程序进行清退处理。 微信小程序备案系统已于9月4日上线。 二、备案流程 [找备案入口]–[填主体信息]–[填小程…...

无人机新手防炸飞行技巧
不要在室内飞行,容易撞墙。起飞前设置好避障和返航模式。使用模拟器熟练掌握操控。选择开阔环境目视起飞。使用低速档平稳飞行。合理使用避障功能,不要盲目依赖。使用九宫格避障法。留意电量,及时返航。极低电量时放弃强行返航。飞行后及时为电池充电保养。...
webrtc opus 音频编码支持SILK和CELT模式
SILK CELT是指将SILK编解码器和CELT编解码器结合在一起的混合音频编码方案。 SILK(Super-wideband audio coding)是一种低延迟的音频编解码器,用于实时的语音通信。它提供高质量的音频传输,并且适用于各种比特率和带宽条件。SILK…...

Python:操作 Excel 折叠
💖亲爱的技术爱好者们,热烈欢迎来到 Kant2048 的博客!我是 Thomas Kant,很开心能在CSDN上与你们相遇~💖 本博客的精华专栏: 【自动化测试】 【测试经验】 【人工智能】 【Python】 Python 操作 Excel 系列 读取单元格数据按行写入设置行高和列宽自动调整行高和列宽水平…...
Java如何权衡是使用无序的数组还是有序的数组
在 Java 中,选择有序数组还是无序数组取决于具体场景的性能需求与操作特点。以下是关键权衡因素及决策指南: ⚖️ 核心权衡维度 维度有序数组无序数组查询性能二分查找 O(log n) ✅线性扫描 O(n) ❌插入/删除需移位维护顺序 O(n) ❌直接操作尾部 O(1) ✅内存开销与无序数组相…...
在rocky linux 9.5上在线安装 docker
前面是指南,后面是日志 sudo dnf config-manager --add-repo https://download.docker.com/linux/centos/docker-ce.repo sudo dnf install docker-ce docker-ce-cli containerd.io -y docker version sudo systemctl start docker sudo systemctl status docker …...
AI编程--插件对比分析:CodeRider、GitHub Copilot及其他
AI编程插件对比分析:CodeRider、GitHub Copilot及其他 随着人工智能技术的快速发展,AI编程插件已成为提升开发者生产力的重要工具。CodeRider和GitHub Copilot作为市场上的领先者,分别以其独特的特性和生态系统吸引了大量开发者。本文将从功…...

学校时钟系统,标准考场时钟系统,AI亮相2025高考,赛思时钟系统为教育公平筑起“精准防线”
2025年#高考 将在近日拉开帷幕,#AI 监考一度冲上热搜。当AI深度融入高考,#时间同步 不再是辅助功能,而是决定AI监考系统成败的“生命线”。 AI亮相2025高考,40种异常行为0.5秒精准识别 2025年高考即将拉开帷幕,江西、…...
AGain DB和倍数增益的关系
我在设置一款索尼CMOS芯片时,Again增益0db变化为6DB,画面的变化只有2倍DN的增益,比如10变为20。 这与dB和线性增益的关系以及传感器处理流程有关。以下是具体原因分析: 1. dB与线性增益的换算关系 6dB对应的理论线性增益应为&…...

逻辑回归暴力训练预测金融欺诈
简述 「使用逻辑回归暴力预测金融欺诈,并不断增加特征维度持续测试」的做法,体现了一种逐步建模与迭代验证的实验思路,在金融欺诈检测中非常有价值,本文作为一篇回顾性记录了早年间公司给某行做反欺诈预测用到的技术和思路。百度…...
作为测试我们应该关注redis哪些方面
1、功能测试 数据结构操作:验证字符串、列表、哈希、集合和有序的基本操作是否正确 持久化:测试aof和aof持久化机制,确保数据在开启后正确恢复。 事务:检查事务的原子性和回滚机制。 发布订阅:确保消息正确传递。 2、性…...
深度剖析 DeepSeek 开源模型部署与应用:策略、权衡与未来走向
在人工智能技术呈指数级发展的当下,大模型已然成为推动各行业变革的核心驱动力。DeepSeek 开源模型以其卓越的性能和灵活的开源特性,吸引了众多企业与开发者的目光。如何高效且合理地部署与运用 DeepSeek 模型,成为释放其巨大潜力的关键所在&…...

C++_哈希表
本篇文章是对C学习的哈希表部分的学习分享 相信一定会对你有所帮助~ 那咱们废话不多说,直接开始吧! 一、基础概念 1. 哈希核心思想: 哈希函数的作用:通过此函数建立一个Key与存储位置之间的映射关系。理想目标:实现…...