首页 > 其他分享 >使用位运算技巧比较两个数中较大的数

使用位运算技巧比较两个数中较大的数

时间:2022-08-30 22:36:01浏览次数:117  
标签:return 技巧 符号 int scA flip sign 数中 运算

使用位运算技巧比较两个数中较大的数

作者:Grey

原文地址:

博客园:使用位运算技巧比较两个数中较大的数

CSDN:使用位运算技巧比较两个数中较大的数

题目要求

如何不要用任何比较判断符(>,==,<),返回两个数( 32 位整数)中较大的数。

主要思路

方法1(不考虑溢出)

要比较 a 和 b 的大小,因为不能用比较符号,我们可以通过 a - b 的符号位来判断,如果 a - b 的符号位是 1,说明 a - b < 0,则 a 小,否则 a 大或者 a 和 b 相等。

如何判断一个数的符号位是 0 还是 1 ?

由于是 32 位整数,所以如果将一个数右移 31 位,然后和 1 相与(&),如果得到 1,则这个数是负数,如果得到 0,则这个数是正数。

举个具体例子,如果要求 a 和 b 谁大,我们可以先通过

((a - b) >> 31) & 1 得到一个值,如果这个值是 1 ,说明 a 小,否则 a 大或者 a 和 b 相等。

由于不能出现比较符号,所以无法使用如下代码

return ((a - b) >> 31) & 1 == 1?b:a;

也无法使用如下代码

if (((a - b) >> 31) & 1 == 1){
    return b;
}
return a;

但是我们可以巧妙利用((a - b) >> 31) & 1结果去构造一个公式,这个公式可以在((a - b) >> 31) & 1 == 1情况下得到 b, 在((a - b) >> 31) & 1 == 0情况下得到 a。

我们可以利用一个反转函数

public int flip(int n) {
    return n ^ 1;
}

这个函数的作用就是,当n == 1时,返回 0,当n == 0时,返回 1,我们将判断符号的结果flip一次,如下代码

    public static int sign(int n) {
        return flip((n >> 31) & 1);
    }

这个方法的作用就是

当符号位是 1 的时候,返回 0,符号位是 0 的时候,返回 1。

这样flip后,

sign(a - b)如果得到 1, 则:a - b > 0,则返回 a。

sign(a - b)如果得到 0, 则:a - b <= 0,则返回 b。

公式可以定义成

sign(a - b) * a + flip(sign(a - b)) * b

主函数直接做如下调用

public static int getMax1(int a, int b) {
    int c = a - b;
    //当符号位是 1 的时候,scA = 0,符号位是 0 的时候,scA = 1。
    int scA = sign(c);
    // scA = 1 时,scB = 0,scA = 0时,scB = 1
    int scB = flip(scA);
    // 如果 scA = 0,说明 b 大,直接返回b
    // 如果 scA = 1,说明 a 大,直接返回a
    return a * scA + b * scB;
}

这个方法没有考虑溢出的情况,比如

a = 2147483647;
b = -2147480000;

a - b直接就溢出了,后面的算法就都不适用了。

方法2(考虑溢出情况)

那我们可以先比较 a 与 b 两个数的符号,

会有如下几种情况:

情况1:符号不同,则直接返回符号为正的那个数。

情况2:如果符号相同,则这种情况下,a - b的值绝对不会溢出,那么就看 c 的符号(c为正返回a,c为负返回b)

方法2的核心代码如下

int c = a - b;
int sa = sign(a);
int sb = sign(b);
int sc = sign(c);
int difSab = sa ^ sb;
int sameSab = flip(difSab);
int returnA = difSab * sa + sameSab * sc;
int returnB = flip(returnA);
return a * returnA + b * returnB;

其中: int difSab = sa ^ sb就是判断 a 和 b 的符号是否一样,如果 difSab == 1,则 a 和 b 符号一样,如果difSab == 0,则a 和 b符号不一样。

只有当difSab == 0的时候,要考虑 c 的符号。因为difSab == 0,所以int returnA = 0 * sa + 1 * sc;,即int returnA = sc,如果 sc 为 1,说明 c 的符号是 0,则a - b > 0,返回 a 即可,否则返回 b。

