C for ios --- 位运算

来源:互联网 发布:centos镜像配置yum源 编辑:程序博客网 时间:2024/06/08 05:11

位运算符介绍(一)


1.什么是位运算符?

  • 位运算是指按二进制进行的运算。在系统软件中,常常需要处理二进制位的问题。 C语言��供了6个位操作运算符。这些运算符只能用于整型操作数,即只能用于带符号或无符号的 char,short,int与long类型。

2.位运算符与

  • & 按位与

    • 只有对应的两个二进位均为1时,结果位才为1,否则为0
    • 口诀: 同1为1
  • 示例:

~~ 9&5 = 1~~ ~~  1001~~ &0101~~ ------~~  0001
  • 规律

    • 二进制中,与1相&就保持原位,与0相&就为0
  • 应用场景:

    • 按位与运算通常用来对某些位清0或保留某些位。例如把a的高位都清0,保留低八位,那么就a&255
    • 判断奇偶: 将变量a与1做位与运算,若结果是1,则 a是奇数;若结果是0,则 a是偶数
    • 任何数和1进行&操作,得到这个数的最低位
~~   1001~~ & 0001~~ ------------~~   0001 ```'' + 想把某一位置0

~~ 11111111
~~ & 11111011
~~ ————
~~ 11111011

---## 3.位运算符或- | 按位或    + 只要对应的二个二进位有一个为1时,结果位就为1,否则为0- 示例 ```~~ 9|5 = 13~~ ~~  1001~~ |0101~~ ------~~  1101 ```---## 4.位运算符异或- ^ 按位异或    + 当对应的二进位相异(不相同)时,结果为1,否则为0- 示例

~~ 9^5 = 12
~~
~~ 1001
~~ ^0101
~~ ——
~~ 1100
“`

  • 规律
    • 相同整数相^的结果是0。比如5^5=0

* 任何数异或另一个数两次,还是原数本身

5.位运算符取反

  • \~ 取反

    • 各二进位进行取反(0变1,1变0)
    • 多个整数相^的结果跟顺序无关。比如5^6^7=5^7^6
    • 因此得出结论:a^b^a = b
  • 示例

~~ ~9 =-10~~ ~~ 9的原码:0000 0000 0000 0000 0000 1001~~    反码:1111 1111 1111 1111 1111 0110~~ ~~ 知道补码求原码:也是符号位不变,其他各位取反+1~~ 1111 1111 1111 1111 1111 0110~~ 取反~~ 1000 0000 0000 0000 0000 1001~~ +1~~ 1000 0000 0000 0000 0000 1010 // -10~~ 

6.练习

  • 判断一个数的奇偶性,如果是奇数则打印“奇数”,否则打 印“偶数”
~~ int a = num&1;~~ If(a){~~     printf("奇数");~~ }else{~~     printf("偶数");~~ } ```- 面试题使用位运算实现交换两个数的值 ```    ////  main.c//  位运算符demo////  Created by D9ingCompany on 2017/3/24.//  Copyright © 2017年 D9ingCompany. All rights reserved.///** swap ab value @param argc <#argc description#> @param argv <#argv description#> @return <#return value description#> */#include <stdio.h>void swapNormal(int,int);void swapBigger(int,int);void swapSupper(int,int);int main(int argc, const char * argv[]) {    int a = 5;    int b = 10;    printf("交换前a = %i,b = %i\n",a,b);    swapSupper(a, b);//    swapNormal(a, b);//    swapBigger(a, b);    return 0;}/** 位运算交换 @param a <#a description#> @param b <#b description#> */void swapSupper(int a,int b ){    a = a^b;    b = a^b;    a = a^b;    printf("a = %i,b = %i\n",a,b);}/** 运算交换 @param a <#a description#> @param b <#b description#> */void swapBigger(int a, int b ){    a = a+b;    b = a-b;    a = a-b;    printf("a = %i,b = %i\n",a,b);}/** 容器交换 @param a <#a description#> @param b <#b description#> */void swapNormal(int a,int b){    int tmp = a;    a = b;    b = tmp;    printf("a = %i,b = %i\n",a,b);}

位运算符介绍(二)

1.左移位运算符

  • 把整数a的各二进位全部左移n位,高位丢弃,低位补0。左移n位其实就是乘以2的n次方

    • 由于左移是丢弃最高位,0补最低位,所以符号位也会被丢弃,左移出来的结果值可能会改变正负性
  • 示例

~~ 2<<1; //相当于 2 *= 2 // 4~~ 2<<2; //相当于 2 *= 2^2; // 8
  • 应用场景
    • 快速计算一个数乘以2的n次方
    • 左移动就是乘以2的n次方

2.右移位运算符

  • 把整数a的各二进位全部右移n位,保持符号位不变。右移n位其实就是除以2的n次方

    • 为正数时, 符号位为0,最高位补0
    • 为负数时,符号位为1,最高位是补0或是补1
    • 取决于编译系统的规定
  • 示例

~~ 2>>1; //相当于 2 /= 2 // 1~~ 4>>2; //相当于 4 /= 2^2 // 1~~ 
  • 应用场景
    • 快速计算一个数除以2的n次方

+ 右移就是一个数除以2的n次方

负数的左移右移

  • 负数的左移右移是补码再移动,负数是以补码的形式存储在内存中

3.练习

  • 写一个函数把一个10进制数按照二进制格式输出
  • 分析
~~ 13 -----> 0000 0000 0000 0000 0000 0000 0000 1101~~ 0000 0000 0000 0000 0000 0000 0000 1101 13~~ 0000 0000 0000 0000 0000 0000 0000 0001 1~~ 每次取 一个数的最后一个二进制位~~ 任何一个数和1进行&(按位与)得到任何一个数的二进制的最后~~ 一位
  • 实现
~~     int len = sizeof(int)*8;~~     int temp;~~     for (int i=0; i<len; i++) {~~         temp = num; //每次都在原数的基础上进行移位运算~~         temp = temp>>(31-i); //每次移动的位数~~         int t = temp&1; //取出最后一位~~         if(i!=0&&i%4==0)printf(" "); printf("%d",t);~~     }

0 0