《C和指针》笔记24: 指针和间接访问

news2025/1/22 15:08:14

本文主要讲指针和间接访问,标题对应《C和指针对应的章节》,引用的地方是自己写的一些注释、理解和总结。

指针、间接访问和左值

先回顾一下左值和右值

左值代表着一个位置。右值代表着一个值。赋值等号左边是个左值,赋值等号右边是一个右值。

在使用右值的地方也可以使用左值;但在需要左值的地方不能使用右值。
a=12;是可以的
a=b;也是可以的
12=a;是不可以的

涉及指针的表达式能不能作为左值?如果能,又是哪些呢?间接访问操作符*所需要的操作数是个右值,但这个操作符所产生的结果是个左值(参考这个文章中的《C和指针》笔记21:表达式求值)

int a;
int *d = &a;
表达式左值指定位置
aa
dd
*da

指针变量可以作为左值,并不是因为它们是指针,而是因为它们是变量。对指针变量进行间接访问表示我们应该访问指针所指向的位置。间接访问指定了一个特定的内存位置,这样我们可以把间接访问表达式的结果作为左值使用

指针、间接访问和变量

如果你自以为已经精通了指针,请看一下这个表达式,看看你是否明白它的意思。

*&a = 25;

答案:把值25赋值给变量a。首先,&操作符产生变量a的地址,它是一个指针常量(注意,使用这个指针常量并不需要知道它的实际值)。接着,*操作符访问其操作数所表示的地址。在这个表达式中,操作数是a的地址,所以值25就存储于a中。

指针表达式

上面的式子可能你不过瘾,而且你在这篇文章已经了解了左值和右值的含义,来看看下面的其他指针的更加复杂的实例吧!希望看完以后可以对如何阅读和编写这类复杂的代码有所启发。

char ch = 'a';
char *cp = &ch;

在这里插入图片描述
图中还显示了ch后面的那个内存位置,因为我们所求值的有些表达式将访问它(尽管是在错误情况下才会对它进行访问)。由于我们并不知道它的初始值,所以用一个问号来代替。

  1. ch

当它作为右值使用,表达式的值是a

在这里插入图片描述
那个粗椭圆提示变量ch的值就是表达式的值。

如果当它作为左值使用,表达式的值是ch的地址。

在这里插入图片描述
该位置用粗方框标记,它是这个内存的地址而不是该地址所包含的值,所以它的图示方式有所不同,它的值并未显示,因为它并不重要。事实上,这个值将被某个新值所取代。

  1. &ch

在这里插入图片描述
作为右值,这个表达式的值是变量ch的地址。注意这个值同变量cp中
所存储的值一样,但这个表达式并未提及cp,所以这个结果值并不是因为它而产生的。这样,图中椭圆并不画于cp的箭头周围。

为什么这个表达式不是一个合法的左值?优先级表格显示&操作符的结果是个右值,它不能当作左值使用。但是为什么呢?答案很简单,当表达式&ch进行求值时,它的结果应该存储于计算机的什么地方呢?它肯定会位于某个地方,但你无法知道它位于何处。这个表达式并未标识任何机器内存的特定位置,所以它不是一个合法的左值

  1. cp

在这里插入图片描述
它的右值如图所示就是cp的值。它的左值就是cp所处的内存位置。由于这个表达式并不进行间接访问操作,所以不必依箭头所示进行间接访问。

  1. &cp

在这里插入图片描述

这个例子与&ch类似,不过我们这次所取的是指针变量的地址。这个结果的类型是指向字符的指针的指针。同样,这个值的存储位置并未清晰定义,所以这个表达式不是一个合法的左值。

  1. *cp

在这里插入图片描述

现在我们加入了间接访问操作,所以它的结果应该不会令人惊奇。但
接下来的几个表达式就比较有意思。

  1. *cp+1

在这里插入图片描述
这个图涉及的东西更多,所以让我们一步一步来分析它。这里有两个操作符。*的优先级高于+,所以首先执行间接访问操作(如图中cp到ch的实线箭头所示),我们可以得到它的值(如虚线椭圆所示)。我们取得这个值的一份拷贝并把它与1相加,表达式的最终结果为字符b。图中虚线表示表达式求值时数据的移动过程 。这个表达式的最终结果的存储位置并未清晰定义,所以它不是一个合法的左值。优先级表格证实+的结果不能作为左值。

  1. *(cp+1)

