06. 运算符与表达式

一、运算符与表达式

  运算符 就是对字面量或者变量进行操作的符号;表达式 是指用运算符把字面量或者变量连接起来,符合 C 语言语法的式子。不同运算符连接的表达式体现的是不同类型的表达式;每个表达式都有一个值。要想获取这个值,必须根据运算符优先级的顺序来执行操作。

二、算数运算符

  数字类型进行运算时,数据类型不一样是不能进行运算的,需要转换成一样的类型,才能运算;当容量小的数据类型的变量与容量大的数据类型的变量做运算时,结果自动提升为容量大的数据类型(此时的容量大小指的是,表示数的范围的大和小)。

算数运算符 运算 范例 结果
+ 5 + 2
'a' + 5
7
'f'
- 5 - 2 3
* 5 * 2 10
/ 5 / 2
5.0 / 2
2
2.5
% 取模(取余) 5  % 2 1
#include <stdio.h>

int main(void)
{
    int a = 5;
    int b = 2;
    char ch = 'a';
    double c = 5;

    printf("%d + %d =  %d\n",a,b,a+b);      // 5 + 2 = 7
    printf("%d - %d =  %d\n",a,b,a-b);      // 5 - 2 = 3
    printf("%d * %d =  %d\n",a,b,a*b);      // 5 * 2 = 10
    printf("%d / %d =  %d\n",a,b,a/b);      // 5 / 2 = 2
    printf("%d %% %d =  %d\n",a,b,a%b);     // 5 % 2 = 1

    printf("%c + %d = %c\n",ch,a,ch+a);     // a + 5 = f
    printf("%lf / %d = %lf\n",c,b,c/b);     // 5.000000 / 2 = 2.500000

    return 0;
}

在代码中,如果有小数参与运算,其结果可能是不精确的;

当参与 / 运算的两个操作数都是整数时,表示整数除法;否则,表示浮点除法;

float 与 double 不能进行取余运算;

C99 规定 “趋零截断”,即,% 取模运算结果的正负与被模数一致,即:a % b 等价于 a - a/ b * b,取模结果的正负与 a 的正负一致;

字符 + 字符 或者 字符 + 数字 的时候,会把字符通过 ASCII 码表 查询到对应的数字再进行计算;

三、自增自减运算符

  ++ 和 -- 即可以放在变量的前面,也可以放在变量的后面;++ 和 -- 无论是放在变量的前面还是后面,单独写一行的结果是一样;如果 前 ++前 -- 参与运算的话,那 先用后加(减);如果 后 ++后 -- 参与运算的话,那 先加(减)后用

自增自减运算符 运算 范例 结果
++ 变量的值加 1 a = 2; b = ++a;
a = 2;b = a++;
a = 3; b = 3;
a = 3; b = 2;
-- 变量的值减 1 a = 2; b = --a;
a = 2; b = a--;
a = 1; b = 1;
a = 1; b = 2;
#include <stdio.h>

int main(void)
{
    int a = 2;
    int b;
  
    // 运算规则等价于 a=a+1; int b=a;
    b = ++a;
    printf("前++\n");
    printf("a = %d\n",a);   // 3
    printf("b = %d\n",b);   // 3
    putchar('\n');

    // 运算规则等价于 int b=a; a=a+1;
    a = 2;
    b = a++;
    printf("后++\n");
    printf("a = %d\n",a);   // 3
    printf("b = %d\n",b);   // 2
    putchar('\n');

    // 运算规则等价于 a=a-1; int b=a;
    a = 2;
    b = --a;
    printf("前--\n");
    printf("a = %d\n",a);   // 1
    printf("b = %d\n",b);   // 1
    putchar('\n');

    // 运算规则等价于 int b=a; a=a-1;
    a = 2;
    b = a--;
    printf("后--\n");
    printf("a = %d\n",a);   // 1
    printf("b = %d\n",b);   // 2
    putchar('\n');

    return 0;
}

自增和自减运算符只能影响一个变量,或者说,只能影响一个可修改的左值;

四、赋值运算符

  赋值运算符 用于将表达式的值赋值给变量。

