C++的运算符与表达式

简介: 在程序中,运算符是用来操作数据的,因此这些数据也被称为操作数,使用运算符将操作数连接而成的式子称为表达式

C++的运算符与表达式


运算符

●  运算符是一种告诉编译器执行特定的数学或逻辑操作的符号。C++内置了丰富的运算符,并且提供了以下类型的运算符:

  • 算数运算符
  • 关系运算符
  • 逻辑运算符
  • 位运算符
  • 赋值运算符
  • 杂项运算符


表达式

●  在程序中,运算符是用来操作数据的,因此这些数据也被称为操作数,使用运算符将操作数连接而成的式子称为表达式


表达式具有如下特点:


●  常量和变量都是表达式 例如常量3.14 变量i

●  运算符的类型对应表达式的类型,例如算数运算符对应算数表达式

●  每一个表达式都有自己的值,即表达式都有运算结果


算数运算符

假设变量A的值为10,变量B的值为20则:

运算符 描述 实例
+ 把两个操作数相加 A + B 将得到 30
- 从第一个操作数中减去第二个操作数 A - B 将得到 -10
* 把两个操作数相乘 A * B 将得到 200
/ 分子除以分母 B / A 将得到 2
% 取模运算符,整除后的余数 B % A 将得到 0
++ 自增运算符,整数值增加 1 A++ 将得到 11
自减运算符,整数值减少 1 A-- 将得到 9

关系运算符

运算符 描述 实例
== 检查两个操作数的值是否相等,如果相等则条件为真。 (A == B) 不为真。
!= 检查两个操作数的值是否相等,如果不相等则条件为真。 (A != B) 为真。
> 检查左操作数的值是否大于右操作数的值,如果是则条件为真。 (A > B) 不为真。
< 检查左操作数的值是否小于右操作数的值,如果是则条件为真。 (A < B) 为真。
>= 检查左操作数的值是否大于或等于右操作数的值,如果是则条件为真。 (A >= B) 不为真。
<= 检查左操作数的值是否小于或等于右操作数的值,如果是则条件为真。 (A <= B) 为真。

逻辑运算符

运算符 描述 实例
&& 称为逻辑与运算符。如果两个操作数都非零,则条件为真。 (A && B) 为假。
|| 称为逻辑或运算符。如果两个操作数中有任意一个非零,则条件为真。 (A || B) 为真。
称为逻辑或运算符。如果两个操作数中有任意一个非零,则条件为真。 !(A && B) 为真。

赋值运算符

运算符 描述 实例
= 简单的赋值运算符,把右边操作数的值赋给左边操作数 C = A + B 将把 A + B 的值赋给 C
+= 加且赋值运算符,把右边操作数加上左边操作数的结果赋值给左边操作数 C += A 相当于 C = C + A
-= 减且赋值运算符,把左边操作数减去右边操作数的结果赋值给左边操作数 C -= A 相当于 C = C - A
*= 乘且赋值运算符,把右边操作数乘以左边操作数的结果赋值给左边操作数 C *= A 相当于 C = C * A
/= 除且赋值运算符,把左边操作数除以右边操作数的结果赋值给左边操作数 C /= A 相当于 C = C / A
%= 求模且赋值运算符,求两个操作数的模赋值给左边操作数 C %= A 相当于 C = C % A
<<= 左移且赋值运算符 C <<= 2 等同于 C = C << 2
>>= 右移且赋值运算符 C >>= 2 等同于 C = C >> 2
&= 按位与且赋值运算符 C &= 2 等同于 C = C & 2
^= 按位异或且赋值运算符 C ^= 2 等同于 C = C ^ 2
|= 按位或且赋值运算符 C |= 等同于 C = C |2

位运算符

运算符 描述 实例
& 如果同时存在于两个操作数中,二进制 AND 运算符复制一位到结果中。 (A & B) 将得到 12,即为 0000 1100
| 如果存在于任一操作数中,二进制 OR 运算符复制一位到结果中。 (A | B) 将得到 12,即为 0000 1100
^ 如果存在于其中一个操作数中但不同时存在于两个操作数中,二进制异或运算符复制一位到结果中。 (A ^ B) 将得到 49,即为 0011 0001
~ 二进制补码运算符是一元运算符,具有"翻转"位效果,即0变成1,1变成0。 (~A ) 将得到 -61,即为 1100 0011,一个有符号二进制数的补码形式。
<< 二进制左移运算符。左操作数的值向左移动右操作数指定的位数。 A << 2 将得到 240,即为 1111 0000
>> 二进制右移运算符。左操作数的值向右移动右操作数指定的位数。 A >> 2 将得到 15,即为 0000 1111