在这里插入图片描述
这个括号使表达式先执行加法运算,就是把1和cp中所存储的地址相加。此时的结果值是图中虚线椭圆所示的指针。接下来的间接访问操作随着箭头访问紧随ch之后的内存位置。这样,这个表达式的右值就是这个位置的值,而它的左值是这个位置本身

在这里我们需要学习很重要的一点。注意指针加法运算的结果是个右值,因为它的存储位置并未清晰定义。如果没有间接访问操作,这个表达式将不是一个合法的左值。然而,间接访问跟随指针访问一个特定的位置。这样,*(cp+1)就可以作用左值使用,尽管cp+1本身并不是左值。间接访问操作符是少数几个其结果为左值的操作符之一。我们也无法得知ch后面的那个内存位置,所以像这样的表达式是非法的。
8. ++cp

在这里插入图片描述

++和–操作符在指针变量中使用得相当频繁,所以在这种上下文环境中理解它们是非常重要的。在这个表达式中,我们增加了指针变量cp的值。(为了让图更清楚,我们省略了加法)。表达式的结果是增值后的指针的一份拷贝,因为前缀++先增加它的操作数的值再返回这个结果。这份拷贝的存储位置并未清晰定义,所以它不是一个合法的左值

  1. cp++

在这里插入图片描述
后缀++操作符同样增加cp的值,但它先返回cp值的一份拷贝然后再增加cp的值。这样,这个表达式的值就是cp原来的值的一份拷贝。

前面两个表达式的值都不是合法的左值。但如果我们在表达式中增加了间接访问操作符,它们就可以成为合法的左值,如下面的两个表达式所示

  1. *++cp

在这里插入图片描述
这里,间接访问操作符作用于增值后的指针的拷贝上,所以它的右值是ch后面那个内存地址的值,而它的左值就是那个位置本身。

  1. *cp++

在这里插入图片描述
使用后缀++操作符所产生的结果不同:它的右值和左值分别是变量ch的值和ch的内存位置,也就是cp原先所指。同样,后缀++操作符在周
围的表达式中使用其原先操作数的值。间接访问操作符和后缀++操作符的组合常常令人误解。优先级表格显示后缀++操作符的优先级高于*操作符,但表达式的结果看上去像是先执行间接访问操作。事实上,这里涉及3个步骤:(1)++操作符产生cp的一份拷贝,(2)然后++操作符增加cp的值,(3)最后,在cp的拷贝上执行间接访问操作。

这个表达式常常在循环中出现,首先用一个数组的地址初始化指针,
然后使用这种表达式就可以依次访问该数组的内容了。

  1. ++*cp

在这里插入图片描述
在这个表达式中,由于这两个操作符的结合性都是从右向左,所以首先执行的是间接访问操作。然后,cp所指向的位置的值增加1,表达式的结果是这个增值后的值的一份拷贝。

注意加法的操作是在ch里面实现的,这点要和上面的*cp+1区分开。而且执行的顺序有点不同。

  • ++*cp是先加1再拷贝
  • *cp+1是先拷贝再加1
  1. (*cp)++

在这里插入图片描述
使用后缀++操作符,我们必须加上括号,使它首先执行间接访问操
作。这个表达式的计算过程与前一个表达式相似,但它的结果值是ch增值前的原先值。

  1. ++*++cp

在这里插入图片描述
这个表达式看上去相当诡异,但事实上并不复杂。但是,如果你逐个对它们进行分析,你会发现它们都很熟悉。事实上,我们先前已经计算了*++cp,所以现在我们需要做的只是增加它的结果值。但是,让我们还是从头开始。记住这些操作符的结合性都是从右向左,所以首先执行的是++cp。cp下面的虚线椭圆表示第1个中间结果。接着,我们对这个拷贝值进行间接访问,它使我们访问ch后面的那个内存位置。第2个中间结果用虚线方框表示,因为下一个操作符把它当作一个左值使用。最后,我们在这个位置执行++操作,也就是增加它的值。我们之所以把结果值显示为?+1是因为我们并不知道这个位置原先的值。

  1. ++*cp++