赋值运算符 运算 示例
= 赋值 a = 10
+= 加后赋值 a += b 等价于 a = a + b
-= 减后赋值 a -= b 等价于 a = a - b
*= 乘后赋值 a *= b 等价于 a = a * b
/= 除后赋值 a /= b 等价于 a = a / b
%= 取余后赋值 a %= b 等价于 a = a % b
<<= 左移后赋值 a <<= b 等价于 a = a << b
>>= 右移后赋值 a >>= 3 等价于 a = a >> b
&= 按位与后赋值 a &= b 等价于 a = a & b
|= 按位或后赋值 a| b 等价于 a = a | b
^= 按位异或后赋值 a ^ b 等价于 a = a ^ b
#include <stdio.h>

int main(void)
{
    int a = 10;
    int b = 5;

    a += b;
    printf("a =  %d\n",a);      // 15
    printf("b = %d\n",b);       // 5
    putchar('\n');

    a = 10;
    b = 5;

    a -= b;
    printf("a =  %d\n",a);      // 5
    printf("b = %d\n",b);       // 5
    putchar('\n');

    a = 10;
    b = 5;

    a *= b;
    printf("a =  %d\n",a);      // 50
    printf("b = %d\n",b);       // 5
    putchar('\n');

    a = 10;
    b = 5;

    a /= b;
    printf("a =  %d\n",a);      // 2
    printf("b = %d\n",b);       // 5
    putchar('\n');

    a = 10;
    b = 5;

    a %= b;
    printf("a =  %d\n",a);      // 0
    printf("b = %d\n",b);       // 5
    putchar('\n');

    return 0;
}

五、关系运算符

  关系运算符 也被称为 比较运算符;关系运算符 用于表达式的比较,并返回一个真值或假值。在 C 语言中,0 为假,非 0 为真

关系运算符 运算 范例 结果
== 相等 4 == 3 0
!= 不等于 4 != 3 1
< 小于 4 < 3 0
> 大于 4 > 3 1
<= 小于等于 4 <= 3 0
>= 大于等于 4 >= 3 1
#include <stdio.h>

int main(void)
{
    int a = 4;
    int b = 3;

    printf("%d == %d: %d\n",a,b,(a==b));    // 4 == 3: 0
    printf("%d != %d: %d\n",a,b,(a!=b));    // 4 != 3: 1
    printf("%d < %d: %d\n",a,b,(a<b));      // 4 < 3: 0
    printf("%d <= %d: %d\n",a,b,(a<=b));    // 4 <= 3: 0
    printf("%d > %d: %d\n",a,b,(a>b));      // 4 > 3: 1
    printf("%d >= %d: %d\n",a,b,(a>=b));    // 4 >= 3: 1

    return 0;
}

六、逻辑运算符

  逻辑运算符 用于根据表达式的值返回真值或假值。在 C 语言中,0 为假,非 0 为真

逻辑运算符 功能
&& 逻辑与,全真为真,有假为假
|| 逻辑或,有真为真,全假为假
! 逻辑非,非假为真,非真为假

  运算结果:

a b a && b a|| b !a
1 1 1 1 0
1 0 0 0 0
0 1 0 0 1
0 0 0 0 1
#include <stdio.h>

int main(void)
{
    int a = 1;
    int b = 1;
    int c = 0;
    int d = 0;

    // 逻辑与,两边都为真,结果才为真
    printf("%d && %d: %d\n",a,b,(a&&b));    // 1 && 1: 1
    printf("%d && %d: %d\n",a,c,(a&&c));    // 1 && 0: 0
    printf("%d && %d: %d\n",c,d,(c&&d));    // 0 && 0: 0
    printf("\n");

    // 逻辑或,两边都为假,结果才为假
    printf("%d || %d: %d\n",a,b,(a||b));    // 1 || 1: 1
    printf("%d || %d: %d\n",a,c,(a||c));    // 1 || 0: 1
    printf("%d || %d: %d\n",c,d,(c||d));    // 0 || 0: 0
    printf("\n");

    // 逻辑非
    printf("!%d: %d\n",a,!a);               // !1 = 0
    printf("!%d: %d\n",c,!c);               // !0 = 1
    printf("\n");

    return 0;
}

  短路与逻辑运算符 和 逻辑或运算符 具有短路效果,即当左边的表达式能确定最终结果时,那么右边就不会参与运算了;

  • 逻辑与运算符;当符号左边是 true 时,&& 会执行右边的运算;当符号左边是 false 时,&& 不再执行符号右边的运算;
  • 逻辑或运算符:当符号左边是 false 时,|| 会执行右边的运算;当符号左边是 true 时,|| 不再执行右边的运算;
