【问题标题】:Does each expression in C++ have a non-pointer type like non-reference typeC++ 中的每个表达式是否都具有非指针类型,如非引用类型
【发布时间】:2021-07-28 14:21:00
【问题描述】:

您好,我正在阅读 C++ 中的表达式以及整个语句

声明 0.0

每个表达式都有一些非引用类型

引用的声明来自 en.cppreference.com/w/cpp/language/value_category。检查页面顶部的第 2 行。

现在我举了一些例子来理解这意味着什么。例如:

int i = 100; // this expression has type int
int &j = i; // this expression has type int or int&?

我的困惑是我知道 j 是对 int 的引用,即 j 是 int& 但根据引用的语句,每个表达式都有一个非引用类型将暗示 int &j = i; 具有 int 类型。它是否正确? 我感到困惑的其他示例:

int a[4] = {2,4,4,9};
a[3]; // will this expression be int& type or int type?

现在在a[3]; 语句中,我知道 a 是一个数组左值,因此 a[3] 返回一个对最后一个元素的左值引用。但是对引用的语句 0.0 是否意味着整个表达式 a[3]; 是 int 或 int& 类型感到困惑?

这是另一个例子:

b[4]; // Here assume that b is an array rvalue. So will this expression has  type int&& or int?

所以我的问题是指针也会发生类似的事情吗?这意味着我们对指针也有类似的声明(0.0)吗?

int x = 34;
int *l = &x; // will this expression have type int* or int?

我现在这里 l 是一个指向 int(复合类型)的指针。如果没有类似的指针语句,那么引用语句需要什么?这就是为什么我们只去掉参考部分?

【问题讨论】:

  • 引用的语句来自哪里?正如你所指出的那样,这似乎不是真的。
  • 哪里是这样写的?可以分享一下链接吗?
  • 引用的语句来自en.cppreference.com/w/cpp/language/value_category。检查页面顶部的第 2 行。
  • 整个短语为:“每个表达式都有一些非引用类型,每个表达式恰好属于三个主要值类别之一:prvalue、xvalue 和 lvalue。”相反,这样做更有意义。
  • 您引用的完整语句的两半由“and”连接,这意味着它们也应该分别有意义。 @al3c

标签: c++ c++11 reference c++14 language-lawyer


【解决方案1】:
int i = 100; // this expression has type int
int &j = i; // this expression has type int or int&?

这些语句根本不是表达式。这些是声明。它们确实包含子表达式 100 和 i,它们的类型都是 int。如果在此声明之后使用 id 表达式 j,则该表达式的类型将为 int

所以我的问题是指针也会发生类似的事情吗?

没有。指针是非引用类型,类似的事情不会发生在具有指针类型的表达式上。

为什么我们只去掉参考部分?

这就是语言的工作原理。它允许我们以同样的方式对待对象和对对象的引用。

这就是您不需要(也不能)显式使用间接运算符来访问被引用对象的部分原因,这与需要使用间接运算符来访问指向对象不同。


这是实际的语言规则(来自最新的标准草案):

[表达式类型]

如果表达式最初具有类型“对 T 的引用”([dcl.ref],[dcl.init.ref]),则在进一步分析之前将类型调整为 T。 表达式指定引用所指的对象或函数,表达式是左值还是x值,具体取决于表达式。

【讨论】:

    【解决方案2】:

    C++ 表达式可以有j+0(j) 之类的形式。这些表达式肯定有 int 类型。

    为了简化语法,变量名本身也可以用作表达式。如果我们没有这条规则,在语法中我们会有很多variable-or-expression 构造。但这意味着表达式j表达式j+0的类型相同,即int

    【讨论】:

      【解决方案3】:

      表达式:表达式由一个或多个操作数组成,并在计算时产生结果。

      请注意“评估时”的措辞。

      表达式的示例是:文字 4,或某个变量 n。再次注意,表达式尚未计算,因此没有结果。您还可以从运算符和一个或多个操作数创建更复杂的表达式。例如 3 + 4*5 是一个未计算的表达式。具有两个或多个运算符的表达式称为复合表达式。

      C++ 中的每个表达式要么是右值,要么是左值。

      表达式语句: 语句以分号结尾。当我们在表达式中添加分号时,它就变成了表达式语句。这样做的影响是,这会导致表达式被计算并且其结果在语句结束时被丢弃。因此,例如文字 5 是一个表达式,但如果添加分号 ;在它之后,我们将有一个表达式语句。此表达式的结果也将在表达式语句的末尾被丢弃。让我们看另一个例子,cout << n; 是一个整体的表达式语句。它由以下表达式组成:

      1. expression `cout`
      2. expression 'n'
      

      它由一个运算符<<和一个空语句组成; 整个语句将产生副作用,即在屏幕上打印 n 的值。

      更新:

      示例 1: std::cout << n; 会产生在屏幕上打印 n 值的副作用,但更重要的是,它还会产生一个结果值,即对象 std::cout在语句末尾被丢弃。

      示例 2: int i(20 + 1); 包含 3 个东西:

      1. 类型:int
      2. 标识符 i
      3. 表达式 20 + 1

      示例 3: float p; 这没有表达式。这只是变量定义。也称为声明语句。

      示例 4: float k = 43.2; 这在右侧有一个表达式,即 43.2、一个浮点类型和一个标识符 k。

      示例 5: i = 43;。这是一个表达式语句。这里有两个表达式和一个运算符。结果就是变量 i。

      示例 6: int &r = i;。这是一个声明语句,因为它由右侧的表达式 i 组成。同样在左侧,我们有一个类型(int)和一个声明符(&r)。由于它不是表达式语句,因此不会丢弃任何值。

      示例 7: int *p = &i; 这是一个声明语句,因为它由右侧的表达式 i 组成。同样在左侧,我们有一个类型(int)和一个声明符(&r)。由于它不是表达式语句,因此不会丢弃任何值。

      示例 8: i = a < b ? a : b; 这是一个表达式语句。这里的表达式是:

      1. 左侧变量 i

      2. 右边的变量a

      3. 右边的变量b

      中间还有一个条件(a

      【讨论】:

      • int &r = i; 是一个定义,因此不可能是一个表达式语句。但是,它确实包含一个初始化表达式。由于它不是表达式语句,因此没有丢弃任何值。 int *p = &i 也一样。
      • 是的,它们都是声明语句。所以什么都不会被丢弃。
      猜你喜欢
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 2016-08-23
      • 1970-01-01
      • 1970-01-01
      相关资源
      最近更新 更多