在这里插入图片描述
这个表达式和前一个表达式的区别在于这次第1个++操作符是后缀
形式而不是前缀形式。由于它的优先级较高,所以先执行它。间接访问操作所访问的是cp所指向的位置而不是cp所指向位置后面的那个位置。

相关参考

  1. 《C和指针》

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1006383.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

fabirc 将图像绘制原点定为图形内部

添加元素时,设置属性: originX: center, originY: center, 如我定义两个矩形: addrect () { // 矩形1var rect new fabric.Rect({top: 50,left: 100,width: 100,height: 70,fill: #F56C6C,strokeUniform: true // 限制边框宽度缩放})canva…

vue2实现自定义主题webpack-theme-color-replacer

需求:根据element的自定义主题色,之后改变element的全局所有颜色,解决页面刷新后主题色失效问题,这个需要把颜色存入到浏览器的存储中,如果换个浏览器就得重新选择了哈,如果需要在不同的浏览器保持一致的主…

langchain主要模块(四):Memory

langchain2之Memory langchain1.概念2.主要模块模型输入/输出 (Model I/O)数据连接 (Data connection)链式组装 (Chains)代理 (Agents)内存 (Memory)回调 (Callbacks) 3.MemoryConversationBufferMemoryConversationBufferWindowMemoryConversationTokenBufferMemoryConversati…

Error: svn: E155004: Run ‘svn cleanup‘ to remove locks

解决办法如下:点击settings 点击清除缓存按钮,然后再使用svn进行提交更新操作,但是可能还会有其它的错误,比如svn: E230001: Server SSL certificate verification failed,解决这个错误请参考我另一篇文章:…

【LeetCode: 2596. 检查骑士巡视方案:深度优先搜索】

🚀 算法题 🚀 🌲 算法刷题专栏 | 面试必备算法 | 面试高频算法 🍀 🌲 越难的东西,越要努力坚持,因为它具有很高的价值,算法就是这样✨ 🌲 作者简介:硕风和炜,…

力扣 -- 300. 最长递增子序列

解题步骤&#xff1a; 参考代码&#xff1a; class Solution { public:int lengthOfLIS(vector<int>& nums) {int nnums.size();vector<int> dp(n,1);int ret1;for(int i0;i<n;i){for(int j0;j<i;j){if(nums[j]<nums[i]){dp[i]max(dp[i],dp[j]1);}}r…

分布式系统第五讲:分布式事务及实现方案

分布式系统第五讲&#xff1a;分布式事务及实现方案 事务是一个程序执行单元&#xff0c;里面的所有操作要么全部执行成功&#xff0c;要么全部执行失败。而分布式事务是指事务的参与者、支持事务的服务器、资源服务器以及事务管理器分别位于不同的分布式系统的不同节点之上。本…

微服务07-认识MQ+RabbitMQ入门

1.前言 了解同步调用和异步调用 1.1.同步调用 比如这里的支付服务&#xff0c;需要等待订单服务、短信服务…执行完毕才能执行&#xff0c;这样支付整个流程完毕需要500ms 然后如果订单、仓储等其中一个服务挂掉了&#xff0c;那么支付服务请求请求不了&#xff0c;挂掉的服…

Python实现进制转换

1 问题 如何将一个数从十进制转换成二进制&#xff0c;这个数从键盘输入。 2 方法 &#xff08;1&#xff09;输入一个十进制的数; &#xff08;2&#xff09;将这个数除以2求商和余数 &#xff08;3&#xff09;再用2去除商&#xff0c;又会得到一个商和余数&#xff0c;如此进…

企业用设备管理系统有什么好处?的修大数据管理平台的功能强大吗?

在现代工业运营中&#xff0c;设备的高效管理与维护对于维持生产稳定及提升企业竞争力具有重大意义。而设备管理系统作为一项重要工具&#xff0c;可以极大地提高企业的生产效率与设备维护的准确性。 企业使用设备管理系统的原因如下&#xff1a;   提高设备利用率&#xff1…

小白自助发卡销售系统|支持自助选号批量购买Thinkphp6.0+Layui

小白自助发卡销售系统 本系统由小白独立开发thinkphp+layui 功能特色: 1.前台下单支持自助选号。 2.前台库存显示方案增加3种。 3.自定义商品图片。 4.支持邮件通知功能具体可到后台自行体验。 5.支持批量下单购买。 6.支持易支付接口(目前用的比较多) 7.支持批量导入卡密 8.P…

从0到1理解ChatGPT原理

目录 写在前面 1.Tansformer架构模型 2.ChatGPT原理 3.提示学习与大模型能力的涌现 3.1提示学习 3.2上下文学习 3.3思维链 4.行业参考建议 4.1拥抱变化 4.2定位清晰 4.3合规可控 4.4经验沉淀 机械工业出版社京东自购链接 写在前面 2022年11月30日&#xff0c;ChatG…

【数据结构】二叉树链式结构的实现(三)

目录 一&#xff0c;二叉树的链式结构 二&#xff0c;二叉链的接口实现 1&#xff0c;二叉链的创建 2&#xff0c;接口函数 3&#xff0c;动态创立新结点 4&#xff0c;创建二叉树 5&#xff0c;前序遍历 6&#xff0c;中序遍历 7&#xff0c;后序遍历 三&#xff0c;结点个…

嵌入式学习笔记(25)串口通信的基本原理

三根通信线&#xff1a;Tx Rx GND &#xff08;1&#xff09;任何通信都要有信息作为传输载体&#xff0c;或者有线的或则无线的。 &#xff08;2&#xff09;串口通信时有线通信&#xff0c;是通过串口线来通信的。 &#xff08;3&#xff09;串口通信最少需要2根&#xff…

“熊猫杯” | 赛宁网安获网络安全优秀创新成果大赛优胜奖

9月11日&#xff0c;四川省2023年国家网络安全宣传周正式启动。由四川省委网信办指导&#xff0c;中国网络安全产业联盟&#xff08;CCIA&#xff09;主办&#xff0c;成都信息工程大学、四川省网络空间安全协会承办的“2023年网络安全优秀创新成果大赛—成都分站赛(暨四川省‘…

Linux HTTP协议

目录 1.浏览器与服务器通信过程2.HTTP请求报头&#xff08;1&#xff09;HTTP的请求报头结构&#xff08;2&#xff09;HTTP的请求方法 3.HTTP应答报头&#xff08;1&#xff09;HTTP的应答报头结构&#xff08;2&#xff09; HTTP的应答状态 1.浏览器与服务器通信过程 浏览器…

小程序源码:多功能口袋工具箱微信小程序源码-带流量主|云开发(更新)

这里主要分享多功能口袋工具箱微信小程序源码&#xff0c;有带流量主&#xff0c;而且超多功能工具箱组合的微信小程序源码。无需服务器即可搭建&#xff0c;可以设置流量主赚取收益。 源码链接&#xff1a; 网盘源码 密码&#xff1a;hma8 工具箱的应用一览&#xff1a; 1…

部署kibana可视化平台

部署kibana可视化平台 案例版本&#xff1a;kibana 8.6.2 1、下载 去官网中查找与安装的ES版本一致的安装包&#xff1a;官网kibana下载。 可以选择自行下载好后放入服务器中进行安装&#xff0c;也可以使用wget下载&#xff1a; # 下载 cd /home/elasticsearch/kibana wge…

LeetCode 1132.申请的报告2

数据准备 Create table If Not Exists Actions (user_id int, post_id int, action_date date, action ENUM(view, like, reaction, comment, report, share), extra varchar(10)); create table if not exists Removals (post_id int, remove_date date); Truncate table Act…

【ccf-csp题解】第2次csp认证-第四题-最优配餐-多源BFS

题目描述 思路讲解 这道题的数据量是10^6&#xff0c;直接用floyd算法&#xff0c;肯定是不行的 此处介绍新的思路&#xff1a;多源BFS 我们的核心目的是&#xff0c;对于每一个终点&#xff08;即此处的顾客&#xff09;能够找到离它最近的源点&#xff08;即此处的餐馆&am…