计算机原码反码补码,加减乘除及常用位运算技巧

前言

程序中的所有数在计算机内存中都是以二进制的形式储存的,位运算就是直接对整数在内存中的二进制位进行操作, 所以位运算更能够高效率的完成数值的计算,也可以节约内存,程序在计算的时候所有的数值或者对象最终都要转化为二进制,计算机运算只有加法和位运算, 减法也是将数转成负数二进制的补码再相加取值, 乘法转换为加法运算,除法转换为减法运算(减法再转为加法运算)

一:原码,反码,补码与加减乘除运算

1:原码,反码与补码

正数的原码,反码,补码都一至.
负数原码为绝对值二进制最高位取1, 负数的反码是原码(符号位除外)按位取反, 负数补码是反码+1
如9的原码,反码,补码都是 00000000 00000000 00000000 00001001
-9 原码 10000000 00000000 00000000 00001001
-9的反码 11111111 11111111 11111111 11110110
-9的补码 11111111 11111111 11111111 11110111

2:加法运算(与十进制类似例如6+9)

6的二进制 00000000 00000000 00000000 00000110
9的二进制 00000000 00000000 00000000 00001001
相加结果 00000000 00000000 00000000 00001111 转成十进制就是15

3:减法运算,减法其实就是将减的数转成负数取补码相加,例如6-9

正6的二进制 00000000 00000000 00000000 00000110
-9的二进制(补码) 11111111 11111111 11111111 11110111
相加结果 11111111 11111111 11111111 11111101 // 这个数就是-3的二进制
减1成反码 11111111...11111100 取反 10000000 ... 00000011 就是-3的原码喽

4:乘法运算(通过左移化解成加法运算)

十进制中例如140 * 121 = 140 *(1 * 10^0 +2 * 10^1+1 * 10^2) = 140+2800+14000 = 16940,二进制也是一样,
算9 * 6, 6的二进制110, 即 9 * (0 * 2^0 + 1 * 2^1 + 1 * 2^2)位数为0的都等于0,分解出来就是 0 + (9 <<1) + (9<<2)
9的二进制1001 上面分解就等于 0+10010+100100 = 110110 十进制就是54

5:除法(与十进制除法相似从高往低)

如73 / 5 , 73二进制1001001 , 5二进制101
从第一位 1 < 101 结果为0, 余1
到第二位1 0 <101结果为0,余10
到第三位 10 0 < 101 结果为0余100
到第四位 100 1 > 101 结果为1, 余为1001-101 = 100,
到第五位 100 0 > 101结果为1 余为1000 -101 = 11
到第六位11 0 > 101 结果为1 余为110 -101 = 1
到第七位 1 1 < 101 结果为0 余为 11
合起来结果就是 0001110 ,余为11 转十进制就是14余3

二:常用位运算技巧

1:左移 << 与 右移>>

左移<<各二进位全部左移若干位,高位丢弃,低位补0, 右移>>各二进位全部右移若干位,对无符号数,高位补0, 有符号时会补上符号位,在JAVA中若无符号右移为>>>,符号位补0
左移n位即二进制右边补了n个0, 相当乘于2^n, 右移n位相当除2^n, 最常见 除2的操作 num >> 1 , 取颜色值
例如求int最小值,最大值

int minInt() {
    return  1 << 31;//10000000 00000000 00000000 00000000
}
int maxInt() {
    return ~(1 << 31); //上面取反即可
}
c上获取 int最大值, 其他类型最大最小同理
int cMaxInt() {
    return ((unsigned int)-1) >> 1;//右移一位相当符号位即可,取最小值将最大值取反即可
}

例如颠倒二进制位 00000000 00000000 10000000 10001110 变成01110001 00000001 00000000 00000000

uint32_t reverseBits(uint32_t n) {
    int i = 1;
    uint32_t r = n & 1;
    while (i < 32) {
        n >>= 1;
        r = (r << 1) + (n & 1);
        i++;
    }
    return r;
}

2:~ 取反 0变1, 1变0

如上求最大值最小值,最大值取反即为最小值,最小值取反即为最大值
10000000 最小值 取反 01111111即为最大值

3:&与运算 两个都为1时结果为1

例如:判断奇偶

 二进制最后一位与1与&即可
bool isOdd(int num) {
    return num & 1;
}

判断一个数是否为2的冥次,2的冥即二进制只有一个1

bool is2power(int num) {
    if (num > 0) {
        return (num & (num - 1)) == 0;
    }
    return false;
}

在一个2次冥大小的数组中递减或递加数组下标不越界不小于0,在队列数据结构中会使用到

size要为2的冥次
void queue(int size) {
    int index = 0;
    int maxIndex = size - 1;
    while (true) {
        index = index & maxIndex;
        cout << index << endl;
        index--; //index++
    }
}

求二进制数中1的个数

int binary1Count(unsigned int num) {
    int count = 0;
    while (num > 0) {
        num &= num - 1;
        count++;
    }
    return count;
}

4:| 或运算 两个位都为0时结果为0,否则为1

求一个比n大的,并且是最小的2的幂,比如3->4, 6->8, 100->128, 256->512, 这种算法在需要2次冥大小的数据结构中非常常见

int power2(int num) {
    num |= (num >> 1);
    num |= (num >> 2);
    num |= (num >> 4);
    num |= (num >> 8);
    num |= (num >> 16);
    num++;
    return num;
}

5:^异或运算 两个位相同为0,相异为1

判断两个字符是否相等忽略大小写

bool test(char c0, char c1) {
    return c0 == c1 || (c0 ^ 32) == c1;
}

不用临时变量交换两个数

void swap(int &a, int &b) {
    a ^= b;
    b ^= a;
    a ^= b;
}

判断是否为相同符号

bool isSameSign(int x, int y) {
    return (x ^ y) >= 0;
}

hashCode打散

static final int hash(Object key) {
    int h;
    return (key == null) ? 0 : (h = key.hashCode()) ^ (h >>> 16);
}

给定一个非空整数数组,除了某个元素只出现一次以外,其余每个元素均出现两次。找出那个只出现了一次的元素(java)

public int singleNumber(int[] nums) {
    for (int i = 1; i < nums.length; i++) {
        nums[0] ^= nums[i];
    }
    return nums[0];
}

更多文章请关注:http://www.jianshu.com/u/b1cff340957c

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 162,306评论 4 370
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 68,657评论 2 307
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 111,928评论 0 254
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 44,688评论 0 220
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 53,105评论 3 295
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 41,024评论 1 225
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 32,159评论 2 318
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,937评论 0 212
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,689评论 1 250
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,851评论 2 254
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,325评论 1 265
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,651评论 3 263
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,364评论 3 244
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,192评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,985评论 0 201
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 36,154评论 2 285
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,955评论 2 279