#include <stdio.h>

int main(void)
{
    // 短路逻辑运算符具有短路效果
    // 当左边的表达式能确定最终结果时,那么右边就不会参与运算了
    int a = 10;
    int b = 10;
    int c;
  
    c = (++a < 5) && (++b < 5);
    printf("(++a < 5) && (++b <5): %d\n",c);    // 0
    printf("a = %d\n",a);                       // 11
    printf("b = %d\n",b);                       // 10
    printf("\n");

    a = 10;
    b = 10;

    c = (++a > 5) || (++b > 5);
    printf("(++a > 5) || (++b >5): %d\n",c);    // 1
    printf("a = %d\n",a);                       // 11
    printf("b = %d\n",b);                       // 10
    printf("\n");

    return 0;
}

七、位运算符

  位运算符是直接对整数的二进制进行的运算; 所有数字在计算机底层都以二进制形式执行;所有的整数值底层都以补码的方式存储;

  • 正数:三码合一,符号位为 0;
  • 负数:符号位为 1
    • 原码:直接将数值转成二进制数,最高位是符号位;
    • 反码:对原码按位取反,符号位不变;
    • 补码:其反码加 1;
位运算符 运算 范例
<< 左移,空位补 0,被移除的高位舍弃 3 << 2 = 12
>> 右移,空位使用符号位填充,被移除的低位舍弃 3 >> 1 = 1
& 与运算,二进制位进行与运算,全1为1,有0为0 6 & 3 = 2
| 或运算,二进制位进行或运算,有1为1,全0为0 6| 3 = 7
^ 异或运算,二进制位进行异或运算,相同为0,不同为1 6 ^ 3 = 5
~ 按位取反,二进制位进行取反运算,非0为1,非1为0 ~6 = -7
#include <stdio.h>

int main(void)
{
    printf("3 << 2: %d\n",(3 << 2));        // 12
    printf("3 >> 1: %d\n",(3 >> 1));        // 1
    printf("-3 >> 1: %d\n",(-3 >> 1));      // -2
    printf("6 & 3: %d\n",(6 & 3));          // 2
    printf("6 | 3: %d\n",(6 | 3));          // 7
    printf("6 ^ 3: %d\n",(6 ^ 3));          // 5
    printf("~6: %d\n",(~6));                // -7

    return 0;
}

<<:在一定范围内,每向左移一位,相当于乘以 2;

>>:在一定范围内,每向右移一位,相当于除以 2;

我们可以通过 x | (1 << (y - 1)) 的方式将第 y 位 置1

我们可以通过 x & (~(1 << (y-1))) 的方式将第 y 位 清零

八、三元运算符

  三元运算符的格式如下:

(条件表达式) ? 表达式1 : 表达式2;

  它的执行流程如下:

  • 三元运算符在执行时,会先对条件表达式进行求值判断
    • 如果条件表达式结果为 1,运算后的结果是 表达式1
    • 如果条件表达式结果为 0,运算后的结果是 表达式2
#include <stdio.h>

int main(void)
{
    int a = 10;
    int b = 20;

    // 使用三元运算符获取两个整数的较大值
    printf("%d",((a > b) ? a : b));     // 20

    return 0;
}

三元运算符结果一定要使用,不能单独写在一行;

三元运算符的表达式 1 和表达式 2 为同种类型

三元运算符一定可以改成 if-else 语句,反之不成立;

如果程序既可以使用三元运算符又可以使用 if-else 语句,优先使用三元运算符;

九、逗号运算符

  逗号表达式就是用逗号隔开的一串表达式,逗号表达式的特点是是从左向有依次计算的,整个表达式的结果是最后一个表达式的结果。

#include <stdio.h>

int main(void)
{
    int a = 0;
    int b = 3;
    int c = 5;
    int d = (a=b+2, c=a-4 , b=c+2);	// a=5,c=1,b=3

    printf("%d\n",d);

    return 0;
}

