C和C指针小记(八)-操作符、左值右值

时间:2023-03-10 03:08:18
C和C指针小记(八)-操作符、左值右值

1、移位操作符

移位操作符分为左移操作符(<<)和右移操纵符(>>)

对于无符号数:左右位移操作都是逻辑位移

对于有符号数:到底是采用逻辑位移还是算术位移取决于编译器.如果一个出现使用了有符号数的右移操作,它就是不可移植的.

对于左移操作:右边空出来的位数用0补齐.

对于右移操作:如果是逻辑位移,左边移入的位用0填充;如果是算术位移,左边移入的位由原先该值的符号位决定,符号为为1则移入的位均为1,符号为为0则移入的位均为0.

注意:

a << -5;

这种形式的移位由编译器决定.它产生的效果是不可预知的.

2、位操作符

位操作符对它们的操作数的各个位执行与,或,异或,补等逻辑操作.

包括: AND OR XOR, 分别代表 与,或,异或,(补)

对应的操作符是: & | ^ ~

3、位操作的应用

3.1 统计输入字符中的字符个数,单词个数,换行符个数

 /*
* 统计输入的字符串中 行数,制表符,字符的个数,注意没有break,因为美出现一次换行符就表示单词个数和字符个数也增加了,每出现一次 空字符或制表符,字符个数也增加了
*/
int ch;
int lines = 0;
int words = 0;
int chars = 0;
while ((ch = getchar()) != EOF) {
switch (ch) {
case '\n':
lines += 1;
case ' ':
case '\t':
words += 1;
default:
chars += 1;
}
}

3.2 把一个整数的某位数置为1

 //把value1 的第五位数置为1
int value1 = 5;
printf("value1: %d ",value1);//5 b0101
value1 = value1 | 1 << 5;//把value1 的第五位数置为1
printf("value1: %d\n",value1);//37 b10101

3.3 把一个数的某位数置为0

    //把value2 的第5位数置为0
int value2 = 32;
printf("value2: %d ", value2);//32 b10000
value2 = value2 & ~( 1 << 5);
printf("value2: %d\n", value2);//0 b00000

3.4 测试整数的二进制序列某位是否为1

    //测试第4位是否为1
int value3 = 15;
printf("value3: %d\n", value3);//15
printf("value3 的第4位为 %d \n",(value3 & 1 << 4));//value3 的第4位为 0

3.5 返回函数参数值中值为1的位的个数

//返回函数参数值中值为1的位的个数
int count_one_bits(unsigned value) {
int ones;
for(ones = 0; value != 0; value >>= 1){
if ((value & 1) != 0) {
ones += 1;
}
}
return ones;
}

4、赋值操作符 =

需要注意的点:

a = x = y +3;

如果x是一个字符型变量,那么 y+3的值就会被截去一段,以便容纳与字符类型的变量中.那么a所赋的值就是被截取后的值.

还有之前提到的:

char ch;

while((ch = getchar()) !=EOF)

因为EOF需要的位数比字符型值所能提供的位数要多,这也是getchar返回一个整型值而不是字符值的原因.然而,把 getchar() 的返回值先存储与ch中将导致它被截断.

然后这个被截断的值被提升位整型并与EFO进行比较.这段存在错误的代码在使用有符号字符集的机器上运行时,如果读取了一个值位\377的字节时,循环将会终止.因为这个值截短再提升之后与EOF相等.当这段代码在使用无符号字符集的机器上运行时,这个循环将永远不会终止.

此外还有复合赋值符, 它可以是程序变得简洁

+= -= *= /= %=

<<= >>= &= ^= |=

5、单目操作符

! ++ - & sizeof
-- + * (类型)

! 逻辑反操作

\ ~ 求补操作

- 产生操作数的负值

+ 产生操作数的值,等于啥都不干

& 产生操作数的地址

* 简介访问,与指针一起使用,用于访问指针所指的值

sizeof 判断它的操作数的类型长度,以字节为单位.操作数可以是个表达式(常常是单个变量),也可以是两边加上括号的类型名.

如:sizeof(int) sizeof x.

注意:sizeof(a = b+1) 并没有向a赋任何值.

() 强制类型转换 (cast),用于显式的把表达式转换为另外的类型.

++

这两个表达式有前缀和后缀形式.一般和赋值操作符一起使用.

6、下标

array[n] 等价于 *(array + (n))

. 和 -> 都是用来访问一个结构的成员的.

如果s是一个结构变量,那么 s.a 就访问s中名叫a的成员.

当你拥有一个指向结构体的指针而不是结构本身,切欲访问它的成员时,就需要使用 -> 操作符,而不是. 操作符.

7、左值和右值

左值就是那些能够出现在赋值符号左边的东西,右值就是那些可以出现在赋值符号右边的东西.

左值可是一个变量,也可以是一个表达式.但大多住表达式都能作为左值.表达式作为左值时,它的含义必须是一个特定的内存位置.

例如:

a = b + 1;// a 就是一个左值

但是 b + 1 = a; // b+1就不能作为一个左值,

int a[30];

a[b + 10] = 0;//这里表达式就是一个左值

再如:

int a, *pi;

pi = &a;

*pi = 20;//这里 *p作为一个间接取值表达式 就是一个左值 它表示pi所指定来需要进行修改的位置.

int c = *pi;//这里 *p 的含义就是提取当前存储与这个位置的值.