位段及位运算

1.位段

1.1定义

当结构体或共用体中包含无符号整形或有符号整形时,C语言规定允许用户指定这些成员所占的存储位。这就是位段(或称位域)

1.2声明方式

位段的声明和结构是类似的,有两个不同:

1.2.1位段的成员必须是int. unsigned int或signed int .

1.2.2位段的成员名后边有一个冒号和一个数字。

#include <stdio.h>

void test01()
{
	struct bit_size{
		//一个int类型是32bit 
		int a:2;	//冒号后面指定所占位数 
		int b:5;
		int c:10;
		int d:30;
	};
	struct bit_size bit01;
	bit01.a=3;	//只有2bit ; 最大为3(00 01 10 11 )
	bit01.b=0x1F;	//最大为31 
	bit01.c=0x3FE;	//最大为1777 
	bit01.d=0x3FFFFFFF;
	
	unsigned char *p=&bit01;	 
	//1111111111  11111  11
	printf("*p=%u\n",*p);//1 11111111 01111111
	printf("%#p\n",p);
	printf("%#p\n",p+1);
	
	bit01.a=1;
	printf("*p=%u\n",*p);//1 11111111 01111101
	printf("*p+1=%u\n",*(p+1)); //1 11111111 01111101
	
	bit01.c=0x2FE;
	printf("*p+1=%u\n",*(p+1)); //1 01111111 01111101 打印中间的 
	printf("*p+2=%u\n",*(p+2)); //1 01111111 01111101 打印的是1 
	printf("*p+3=%u\n",*(p+3)); 
	printf("*p+4=%u\n",*(p+4));
	printf("*p+5=%u\n",*(p+5));
	printf("*p+6=%u\n",*(p+6));
	printf("*p+7=%u\n",*(p+7));
}
 
int main()
{
	
	test01();
	return 0;
 } 

1.3位段的跨平台问题

1.3.1. int位段被当成有符号数还是无符号数是不确定的。

1.3.⒉.位段中最大位的数目不能确定。(16位机器最大16,32位机器最大32,写成27,在16位机器会出问题。

1.3.3.位段中的成员在内存中从左向右分配,还是从右向左分配标准尚未定义。(大端、小端)

1.3.4.当一个结构包含两个位段,第二个位段成员比较大,无法容纳于第一个位段剩余的位时,是舍弃剩余的位还是利用,这是不确定的。

        跟结构相比,位段可以达到同样的效果,但是可以很好的节省空间,但是有跨平台的问题存在。

2.位运算

        位运算符用于处理有符号或无符号的整形操作数的各个数位,位运算通常将数据当做无符号数进行处理

2.1常见位运算符

&按位与当两个操作数对应位必须都是1,结果对应位才是1
|按位或当两个操作数对应位至少有一个是1,结果对应位就是1
^按位异或当两个操作数对应位必须不同,结果对应位才是1
<<左移第一个操作数按位左移,移动的位数由第二个操作数指定,右边腾空补0(扩大2^n)
>>右移第一个操作数按位右移,移动的位数由第二个操作数指定,右边腾空补0(缩小2/n)
~按位取反操作数中各个位0变1,1变0

2.2注意

1)移位操作时,需要考虑移位后的结果要在数据类型表示的范围内,否则会出错。

2)负数的右移结果取决于编译器(最高位补0还是1)

3)移动位数大于操作数的存储空间位数或者是负数,移位结果没有被定义

2.3使用

#include <stdio.h>

int main()
{
	printf("%#x\n",65525&1); //
	printf("%#x\n",15|241); //1111  | 1111 0001		
	printf("%#x\n",139^199); //1000 1011   1100 0111   0100 1100
	printf("%#x\n",~21845);	 //0101 0101 0101 0101 取反 1010 1010 1010 1010 
	
	return 0;
 } 

练习:

1.请编写一个程序将一个整型变量右移4位,并以二进制的形式输出移位前和移位后的数值,看看你的系统是补0还是补1?

#include <stdio.h>
/*打印一个整数,右移4位,打印移动前移动后的二进制数*/
int main()
{
	unsigned int num = 0x87654321;
	unsigned int car = 0x1 << 31;
	int i;
	for(i = 1; i < 33; i++){
		car&num ? putchar('1') : putchar('0');
		num=num<<1;
	}
	printf("\n");
	num=-0x87654321;
	num=num>>4;
	for(i = 1; i < 33; i++){
		car&num ? putchar('1') : putchar('0');
		num=num<<1;
	}
	return 0;
}

2.无符号数左移移位相当于该数乘以2,编写一个函数,接收两个整型变量num和pow作为实参,被掉函数使用移位运算计算num*2\^pow的结果。然后分别以整型和二进制形式输出

#include <stdio.h>

void test(unsigned int num,unsigned int pow)
{
	num = num<<1;
	num=num^pow;
	int i;
	printf("%d\n",num);
	unsigned int car=0x1<<31;
	for(i=1;i<33;i++){
		num&car ? putchar('1') : putchar('0');
		num=num<<1;
	}
	
}
//234 324  0001 1101 0100   0001 0100 0100   
int main()
{
	unsigned int num,pow;
	printf("请输入两个整数(回车分隔):"); 
	scanf("%d %d",&num,&pow); 
	test(num,pow);
	return 0;
 } 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值