棋牌类网站怎么做,车工订单网站,优化设计七年级上册英语答案,房产智能建站系统肯尼斯里科《C和指针》第6章 指针#xff08;1#xff09;-CSDN博客
肯尼斯里科《C和指针》第6章 指针#xff08;2#xff09;-CSDN博客
前置知识#xff1a;左值右值 为了理解有些操作符存在的限制#xff0c;必须理解左值(L-value)和右值(R-value)之间的区别。这两个…肯尼斯·里科《C和指针》第6章 指针1-CSDN博客
肯尼斯·里科《C和指针》第6章 指针2-CSDN博客
前置知识左值右值 为了理解有些操作符存在的限制必须理解左值(L-value)和右值(R-value)之间的区别。这两个术语是多年前由编译器设计者所创造并沿用至今的尽管它们的定义并不与C语言严格吻合。 反正我上课是没专门学过这部分内容的在还没看这本书之前我的掌握程度是——把等号右边的值赋给左边。但是我发现仅依靠这点认识是没办法理解下面的“指针表达式”的所以得回头去学。
嘛不好评价。 左值就是那些可以出现在赋值符号左边的东西。右值就是那些可以出现在赋值符号右边的东西。这里有个例子 a b 25; a是个左值因为它标识了一个可以存储结果值的地点b25是个右值因为它指定了一个值。 它们可以互换吗 b 25 a; 原先用作左值的a此时也可以当作右值因为每个位置都包含一个值。然而b25不能作为左值因为它并未标识一个特定的位置。因此这条赋值语句是非法的。 注意当计算机计算b25时它的结果必然保存于机器的某个地方。但是程序员并没有办法预测该结果会存储在什么地方也无法保证这个表达式的值下次还会存储于同一个地方。其结果是这个表达式不是一个左值。基于同样的理由字面值常量也都不是左值。 左值是个表达式它可以出现在赋值符的左边表示计算机内存中的一个位置。右值表示一个值所以它只能出现在赋值符的右边。每个左值表达式同时也是个右值但反过来就不是这样。 表示计算机内存中的一个位置唉我之前就没有这么理解过(′⌒)。
6.11 指针表达式
唉要是我当时学的时候看到这个就好了唉。 现在让我们观察各种不同的指针表达式并看看当它们分别作为左值和右值时是如何进行求值的。有些表达式用得很普遍但有些却不常用。这个练习的目的并不是想给你一本这类表达式的“烹调全书”而是想让你完善阅读和编写它们的技巧。 char ch a;
char *cp ch;现在我们就有了两个变量初始化如下 图中还显示了ch后面的那个内存位置因为我们所求值的有些表达式将访问它尽管是在错误情况下才会对它进行访问。由于我们并不知道它的初始值因此用一个问号来代替。 首先来个简单的作为开始如下面这个表达式 ch 当它作为右值使用时表达式的值为a如下图所示 粗椭圆提示变量ch的值就是表达式的值。但是当这个表达式作为左值使用时它是这个内存的地址而不是该地址所包含的值所以它的图示方式有所不同 要是我当时知道“左”“右”值就好了捂脸。 此时该位置用粗方框标记提示这个位置就是表达式的结果。另外它的值并未显示因为它并不重要。事实上这个值将被某个新值所取代。接下来的表达式将以表格的形式出现。每个表后面的文字是表达式求值过程的描述。 作为右值这个表达式的值是变量ch的地址。注意这个值同变量cp中所存储的值一样但这个表达式并未提及cp所以这个结果值并不是因为它而产生的。这样其中的椭圆并不画于cp的箭头周围。第2个问题是为什么这个表达式不是一个合法的左值答案很简单当表达式ch进行求值时它的结果应该存储于计算机的什么地方呢它肯定会位于某个地方但你无法知道它位于何处。这个表达式并未标识任何机器内存的特定位置所以它不是一个合法的左值。 你以前曾见到过上面这个表达式。它的右值就是cp的值。它的左值就是cp所处的内存位置。由于这个表达式并不进行间接访问操作因此不必依箭头所示进行间接访问。 上面这个例子与ch类似不过我们这次所取的是指针变量的地址。这个结果的类型是指向字符的指针的指针。同样这个值的存储位置并未清晰定义所以这个表达式不是一个合法的左值。 现在我们加入了间接访问操作所以它的结果应该不会令人惊奇。但接下来的几个表达式就比较有意思。 上面这个表涉及的东西更多所以让我们一步一步来分析它。这里有两个操作符。*的优先级高于所以首先执行间接访问操作如图中cp到ch的实线箭头所示我们可以得到它的值如虚线椭圆所示。我们取得这个值的一份副本并把它与1相加表达式的最终结果为字符’b’。图中虚线表示表达式求值时数据的移动过程。这个表达式的最终结果的存储位置并未清晰定义所以它不是一个合法的左值。 在这个例子中我们在前面那个表达式中增加了一个括号。这个括号使表达式先执行加法运算就是把1和cp中所存储的地址相加。此时的结果值是图中虚线椭圆所示的指针。接下来的间接访问操作随着箭头访问紧随ch之后的内存位置。这样这个表达式的右值就是这个位置的值而它的左值是这个位置本身。 在这里我们需要学习很重要的一点。注意指针加法运算的结果是个右值因为它的存储位置并未清晰定义。如果没有间接访问操作这个表达式将不是一个合法的左值。然而间接访问跟随指针访问一个特定的位置。这样*(cp1)就可以作用左值使用尽管cp1本身并不是左值。间接访问操作符是少数几个其结果为左值的操作符之一。 但是这个表达式所访问的是ch后面的那个内存位置我们如何知道原先存储于那个地方的是什么东西呢一般而言我们无法得知所以像这样的表达式是非法的。本章的后面我将更为深入地探讨这个问题。 后缀操作符同样增加cp的值但它先返回cp值的一份拷贝然后再增加cp的值。这样这个表达式的值就是cp原来的值的一份副本。 前面两个表达式的值都不是合法的左值。但如果我们在表达式中增加了间接访问操作符它们就可以成为合法的左值如下面的两个表达式所示。 这里间接访问操作符作用于增值后的指针的副本所以它的右值是ch后面那个内存地址的值而它的左值就是那个位置本身。 使用后缀操作符所产生的结果不同它的右值和左值分别是变量ch的值和ch的内存位置也就是cp原先所指。同样后缀操作符在周围的表达式中使用其原先操作数的值。间接访问操作符和后缀操作符的组合常常令人误解。优先级表格显示后缀操作符的优先级高于*操作符但表达式的结果看上去像是先执行间接访问操作。事实上这里涉及3个步骤操作符产生cp的一份副本然后操作符增加cp的值最后在cp的副本上执行间接访问操作。 这个表达式常常在循环中出现首先用一个数组的地址初始化指针然后使用这种表达式就可以依次访问该数组的内容了。本章后面显示了一些这方面的例子。 在这个表达式中由于这两个操作符的结合性都是从右向左因此首先执行的是间接访问操作。然后cp所指向的位置的值增加1表达式的结果是这个增值后的值的一份副本。