赋值运算符

来自cppreference.com
< c‎ | language

赋值及复合赋值运算符是二元运算符,它们用在其右侧的值替换存储在左侧对象中的值。

运算符 运算符名 示例 描述 等价
= 基本赋值 a = b a 被替换为 b 不适用
+= 加法赋值 a += b a 被替换为 ab 的和 a = a + b
-= 减法赋值 a -= b a 被替换为 ab 的差 a = a - b
*= 乘法赋值 a *= b a 被替换为 ab 的积 a = a * b
/= 除法赋值 a /= b a 被替换为 a 除以 b的商 a = a / b
%= 模赋值 a %= b a 被替换为 a 除以 b 的余数 a = a % b
&= 逐位与赋值 a &= b a 被替换为 ab 的逐位或 a = a & b
|= 逐位或赋值 a |= b a 被替换为 ab 的逐位与 a = a | b
^= 逐位异或赋值 a ^= b a 被替换为 ab 的逐位异或 a = a ^ b
<<= 逐位左移赋值 a <<= b a 被替换为 a 左移 b a = a << b
>>= 逐位右移赋值 a >>= b a 被替换为 a 右移 b a = a >> b

简单赋值

简单赋值表达式的形式为

lhs = rhs
lhs - 任何完整对象类型的可修改左值表达式
rhs - 任何可隐式转换lhs 或与 lhs 兼容类型的表达式

赋值进行从 rhs 的值到 lhs 类型的隐式转换,然后用 rhs 转换后的值替换 lhs 所指代对象中的值。

赋值返回存储于 lhs 中的相同值(故如 a = b = c 的表达式合法)。赋值表达式的值类别是非左值(故形如 (a = b) = c 的表达式非法)。

rhslhs 必须满足下列条件:

  • lhsrhs 均拥有算术类型,此情况下 lhs 可有volatile 限定原子的 (C11 起)
  • lhsrhs 均拥有指向兼容类型的指针类型,或其中一个指针是指向 void 的指针,而转换不为被指向类型添加限定符。 lhs 可以为 volatile restrict (C99 起) 限定,或原子的 (C11 起)
  • lhs 是(可以有限定或原子的 (C11 起))指针,而 rhs 是空指针常量,如 NULL nullptr_t (C23 起)
  • lhs 是(可以有限定或原子的 (C11 起)_Bool 类型而 rhs 是指针nullptr_t (C23 起)
(C99 起)
(C23 起)

注解

rhslhs 在内存中重叠(例如它们是同一联合体的成员),则行为未定义,除非重叠是准确的且两者类型兼容

尽管数组不可赋值,但包裹在结构体内的数组可以赋值给另一个相同(或兼容)的结构体类型。

更新 lhs 的副效应后序于值计算,但 lhsrhs 自己和运算数的求值不是,与通常一样,它们相对于彼此不定序(故如 i = ++i 的表达式结果未定义)。

赋值会剥除浮点表达式的额外范围和精度(见 FLT_EVAL_METHOD )。

C++ 中,赋值表达式是左值表达式,而 C 中不是。

#include <stdio.h>
 
int main(void)
{
    // 普通整形
    int i = 1, j = 2, k = 3; // 初始化,并非赋值
 
    i = j = k;   // i和j现在都为3
//  (i = j) = k; // 错误:等号左侧应该为左值 (注:此写法在cpp中允许)
    printf("%d %d %d\n", i, j, k);
 
    const char c = 'A';    // 初始化,并非赋值
    const char *p = &c;    // 初始化,并非赋值
    const char **cpp = &p; // 初始化,并非赋值
 
//  cpp = &p;   // 错误: char** 不能转换为 const char**
    *cpp = &c;  // OK : char* 能转换为 const char*
    printf("%c \n", **cpp);
    cpp = 0;    // OK :空指针常量能转换为任何指针
 
    int arr1[2] = {1,2}, arr2[2] = {3, 4};
//  arr1 = arr2; // 错误:不能通过数组赋值
 
    struct s { 
         int arr[2]; // 包装于结构体中的数组
    } sam1 = {5, 6}, sam2 = {7, 8};
    sam1 = sam2; // OK :能赋值包装于结构体中的数组
 
    printf("%d %d \n",sam1.arr[0],sam1.arr[1]);
}

输出:

3 3 3
A 
7 8

复合赋值

复合赋值表达式的形式为

lhs op rhs
op - one of *=, /= %=, += -=, <<=, >>=, &=, ^=, |=
lhs, rhs - 拥有算术类型的表达式(其中 lhs 可以有限定或是原子的),除非 op+=-= ,此情况允许接受拥有与 + 和 - 相同限制的指针类型

表达式 lhs @= rhslhs = lhs @ ( rhs ) 准确相同,除了只求值一次 lhs

lhs 拥有原子类型,则运算表现为单个带内存顺序 memory_order_seq_cst 的原子读修改写操作

对于整数原子类型,复合赋值运算 @= 等价于:

T1* addr = &lhs;
T2 val = rhs;
T1 old = *addr;
T1 new;
do { new = old @ val } while (!atomic_compare_exchange_strong(addr, &old, new);
(C11 起)

引用

  • C17 标准(ISO/IEC 9899:2018):
  • 6.5.16 Assignment operators (第 72-73 页)
  • C11 标准(ISO/IEC 9899:2011):
  • 6.5.16 Assignment operators (第 101-104 页)
  • C99 标准(ISO/IEC 9899:1999):
  • 6.5.16 Assignment operators (第 91-93 页)
  • C89/C90 标准(ISO/IEC 9899:1990):
  • 3.3.16 Assignment operators

参阅

运算符优先级

常用运算符
赋值 自增
自减
算术 逻辑 比较 成员
访问
其他

a = b
a += b
a -= b
a *= b
a /= b
a %= b
a &= b
a |= b
a ^= b
a <<= b
a >>= b

++a
--a
a++
a--

+a
-a
a + b
a - b
a * b
a / b
a % b
~a
a & b
a | b
a ^ b
a << b
a >> b

!a
a && b
a || b

a == b
a != b
a < b
a > b
a <= b
a >= b

a[b]
*a
&a
a->b
a.b

a(...)
a, b
(type) a
? :
sizeof
_Alignof
(C11 起)