方法1和方法2的完整代码和测试代码如下:

// 不要用任何比较判断,返回两个数中较大的数
public class Code_GetMax {

    public static int flip(int n) {
        return n ^ 1;
    }


    public static int sign(int n) {
        return flip((n >> 31) & 1);
    }

    public static int getMax1(int a, int b) {
        int c = a - b;
        int scA = sign(c);
        int scB = flip(scA);
        return a * scA + b * scB;
    }

    public static int getMax2(int a, int b) {
        int c = a - b;
        int sa = sign(a);
        int sb = sign(b);
        int sc = sign(c);
        int difSab = sa ^ sb;
        int sameSab = flip(difSab);
        int returnA = difSab * sa + sameSab * sc;
        int returnB = flip(returnA);
        return a * returnA + b * returnB;
    }

    public static void main(String[] args) {
        int a = -16;
        int b = -19;
        System.out.println(getMax1(a, b));
        System.out.println(getMax2(a, b));
        a = 2147483647;
        b = -2147480000;
        System.out.println(getMax1(a, b)); // wrong answer because of overflow
        System.out.println(getMax2(a, b));

    }
}

更多

算法和数据结构笔记

参考资料

算法和数据结构新手班-左程云

标签:return,技巧,符号,int,scA,flip,sign,数中,运算
From: https://www.cnblogs.com/greyzeng/p/16641111.html

相关文章

  • 位运算与计数器,数组中其他数字都出现x次,只有一个数字出现一次
    一个数组,一个数字出现一次,其他数字出现x次,求只出现一次的数字。做法很多,但对空间与时间度有要求的话,位运算是最方便的做法如果x是2的话,仅仅异或运算就可以了,但如果更多次......
  • 使用java处理字符串公式运算的方法
    在改进一个关于合同的项目时,有个需求,就是由于合同中非数据项的计算公式会根据年份而进行变更,而之前是将公式硬编码到系统中的,只要时间一变,系统就没法使用了,因此要求合同中......
  • 运算方法和运算器
    数据与文字的表示方法二进制....八进制......十六进制一般用数字0到9和字母A到F表示,其中:AF相当于十进制的1015八进制和十六进制主要目的:简化二进制的书写进制转化......
  • 可选链操作符、逻辑与、空值合并运算符
    可选链操作符(?.)首先我们的明白一点,以下代码会报错吗?letobj={}leta=obj.nameconsole.log(a);那么,以下代码呢?letobj={}leta=obj.name.firstNameconsol......
  • 第6章 分支语句和逻辑运算符
    说明看《C++PrimerPlus》时整理的学习笔记,部分内容完全摘抄自《C++PrimerPlus》(第6版)中文版,StephenPrata著,张海龙袁国忠译,人民邮电出版社。只做学习记录用途。目......
  • EL表达式和EL运算符
    EL表达式1。概念:ExpressionLanguage表达式语言2.作用:替换和简化jsp页面中java代码的编写3,语法:$表达式}4,注意:"jsp默认支持el表达式的。如果要忽略el表达式1.设......
  • Go语言中的运算符
    ✍️介绍了Go中算术运算符,逻辑运算符,关系运算符等。这些是编程语言通用的,学到这,想起来大学时学c,vb和java的时光了。1.算术运算符1.1加、减、乘、除,取余var( a=1......
  • Codesys提升程序运行效率之AND_THEN、OR_ELSE运算符的使用
    之前看到有博文写Codesys程序编写标准中有一条,多个判断条件的if-else-语句,可能性最大的条件应放到最前面,这样可减少PLC处理的时间。但是根据测试,情况并非如此。下面的例子......
  • PHP 中的三元运算符和or表达式对比[defined() or define()]
    在php代码中我们经常看到这样的写法:$max=$a>$b?$a:$b;mysql_connect($user,$passwd,$db)ordie($mess);下面对这两种常见的写法做以下说明:第一种:典型的三元运算......
  • Vim实用技巧
    这篇文章是看《Vim实用技巧》整理的一些笔记,以及日常使用vim的小技巧的收录,保持更新VIM查找f{char}查找,;向前查找,向后查找跳转到指定字符之上.F反向查找t查找字符......