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 >>= b 等价于 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);                                                          // d = 3
    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 | , | 逗号运算符 | 表达式,表达式,… | 左到右 | 从左向右顺序运算 | 
                    
                
                
            
        
浙公网安备 33010602011771号