十、sizeof运算符

  sizeof 运算符以字节位单位带返回运算对象的大小。运算对象可以是具体的数据对象(如,变量名)或类型(如,int)如果运算对象是类型,则必须用圆括号将其括起来。

  C 语言规定,sizeof 返回 size_t 类型的值。这时候一个无符号整数类型,但它不是新类型。在 C 头文件中,使用 typedef 把 size_t 作为 unsigned int 或 unsigned long 的别名。这样,在使用 size_t 类型时,编译器会根据不同的系统替换标准类型。

  C99 做了进一步调整,新增了 %zd 和 %zu 转换说明用于 printf() 显示 size_t 类型的值。如果系统不支持 %zd 和 %zu,可使用 %u 或 %lu 代替。

#include <stdio.h>

int main(void)
{
    char str[] = "hello world!";

    printf("char: %zu\n",sizeof(char));                         // 1
    printf("short: %zu\n",sizeof(short));                       // 2
    printf("int: %zu\n",sizeof(int));                           // 4
    printf("long: %zu\n",sizeof(long));                         // 4
    printf("long long: %zu\n",sizeof(long long));               // 8
    printf("float: %zu\n",sizeof(float));                       // 4
    printf("double: %zu\n",sizeof(double));                     // 8
    printf("long double: %zu\n",sizeof(long double));           // 16

    printf("str: %zu\n",sizeof str);                            // 13
    printf("integer: %zu\n",sizeof 10);                         // 4
    printf("decimals: %zu\n",sizeof 3.14);                      // 8
  
    return 0;
}

不同的操作系统环境下,使用 sizeof 运算符得到的结果可能不同;

sizeof 何时使用圆括号取决于运算对象是类型还是特定量;运算对象是类型时,圆括号必不可少,但是对于特定量,可有可无;

十一、运算符的优先级

  我们可以使用小括号来提升运算符的优先级

优先级 运算符 名称或含义 使用形式 结合方向 说明
1 [] 数组下标 数组名[常量表达式] 左到右
() 圆括号 (表达式)/函数名(形参表)
. 成员选择(对象) 对象.成员名
-> 成员选择(指针) 对象指针->成员名
2 + 正号运算符 +表达式 右到左 单目运算符
- 负号运算符 -表达式 单目运算符
(类型) 强制类型转换 (数据类型)表达式
++ 前置自增运算符 ++变量名 单目运算符
++ 后置自增运算符 变量名++ 单目运算符
-- 前置自减运算符 --变量名 单目运算符
-- 后置自减运算符 变量名-- 单目运算符
* 取值运算符 *指针变量 单目运算符
& 取地址运算符 &变量名 单目运算符
! 逻辑非运算符 !表达式 单目运算符
~ 按位取反运算符 ~表达式 单目运算符
sizeof 长度运算符 sizeof(表达式)
3 / 表达式/表达式 左到右 双目运算符
* 表达式*表达式 双目运算符
% 余数(取模) 整型表达式/整型表达式 双目运算符
4 + 表达式+表达式 左到右 双目运算符
- 表达式-表达式 双目运算符
5 << 左移 变量<<表达式 左到右 双目运算符
>> 右移 变量>>表达式 双目运算符
6 > 大于 表达式>表达式 左到右 双目运算符
>= 大于等于 表达式>=表达式 双目运算符
< 小于 表达式<表达式 双目运算符
<= 小于等于 表达式<=表达式 双目运算符
7 == 等于 表达式==表达式 左到右 双目运算符
!= 不等于 表达式!=表达式 双目运算符
8 & 按位与 表达式&表达式 左到右 双目运算符
9 ^ 按位异或 表达式^表达式 左到右 双目运算符
10 | 按位或 表达式|表达式 左到右 双目运算符
11 && 逻辑与 表达式&&表达式 左到右 双目运算符
12 || 逻辑或 表达式||表达式 左到右 双目运算符
13 ?: 条件运算符 表达式1?表达式2:表达式3 右到左 三目运算符
14 = 赋值运算符 变量=表达式 右到左
/= 除后赋值 变量/=表达式
*= 乘后赋值 变量*=表达式
%= 取模后赋值 变量%=表达式
+= 加后赋值 变量+=表达式
-= 减后赋值 变量-=表达式
<<= 左移后赋值 变量<<=表达式
>>= 右移后赋值 变量>>=表达式
&= 按位与后赋值 变量&=表达式
^= 按位异或后赋值 变量^=表达式
|= 按位或后赋值 变量|=表达式
15 , 逗号运算符 表达式,表达式,… 左到右 从左向右顺序运算
posted @ 2023-02-23 12:31  星光映梦  阅读(56)  评论(0编辑  收藏  举报