优先级:


位与疏散运算符都是双目运算符,其结合性都是从左到右的,优先级高于逻辑运算符,低于比较云悬浮,且从高到低依次为& ^ |


其他运算符

运算符 描述
sizeof sizeof 运算符返回变量的大小。例如,sizeof(a) 将返回 4,其中 a 是整数。
Condition ? X : Y 三目运算符。如果 Condition 为真 ? 则值为 X : 否则值为 Y。
, 逗号运算符会顺序执行一系列运算。整个逗号表达式的值是以逗号分隔的列表中的最后一个表达式的值。
.(点)和 ->(箭头) 成员运算符用于引用类、结构和共用体的成员。
Cast 强制转换运算符把一种数据类型转换为另一种数据类型。例如,int(2.2000) 将返回 2。
& 指针运算符 &返回变量的地址。例如 &a; 将给出变量的实际地址。
* 指针运算符 * 指向一个变量。例如,*var; 将指向变量 var。

运算符的优先级

●  一般来说,一元运算符优先级高于对应的二元运算符

●  弄不清楚优先级就加括号


运算符代码演示:

# include <assert.h>
# include <iostream>
using namespace std;
int main()
{
    int A = 10;
    int B = 20;
    // 算术运算符示例
    std::cout << A + B << std::endl;    // 30
    cout << A - B << endl;     // -10
    cout << A*B << endl;       // 200
    cout << B / A << endl;     // 2
    cout << 15 / A << endl;    // 1
    cout << 15.0 / A << endl; // 1.5
    cout << B%A << endl;      // 0
    cout << ++A << endl;      // 11
    cout << A << endl;           // 11
    cout << --A << endl;        // 10
    cout << A << endl;           // 10
    // 关系运算符示例
    cout << (A == B) << endl;  // 0
    cout << (A != B) << endl;   // 1
    cout << (A > B) << endl;    // 0
    cout << (A < B) << endl;    // 1
    cout << (A >= B) << endl;  // 0
    cout << (A <= B) << endl;  // 1
    // 逻辑运算符示例
    bool bA = false, bB = true;
    // to be or not to be
    cout << ( bA == true || bA != true ) << endl;   // 1
    cout << (bB == true || bB != true) << endl;   // 1
    // 德*摩根率
    assert( !(bA || bB) == (!bA && !bB) ); //  1
    assert( !(bA && bB) == (!bA || !bB)  ); //  1
    bA = false, bB = false;
    assert( !(bA || bB) == (!bA && !bB) ); //  1
    assert( !(bA && bB) == (!bA || !bB) ); //  1
    bA = true, bB = true;
    assert( !(bA || bB) == (!bA && !bB) ); //  1
    assert( !(bA && bB) == (!bA || !bB) ); //  1
    bA = true, bB = false;
    assert( !(bA || bB) == (!bA && !bB) ); //  1
    assert( !(bA && bB) == (!bA || !bB) ); //  1
    // 位运算
    cout << (A&B) << endl;      //  01010 & 10100  = 00000  ==> 0
    cout << (A | B) << endl;       //  01010 | 10100  = 11110  ==> 30
    cout << (A^B) << endl;     //  01010 ^ 10100  = 11110 ==> 30
    cout << (~A) << endl;       //  ~0000000000001010 = 11111111111110101   ==> 0000000000001011 ==》 -11
    cout << (A << 2) << endl;  // 00001010 << 2 ==> 00101000‬ ==> 40
    cout << (A >> 2) << endl;// 00001010 >> 2 ==> 00000010 ==> 2
    // 赋值运算符
    int C = A + B;
    cout << C << endl;
    C += A;
    cout << C << endl;
    C -= A;
    cout << C << endl;
    C *= A;
    cout << C << endl;
    C /= A;
    cout << C << endl;
    C %= A;
    cout << C << endl;
    C <<= A;
    cout << C << endl;
    C >>= A;
    cout << C << endl;
    C &= A;
    cout << C << endl;
    C ^= A;
    cout << C << endl;
    C |= A;
    cout << C << endl;
    // 杂项运算符示例
    cout << sizeof(A) << endl;   // 4
    int M = A > B ? 1 : 0;
    cout << M << endl;              // 0
    int D = A < B ? 1 : 0;
    cout << D << endl;              // 1
    int E = (A, B, M);
    cout << E << endl;               // 0
    float F = float(E);
    cout << F << endl;               // 0
    cout << &F << endl;            // 0xFFXXX
    float * P = &F;
    cout << P << endl;              // 0xFFXXX
    cout << *P << endl;             // 0
    typedef struct {
      short Sunday = 0;
      short Monday = 1;
      short Tuesday = 2;
      short Wednesday = 3;
      short Thursday = 4;
      short Friday = 5;
      short Saturday = 6;
    } Week;
    Week w;
    cout << w.Friday << endl;  // 5
    cout << sizeof(w) << endl; // ?
    return 0;
}

