c语言 箭头什么作用,为什么C中的箭头(->)运算符存在?

53fe8ab9a991bde08166343349366055.png

万千封印

我将您的问题解释为两个问题:1)为什么->甚至存在,以及2)为什么.不自动取消引用指针。这两个问题的答案都有历史渊源。为什么会->存在?在C语言的第一个版本(1975年5月第6版Unix附带的C语言,我将其称为CRM的参考手册)中,运算符->具有非常排他的含义,不是同义词*和.组合CRM描述的C语言在许多方面与现代C都有很大不同。在CRM结构成员中,实现了字节偏移量的全局概念,该概念可以无限制地添加到任何地址值中。即所有结构成员的所有名称都具有独立的全局含义(因此,必须是唯一的)。例如,您可以声明struct S {  int a;  int b;};并且name a代表偏移量0,而name b代表偏移量2(假设int大小为2,并且没有填充)。该语言要求翻译单元中所有结构的所有成员都必须具有唯一的名称或代表相同的偏移值。例如,在同一翻译单元中,您还可以声明struct X {  int a;  int x;};这样就可以了,因为名称a始终代表偏移量0。但是,此附加声明struct Y {  int b;  int a;};从形式上来说是无效的,因为它试图“重新定义” a为偏移量2和b偏移量0。这就是->运算符的作用。由于每个struct成员名称都有自己的自足全局含义,因此该语言支持此类表达式int i = 5;i->b = 42;  /* Write 42 into `int` at address 7 */100->a = 0; /* Write 0 into `int` at address 100 */编译器将第一个赋值解释为“获取地址5,向其添加偏移量2,然后赋给结果地址中42的int值”。即上述内容将分配42给intaddress的值7。请注意,对的这种使用->并不关心左侧表达式的类型。左侧被解释为右值数字地址(可以是指针或整数)。*和.组合不可能进行这种欺骗。你做不到(*i).b = 42;因为*i已经是一个无效的表达式。由于*运算符与分开.,因此对其操作数施加了更严格的类型要求。为了提供解决此限制的功能,CRM引入了->运算符,该运算符与左侧操作数的类型无关。正如Keith在评论中所指出的,+ ->和*+ .组合之间的区别就是CRM在7.1.8中称为“放松要求”:除了放宽E1指针类型的要求之外,该表达式E1−>MOS完全等同于(*E1).MOS后来,在K&R C中,对CRM中最初描述的许多功能进行了重新设计。完全删除了“结构成员作为全局偏移量标识符”的想法。功能,和->运营商成为了的功能完全一致*和.组合。为什么不能.自动取消引用指针?同样,在该语言的CRM版本中,.需要将运算符的左操作数设为lvalue。那是对该操作数施加的唯一要求(这就是使它不同于的原因->,如上所述)。需要注意的是CRM并没有要求左侧操作数.有一个结构类型。它只是要求它是一个左值,任何左值。这意味着在C的CRM版本中,您可以编写如下代码struct S { int a, b; };struct T { float x, y, z; };struct T c;c.b = 55;在这种情况下,即使type 没有名为的字段,编译器也将写入连续存储块中位于字节偏移2处55的int值。编译器根本不关心实际类型。它关心的只是一个左值:某种可写的内存块。cstruct Tbcc现在请注意,如果您这样做S *s;...s.b = 42;该代码将被视为有效(因为s它也是一个左值),并且编译器将仅尝试以字节偏移量2 将数据写入指针s本身。不用说,类似这样的事情很容易导致内存溢出,但是语言不关心这些事情。即在该语言的该版本中,您提出的有关.指针类型重载运算符的想法将不起作用:.与指针(与左值指针或任何左值一起使用)时,运算符已经具有非常特殊的含义。毫无疑问,这是非常奇怪的功能。但是当时在那里。当然,这种怪异的功能并不是.在重新设计的C-K&R C版本中引入指针的重载运算符的强烈理由(如您所建议的那样),但尚未完成。也许那时是必须要支持一些用CRM版本的C编写的遗留代码。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值