一、round函数

在 Java 中,`Math.round()` 函数是一个常用的数学库函数,它可以完成一个数的四舍五入操作。

该函数的定义如下:

```java
 public static long round(double a)
 ```

该函数接收一个浮点数参数,返回一个长整型结果。它的作用是将参数 `a` 的值进行四舍五入,然后将结果转换为长整型并返回。

具体来说,如果参数 `a` 的小数部分大于等于 0.5,则将其向上取整,否则将其向下取整。例如:

```java
 double a = 3.6;
 long b = Math.round(a);
 System.out.println(b); // 输出 4a = 3.4;
 b = Math.round(a);
 System.out.println(b); // 输出 3
 ```

这个例子中,我们分别传入了 3.6 和 3.4 两个浮点数作为 `Math.round()` 函数的参数。由于 3.6 的小数部分大于 0.5,所以结果向上取整为 4;而 3.4 的小数部分小于 0.5,所以结果向下取整为 3。

除了 `Math.round()` 函数外,还有 `Math.floor()` 和 `Math.ceil()` 函数可以完成向下取整和向上取整的操作。其中 `Math.floor()` 函数返回小于或等于参数值的最大整数,而 `Math.ceil()` 函数返回大于或等于参数值的最小整数。例如:

```java
 double a = 3.6;
 long b = Math.floor(a);
 System.out.println(b); // 输出 3a = 3.4;
 b = Math.ceil(a);
 System.out.println(b); // 输出 4
 ```

这个例子中,我们分别使用 `Math.floor` 和 `Math.ceil()` 函数将浮点数 a 的值进行向下取整和向上取整。由于 3.6 的最大整数为 3,所以 `Math.floor(a)` 返回 3;而 3.4 的最小整数为 4,所以 `Math.ceil(a)` 返回 4。

二、货币精度处理

在 Java 中,通常我们需要对货币进行精度处理,避免浮点数运算时出现精度误差。

通常情况下,我们可以使用 `java.math.BigDecimal` 类来处理货币精度,该类支持高精度计算和常见数学运算。可以使用该类实现进位、舍位、四舍五入等操作。

例如,以下是使用 `java.math.BigDecimal` 类进行货币进度处理的示例:

```java
 import java.math.BigDecimal;
 import java.math.RoundingMode;BigDecimal price = new BigDecimal("125.1234567");
// 进位,保留小数点后两位,结果为 125.13
 BigDecimal roundingUpPrice = price.setScale(2, RoundingMode.UP);// 舍位,保留小数点后两位,结果为 125.12
 BigDecimal roundingDownPrice = price.setScale(2, RoundingMode.DOWN);// 四舍五入,保留小数点后两位,结果为 125.12
 BigDecimal roundingHalfUpPrice = price.setScale(2, RoundingMode.HALF_UP);
 ```

在这个例子中,我们首先创建了一个 `BigDecimal` 对象 `price`,然后分别使用 `setScale()` 方法对其进行了不同的进度处理方式,得到了三个新的 `BigDecimal` 对象,分别为 `roundingUpPrice`、`roundingDownPrice` 和 `roundingHalfUpPrice`。

在 `setScale()` 方法中,第一个参数表示要保留的小数点位数,第二个参数表示进度处理方式。使用 `RoundingMode.UP` 表示进位操作,`RoundingMode.DOWN` 表示舍位操作,`RoundingMode.HALF_UP` 表示四舍五入操作。可以根据实际需求选择对应的进度处理方式。

除了使用 `java.math.BigDecimal` 类外,我们还可以使用 `java.text.DecimalFormat` 类进行货币格式化和精度处理,具体使用方式可参考该类的文档和示例。在进行进度处理时,常用的处理方式包括 `DecimalFormat.format()` 和 `Math.round()` 等,具体选择哪种方式,需要根据项目需求和实际场景来决定。

2.1 向下取整示例

在 Java 中,可以使用 `BigDecimal` 的 `setScale()` 方法进行舍入操作,其中舍入模式可以选择向下取整(`RoundingMode.DOWN`)。

如果在向下取整时指定了小数位数,那么最终结果会保留指定的小数位数,并将多余的部分直接截去,不会进行任何四舍五入操作。

因此,如果将 `1234.56` 按照向下取整的方式舍入到小数点后 2 位(即指定精度为 `2`),则最终结果会保留小数点后 `2` 位,并将多余的部分直接截去,得到的结果为 `1234.00`。

具体的示例代码如下所示:

```java
 BigDecimal num = new BigDecimal("1234.56");
 BigDecimal result = num.setScale(2, RoundingMode.DOWN);
 System.out.println(result);
 ```

输出结果为:

```
1234.00
```

可以看到,最终结果确实为 `1234.00`,符合我们的预期。需要注意的是,在进行数字计算时,`BigDecimal` 类型具有非常高的精度,因此可以保证计算的准确性和精度,避免了使用 `double` 类型可能会遇到的精度损失问题。

2.2 向下取整传参

在 Java 中,可以使用 `setScale()` 方法对 `BigDecimal` 类型的数据进行舍入操作。其中,setScale() 方法接受两个参数,第一个参数为精度,第二个参数为舍入模式。根据题目要求,需要将 `654.5` 向下取整,可以使用如下代码完成操作:

```java
 BigDecimal bd = new BigDecimal("654.5");
 BigDecimal result = bd.setScale(0, RoundingMode.DOWN);
 System.out.println(result);
 ```

在上述代码中,我们先将 `654.5` 转换为 `BigDecimal` 类型,并使用 `setScale()` 方法设置精度为 `0`,即不保留小数,并将舍入模式设置为 `RoundingMode.DOWN`,表示向下取整。

运行上述代码,输出结果为:

```
654
```

可以看到,`654.5` 经过 `setScale()` 方法的处理,被成功向下取整为 `654`。需要注意的是,在实际使用中,我们需要根据要求选择适当的舍入模式和精度设置,确保数据在计算和处理过程中的准确性和精度。

三、BigDecimal 和int型精度差

在 Java 中,`BigDecimal` 类型与 `int` 类型的精度存在很大差别。

`int` 类型在 Java 中是一种整数类型,它只能表示整数值,无法表示小数或者有限小数。虽然 `int` 类型是一种非常快速和高效的类型,在很多场景下都可以满足需求,但在需要处理小数或者精度较高的场景下就无法满足需求。

`BigDecimal` 类型则是 Java 提供的一种高精度数值类型,它可以表示任意精度的十进制数,而不会丢失任何精度。使用 `BigDecimal` 类型进行数字计算时,可以避免 `int` 类型在计算过程中可能发生精度丢失的情况。

比如,假设有一个商品价格为 `123.45`,使用 `int` 类型来表示它时,只能表示其整数部分 `123`,而小数部分无法表示。而使用 `BigDecimal` 类型来表示它时,则可以完整地表示其小数部分,并且还可以对它进行高精度计算,确保计算结果的准确性和精度。

具体的示例代码如下所示:

```java
 int priceInt = 12345;
 BigDecimal priceBD = new BigDecimal("123.45");
 System.out.println("int 类型价格为:" + priceInt);
 System.out.println("BigDecimal 类型价格为:" + priceBD);
 ```

输出结果如下:

```
int 类型价格为:12345
BigDecimal 类型价格为:123.45
```

可以看到,使用 `int` 类型时只能表示整数部分,而使用 `BigDecimal` 类型时可以完整表示价格的小数部分,避免了精度丢失的问题。

四、BigDecimal  与int型类型转换

4.1 int转为BigDecimal

在 Java 中,可以使用 `BigDecimal` 类型的构造函数将 `int` 类型转换为 `BigDecimal` 类型。

具体而言,`BigDecimal` 类型提供了两个构造函数可以用于将 `int` 类型的数字转换为 `BigDecimal` 类型:

1. `public BigDecimal(int val)`: 将 `int` 类型的数字转换为 `BigDecimal` 类型。
2. `public BigDecimal(String val)`: 将 `String` 类型的数字转换为 `BigDecimal` 类型。

相比之下,我们更推荐使用第二个构造函数,因为它可以确保高精度计算时没有精度丢失。而使用第一个构造函数时,会将 `int` 类型的数字先转换为字符串,然后再将字符串转换为 `BigDecimal` 类型,这样进行了两次转换,可能会在转换过程中产生精度丢失问题。

具体的示例代码如下所示:

```java
 int num = 1234;
 BigDecimal bdNum = new BigDecimal(num);  // 使用构造函数1:int 转换为 BigDecimal
 BigDecimal bdNum2 = new BigDecimal(String.valueOf(num));  // 使用构造函数2:String 转换为 BigDecimal
 System.out.println("int 转换为 BigDecimal 后结果为:" + bdNum);
 System.out.println("String 转换为 BigDecimal 后的结果为:" + bdNum2);
 ```

输出结果为:

```
int 转换为 BigDecimal 后结果为:1234
String 转换为 BigDecimal 后的结果为:1234
```

可以看到,`int` 类型的数字被成功转换为了 `BigDecimal` 类型,并且没有出现精度丢失问题。需要注意的是,在实际使用中,我们需要根据实际情况选择适当的精度设置和舍入模式,以确保高精度计算的准确性和精度。

4.2 BigDecimal转为int

在 Java 中,可以使用 `BigDecimal` 类型的 `intValue()` 方法将 `BigDecimal` 类型转换为 `int` 类型。

`intValue()` 方法会将 `BigDecimal` 类型中的数值转换为 `int` 类型的整数部分,并忽略小数部分。如果 `BigDecimal` 类型的数值超出了 `int` 类型能够表示的范围,那么将会发生溢出,导致结果不准确。

具体的示例代码如下所示:

```java
 BigDecimal bdNum = new BigDecimal("1234.56");
 int num = bdNum.intValue();
 System.out.println("BigDecimal 转换为 int 后结果为:" + num);
 ```

输出结果为:

```
BigDecimal 转换为 int 后结果为:1234
```

可以看到,`BigDecimal` 类型的数值被成功转换为了 `int` 类型,并且小数部分被忽略掉了。需要注意的是,如果 `BigDecimal` 类型的数值超过了 `int` 类型能够表示的范围,那么将会导致结果不准确。在这种情况下,我们可以考虑使用 `long` 类型或者 `BigInteger` 类型来进行转换,以提高精度。