测试的时候按项测试。


语言的注释

●  程序注释是解释性语句,你可以在C++代码中包含注释,这将提高源代码可读性。所有的编程语言都允许某种形式的注释

●  C++支持单行注释和多行注释,注释中所有的字符会被C++编译器忽略

#include <iostream>
using namespace std;
/***
 * 多行注释
 * @return
 */
int main()
{
    cout << "单行注释" << std::endl;
    return 0;
}

注释的原则和建议:


●  好的命名和代码本身就是最好的注释;如果代码本身就清除不需要增加额外的注释

●  在重要的代码段或者复杂的代码段先写注释再写代码,这样思路清晰,同时可以保证代码和注释一致性

●  注释不是越多越好,它对代码提示,如果要写就要清楚,并且保证和代码一致性,如果更新了新代码,请更新相应的注释


补码的补充

有没有一个办法使用加法来做减法?


机器数和真值

●  机器数:

一个数在计算机中的二进制表示形式,叫做这个数的机器数。


机器数是带符号的,在计算机中用一个数的最高位存放符号位,正数为0,负数为1


比如 +3 就是 00000000000000000000000000000011


-3 按照道理就是10000000000000000000000000000011


实际上-3是 11111111111111111111111111111101


●  真值:


真正的数学意义上的数值。


因为第一位是符号位,所以机器数形式值就不等于真正的数值


00000000000000000000000000000011 --> +3


11111111111111111111111111111101 --> -3


无符号数的编码

屏幕截图 2023-08-04 112028.png

0001 = 1

0101 = 5


有符号数的编码

屏幕截图 2023-08-04 111941.png

0001 = 1

0101 = 5

1011 = -5

1111 = -1

另外一种计算补码的方法:

对正数:直接按位计算权重之和

对负数:保留符号位,对后面每位取反+1


字节序

一个字(32位机器采用32bits字长4bytes)在内存中如何以byte来存放


两个传统:

●  大端法:大多数IBM机器,Internet传输

●  小端法:Intel兼容机

using namespace std;
// 二进制转无符号整型
unsigned int B2U(unsigned int num)
{
    return (unsigned int)(num);
}
// 二进制转有符号整型
int B2T(int num)
{
    return (int)(num);
}
int main()
{
    //  补码机器数调整展示
    int i1 = 0;
    int i2 = -1;
    int i3 = -2147483648;
    int i4 = 2147483647;
    unsigned int  u1 = 0;
    unsigned int  u2 = 4294967295;
    unsigned int  u3 = 2147483648;
    unsigned int  u4 = 2147483647;
    // 补码的真值验证
    cout << B2T(0x00000000) << endl;
    cout << B2T(0xFFFFFFFF) << endl;
    cout << B2T(0x80000000) << endl;
    cout << B2T(0x7FFFFFFF) << endl;
    cout << endl;
    cout << B2U(0x00000000) << endl;
    cout << B2U(0xFFFFFFFF) << endl;
    cout << B2U(0x80000000) << endl;
    cout << B2U(0x7FFFFFFF) << endl;
    return 0;
}

