整数相除取余数java java除以一个数取整_求一批整数中出现最多的个位数字


文章大纲
普通实现方式以及弊端
两种位运算实现方式以及区别
题外话-求多个数的平均值

如何求两个整数的平均值?

问题看起来简单,如果用代码实现起来却有很多值得研究的地方。

下面我们使用Java代码来实现。

1 普通实现1

求两个整数的平均值,最简单的实现方法就是两个数相加再除以二。


private static int mean(int x, int y) {
  return (x + y) / 2;
}


但是, 如果传入的是两个整数都是Integer.MAX_VALUE, 下面的断言就过不了。


Assert.assertEquals(Integer.MAX_VALUE, 
    mean(Integer.MAX_VALUE, Integer.MAX_VALUE));


我们期望的结果是Integer.MAX_VALUE,但是实际上返回的结果却是-1。因为 Integer.MAX_VALUE + Integer.MAX_VALUE = -2, 结果已经溢出了。


java.lang.AssertionError: 
Expected :2147483647
Actual   :-1


2 普通实现2

如果我们使用无符号的右移运算符。


private static int mean1(int x, int y) {
  return (x + y) >>> 1;
}


这样虽然可以得到我们想要的结果Integer.MAX_VALUE, 但是却是不支持负数。

比如我们求-9和-3的平均值,


Assert.assertEquals(-6, mean1(-9, -34));


期望返回的结果是-6,但是实际上却返回了2147483642


java.lang.AssertionError: 
Expected :-6
Actual   :2147483642


3 普通实现3

那么如果我们不把两个整数直接相加, 而是分别除以2再相加, 是不是就不会溢出了呢?


private static int mean3(int x, int y) {
  return (x >> 1) + (y >> 1);
}


这样实现的话, 确实是不会溢出了,但是精度也丢失了。


java.lang.AssertionError: 
Expected :2147483647
Actual   :2147483646


那么有没有一种实现方式, 既不会溢出,又可以同时支持正负整数呢?

答案是不止一种!

4 位运算实现1

我们来看看Google的guava工具类是怎么实现的


/**
 * IntMath#mean
 * @link https://github.com/google/guava/blob/master/guava/src/com/google/common/math/IntMath.java
 */
// 小数 向下取整
private static int meanRoundDown(int x, int y) {
  return (x & y) + ((x ^ y) >> 1);
}


我们可以通过集合来理解上面的代码。

比如我们求整数9和3的平均值。

9的二进制是1001, 3的二进制是0011,

那么这两个数的交集就是9&3=0001,

差集就是1010。0001+(1010>>1)结果就是6。

我们都知道二进制数字都是一串0和1,那么可以把整数x和y都看作是一个有很多不同的0和1组成的集合。


整数相除取余数java java除以一个数取整_取整_02


那么x & y就表示两个集合的交集, 因为1&1=1;

x ^ y得到的就是两个集合的差集, 因为1^1=0,1^0=1;

交集加上差集的一半, 就得到了两个数的二进制平均值。


整数相除取余数java java除以一个数取整_取整_03


5 位运算实现2

下面我们来看看用位运算的第二种实现方式


// 小数 向上取整
private static int meanRoundUp(int x, int y) {
  return (x | y) - ((x ^ y) >> 1);
}


这段代码怎么理解呢?

x | y得到的就是两个集合的去重并集。

并集减去差集的一半,就得到了两个数的二进制平均值。


整数相除取余数java java除以一个数取整_取整_04


6 两种方式的区别

如果两个整数的平均值是小数时,

第一种方式是向下取整;

第二种方式是向上取整。

比如我们求整数9和4的平均值,这两个数的平均值应该是6.5,向上取整就是7,向下取值就是6。


Assert.assertEquals(6, meanRoundDown(9, 4)); // 6
Assert.assertEquals(7, meanRoundUp(9, 4)); // 7


X 题外话

上面探讨的只是求两个整数的平均值, 那么求多个整数的平均值又有哪些值得参考的实现方式呢?

同样我们来看看Google的guava工具类是怎么实现的。


/**
   * Stats#meanOf
   * @link https://github.com/google/guava/blob/master/guava/src/com/google/common/math/Stats.java
   * 
   * Returns the <a href="http://en.wikipedia.org/wiki/Arithmetic_mean">arithmetic mean</a> of the
   * values. The count must be non-zero.
   *
   * <p>The definition of the mean is the same as {@link Stats#mean}.
   *
   * @param values a series of values
   * @throws IllegalArgumentException if the dataset is empty
   */
  public static double meanOf(int... values) {
    checkArgument(values.length > 0);
    double mean = values[0];
    for (int index = 1; index < values.length; index++) {
      double value = values[index];
      if (isFinite(value) && isFinite(mean)) {
        // Art of Computer Programming vol. 2, Knuth, 4.2.2, (15)
        mean += (value - mean) / (index + 1);
      } else {
        mean = calculateNewMeanNonFinite(mean, value);
      }
    }
    return mean;
  }


请注意这句注释:

Art of Computer Programming vol. 2, Knuth, 4.2.2, (15)

Guava的实现是参考了高德纳老爷子的经典之作《计算机编程的艺术》, 这套书规划有 7 卷, 目前已经出版了 4 卷。

比尔盖茨给出的评价是:“如果你能读完此书,你绝对得给我发份简历。”


整数相除取余数java java除以一个数取整_java_05


完结撒花。

速记卡
求两个整数的平均值
(x & y) + ((x ^ y) >> 1)向下取整;
(x | y) - ((x ^ y) >> 1)向上取整;