一、如何用移位操作实现乘法运算

        把一个数向左移动n位相当于把该数乘以2的n次方,因此当乘法运算中的某个数字满足这个特点时,就可以用移位操作来代替乘法操作,从而提高效率。
        示例如下:

public class numChengfa {

    public static int powerN(int m,int n){
        for(int i = 0;i < n;i++){
            m = m << 1;
        }
        return m;
    }
    public static void main(String[] args) {

        System.out.println("3乘以8="+powerN(3, 3));
        System.out.println("3乘以32="+powerN(3, 5));

    }

}

程序运行结果:
3乘以8=24
3乘以32=96

二、如何判断一个数是否为2的n次方

        2的n次方可以表示为:2^0,2^1……,2^n。最直观的思想是用1做移位操作,然后判断移位后的值是否与给定的值相等,具体实现代码如下:

public class twoMIJudge {

    /**
     * 判断一个数字是否为2个次方
     * 把一个数左移n位相当于把该数乘以2的n次方
     * @param n
     * @return
     */
    public static boolean isPowerOne(int n){
        if(n < 1){
            return false;
        }

        int i = 1;
        while(i <= n){
            if(i == n){
                return true;
            }

            i <<= 1;
        }
        return false;
    }

    public static void main(String[] args) {

        System.out.println("方法1:");
        System.out.println(isPowerOne(45));
        System.out.println(isPowerOne(128));
    }
}

程序运行结果:
方法1:
false
true

        上述算法的时间复杂度为O(logn)。那么是否存在效率更高的算法呢?通过对2^0,2^1……,2^n进行分析,发现这些数字的二进制形式分别为:1,10,100…。从二进制的表示可以看出,如果一个数是2的n次方,那么这个数对应的二进制表示中只有一位是1,其余位都为0。因此,判断一个数是否为2的n次方可以转换为这个数对应的二进制表示中是否只有一位为1。如果一个数的二进制表示只有一位是1,例如num=00010000,那么num-1的二进制表示为num-1=0000111,由于num与num-1二进制表示中每一位都不相同,因此num&(num-1)的运算结果为0,可以利用这种方法来判断一个数是否为2的n次方。具体实现代码如下:

public class twoMIJudge {

    /**
     * 通过与运算的方法
     * @param n
     * @return
     */
    public static boolean isPowerTwo(int n){
        if(n < 1){
            return false;
        }

        int m=n&(n-1);
        return m == 0;
    }
    public static void main(String[] args) {

        System.out.println("方法2:");
        System.out.println(isPowerTwo(45));
        System.out.println(isPowerTwo(128));

    }
}
程序运行结果:
方法2:
false
true

三、如何求二进制数中1的个数

        问题描述:给定一个整数,输出这个整数二进制表示中1的个数,例如,给定整数7,其二进制表示为111,因此输出结果为3。
        该问题可以采用位操作来完成。具体思路如下:首先,判断这个数的最后一位是否为1,如果为1,则计数器加1,然后,通过右移丢弃掉最后一位。循环执行操作直到这个数等于0为止。在判断二进制表示的最后一位是否为1时,可以采用与位运算来达到这个目的。具体实现代码如下:

public class countOne {

    public static int countOneFirst(int n){
        int count = 0;  //用来计数
        while(n > 0){
            if((n&1)==1){
                count++;
            }
            n >>= 1;
        }

        return count;
    }

    public static void main(String[] args) {

        System.out.println(countOneFirst(7));
        System.out.println(countOneFirst(8));
    }

}

程序运行结果:
3
1

        以上这个算法的时间复杂度为O(n),其中n代表二进制的位数。由于题目要求求出1的个数,此时可以只考虑1的个数,把二进制表示中的每个1看做独立的个体,利用上面介绍的算法可以求出1的个数。给定一个数n,没进行一次n&(n-1)计算,其结果中都会少了一位1,而且是最后一位。利用这个特性可以编写出以下代码:

package weiyunsuan;

/**
 * @author wilson_m
 * @date 2018年8月19日 
 * @Description:  
 * @version 1.0.0 
 */
public class countOne {

    public static int countOneNumber(int n){
        int count=0;
        while(n!=0){
            count++;
            n = n & (n - 1);
        }
        return count;
    }
    public static void main(String[] args) {

        System.out.println(countOneNumber(11));
        System.out.println(countOneNumber(13));
    }

}
程序运行结果:
3
3

        2的n次方可以表示为:2^0,2^1……,2^n。改进后的算法时间复杂度为O(m),其中m为二进制中1的个数,显然在二进制数中1的个数比较少时,这个算法的效率更高。

        常规解法,将数字转换为二进制后,对转换后的二进制数,逐个元素与1进行比较。利用这个特性可以编写出以下代码:

//常规解题方法:
public static int countOneTwo(int n){
         //在Integer类中有静态方法toBinaryString(int i)方法,此方法返回int变量的二进制表示的字符串。
        String str = Integer.toBinaryString(n);
        System.out.println("输入的数字进行二进制转换后得到的字符串为:"+str);

        int count = 0;
        for(int i = 0;i < str.length();i++){
            if(str.charAt(i)=='1'){
                count++;
            }
        }
        return count;
    }

输入实例:
System.out.println(countOneTwo(11));
System.out.println(countOneTwo(13));

输出结果:
输入的数字进行二进制转换后得到的字符串为:1011
3
输入的数字进行二进制转换后得到的字符串为:1101
3