我们在设计软件系统时总是希望软件系统尽可能的简单通用,于是人们希望在只有加法运算器的情况下设计一种方法能计算减法。


位运算补充

●  左移比较简单:

●  右移运算会有两种情况:

   ○  逻辑右移:移走的位填充0;

   ○  算数右移:移走的位填充与符号位有关,负数填充1;


对于有符号的数,尽可能不适用右移运算 因为到底是逻辑右移还是算数右移取决于编译器。

目录
相关文章
|
27天前
|
算法 编译器 C++
【C++11】lambda表达式
C++11 引入了 Lambda 表达式,这是一种定义匿名函数的方式,极大提升了代码的简洁性和可维护性。本文详细介绍了 Lambda 表达式的语法、捕获机制及应用场景,包括在标准算法、排序和事件回调中的使用,以及高级特性如捕获 `this` 指针和可变 Lambda 表达式。通过这些内容,读者可以全面掌握 Lambda 表达式,提升 C++ 编程技能。
64 3
|
3月前
|
C++
【C++基础】运算符详解
这篇文章详细解释了C++中运算符的用法,包括算术运算符、赋值运算符、比较运算符和逻辑运算符,以及它们在表达式中的作用和示例。
47 2
|
3月前
|
算法 编译器 程序员
C++ 11新特性之Lambda表达式
C++ 11新特性之Lambda表达式
19 0
|
4月前
|
C++
c++学习笔记02 运算符
C++学习笔记,介绍了C++中的运算符,包括基本的加减乘除、求模、前后置递增递减、赋值运算符、比较运算符和逻辑运算符的使用及其注意事项。
44 6
|
5月前
|
安全 编译器 C++
C++一分钟之-泛型Lambda表达式
【7月更文挑战第16天】C++14引入泛型lambda,允许lambda接受任意类型参数,如`[](auto a, auto b) { return a + b; }`。但这也带来类型推导失败、隐式转换和模板参数推导等问题。要避免这些问题,可以明确类型约束、限制隐式转换或显式指定模板参数。示例中,`safeAdd` lambda使用`static_assert`确保只对算术类型执行,展示了一种安全使用泛型lambda的方法。
72 1
|
6月前
|
算法 编译器 C++
C++一分钟之—Lambda表达式初探
【6月更文挑战第22天】C++的Lambda表达式是匿名函数的快捷方式,增强函数式编程能力。基本语法:`[capture](params) -&gt; ret_type { body }`。例如,简单的加法lambda:`[](int a, int b) { return a + b; }`。Lambda可用于捕获外部变量(值/引用),作为函数参数,如在`std::sort`中定制比较。注意点包括正确使用捕获列表、`mutable`关键字和返回类型推导。通过实践和理解这些概念,可以写出更简洁高效的C++代码。
57 13
|
6月前
|
C++
C++之运算符
C++之运算符
|
6月前
|
C++
C++语言的lambda表达式
C++从函数对象到lambda表达式以及操作参数化
|
6月前
|
安全 程序员 C++
C++一分钟之-重载运算符
【6月更文挑战第21天】C++的运算符重载让程序员能为自定义类型定制运算符行为,增强代码表达力。但要注意清晰性、优先级和返回类型。遵循运算符原有意义,充分测试,并用注释解释非直观设计。示例展示了如何为复数类重载`+`运算符。避免重载内置类型,注意结合性,且慎用隐式转换。重载应提升可读性而非复杂化代码。
50 2
|
6月前
|
C++
C++一分钟之-理解C++的运算符与表达式
【6月更文挑战第18C++的运算符和表达式构成了编程的基础,涉及数学计算、逻辑判断、对象操作和内存管理。算术、关系、逻辑、位、赋值运算符各有用途,如`+`、`-`做加减,`==`、`!=`做比较。理解运算符优先级和结合律至关重要。常见错误包括优先级混淆、整数除法截断、逻辑运算符误用和位运算误解。解决策略包括明确优先级、确保浮点数除法、正确使用逻辑运算符和谨慎进行位运算。通过实例代码学习,如 `(a &gt; b) ? &quot;greater&quot; : &quot;not greater&quot;`,能够帮助更好地理解和应用这些概念。掌握这些基础知识是编写高效、清晰C++代码的关键。
45 3