Java使用注解實(shí)現(xiàn)BigDecimal的四舍五入
在開(kāi)始寫(xiě)代碼之前,我們需要先了解下BigDecimal有哪幾種小數(shù)位的截取方式
關(guān)于“截取”那些事兒
想要了解 BigDecimal 是如何處理小數(shù)位截取的,我們需要看下 RoundingMode 這個(gè)類(lèi):


public static void main(String[] args) {
BigDecimal bigDecimalValue = new BigDecimal("9.455");
BigDecimal result = bigDecimalValue.setScale(2, RoundingMode.FLOOR);
System.out.println("FLOOR: " + result);
BigDecimal result2 = bigDecimalValue.setScale(2, RoundingMode.CEILING);
System.out.println("CEILING: " + result2);
BigDecimal result3 = bigDecimalValue.setScale(2, RoundingMode.HALF_UP);
System.out.println("HALF_UP: " + result3);
BigDecimal result4 = bigDecimalValue.setScale(2, RoundingMode.HALF_EVEN);
System.out.println("HALF_EVEN: " + result4);
BigDecimal result5 = bigDecimalValue.setScale(2, RoundingMode.HALF_DOWN);
System.out.println("HALF_DOWN: " + result5);
BigDecimal result6 = bigDecimalValue.setScale(3, RoundingMode.UNNECESSARY);
System.out.println("UNNECESSARY: " + result6);
}
執(zhí)行結(jié)果如下:

將"9.445"改為"9.455",執(zhí)行結(jié)果如下:

我們可以看到,只有HALF_EVEN模式下,兩次的處理邏輯是不一樣的,不知道大家還其不記得有個(gè)規(guī)律叫什么“奇變偶不變”?
我們?cè)僭囅?/p>
將數(shù)字改為"9.425",執(zhí)行結(jié)果如下:

將數(shù)字改為"9.475",執(zhí)行結(jié)果如下:

RoundingMode.UNNECESSARY 表示該值無(wú)需做截取位數(shù)的操作,這個(gè)可以用來(lái)檢測(cè)位數(shù),因?yàn)槿绻谕奈粩?shù)和數(shù)值的實(shí)際位數(shù)不一致就會(huì)報(bào) ArithmeticException: Rounding necessary的異常,如下圖:

定義自定義注解
首先我們先新建一個(gè)自定義注解,可以直接復(fù)制如下代碼:
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;
@Target({ElementType.FIELD})
@Retention(RetentionPolicy.RUNTIME)
public @interface DecimalScale {
int value() default 2; // 默認(rèn)截取到兩位小數(shù)
}
來(lái)簡(jiǎn)單介紹下上述代碼的具體含義和作用(了解的同學(xué)可以跳過(guò)這p)
@Target({ElementType.FIELD}):表示這個(gè)注解可以應(yīng)用于字段(FIELD)元素@Retention(RetentionPolicy.RUNTIME):表明這個(gè)注解在運(yùn)行時(shí)仍然保留(可以在運(yùn)行時(shí)通過(guò)反射等機(jī)制獲取和使用)@interface DecimalScale:定義了這個(gè)注解的名字為DecimalScale(使用的時(shí)候就是@DecimalScale)int value() default 2;:定義了這個(gè)注解有一個(gè)名為value的屬性,類(lèi)型為整數(shù),并且默認(rèn)值是 2(定義這個(gè)屬性是為了使這個(gè)注解更加靈活,可以動(dòng)態(tài)傳入數(shù)字以指定具體保留的小數(shù)位)
創(chuàng)建 MetaObjectHandler 接口的實(shí)現(xiàn)類(lèi)
我們先來(lái)看看MetaObjectHandler是什么,為什么要實(shí)現(xiàn)這個(gè)接口

大家應(yīng)該經(jīng)常會(huì)碰到一些字幾乎所有的表都有,而且填充方式也是一樣的(例如例如創(chuàng)建時(shí)間、更新時(shí)間、創(chuàng)建人、更新人等),這些也就是我們常說(shuō)的“公共字段”,而為了簡(jiǎn)化這一操作,MyBatis-Plus 就提供了 MetaObjectHandler 接口,允許用戶(hù)可以定義自動(dòng)填充的邏輯,以減少重復(fù)的代碼邏輯,增加代碼的復(fù)用性。
我們今天這篇文章正是用到了它提供的 insertFill() 這個(gè)方法,詳細(xì)代碼如下:
package com.imile.ops.analysis.config.myabtis;
import com.baomidou.mybatisplus.core.handlers.MetaObjectHandler;
import com.imile.ops.analysis.annotation.DecimalScale;
import com.imile.ops.analysis.context.RequestInfoHolder;
import com.imile.ops.analysis.context.UserContext;
import org.apache.ibatis.reflection.MetaObject;
import org.springframework.stereotype.Component;
import java.math.BigDecimal;
import java.math.RoundingMode;
import java.time.LocalDateTime;
import java.util.Arrays;
import java.util.HashMap;
import java.util.Map;
/**
* MybatisPlus自定義填充公共字段
*
* @author Andrew
* @date 2020/08/11
*/
@Component
public class MybatisMetaObjectHandler implements MetaObjectHandler {
private final static String USER_CODE = "USER_CODE";
private final static String USER_NAME = "USER_NAME";
private Map<String, String> getOperator() {
Map<String, String> userMap = new HashMap<>();
UserContext userInfo = RequestInfoHolder.getLoginInfo();
if (userInfo == null) {
userMap.put(USER_CODE, "");
userMap.put(USER_NAME, "");
} else {
userMap.put(USER_CODE, userInfo.getUserCode());
userMap.put(USER_NAME, userInfo.getUserName());
}
return userMap;
}
@Override
public void insertFill(MetaObject metaObject) {
Map<String, String> userMap = getOperator();
// 如果用戶(hù)不為數(shù)據(jù)遷移
Object createUserCode = getFieldValByName("createUserCode", metaObject);
Object id = getFieldValByName("id", metaObject);
if (createUserCode == null) {
setFieldValByName("createUserCode", userMap.get(USER_CODE), metaObject);
}
Object createUserName = getFieldValByName("createUserName", metaObject);
if (createUserName == null) {
setFieldValByName("createUserName", userMap.get(USER_NAME), metaObject);
}
Object createDate = getFieldValByName("createDate", metaObject);
if (createDate == null) {
setFieldValByName("createDate", LocalDateTime.now(), metaObject);
}
Object lastUpdUserCode = getFieldValByName("lastUpdUserCode", metaObject);
if (lastUpdUserCode == null) {
setFieldValByName("lastUpdUserCode", userMap.get(USER_CODE), metaObject);
}
Object lastUpdUserName = getFieldValByName("lastUpdUserName", metaObject);
if (lastUpdUserName == null) {
setFieldValByName("lastUpdUserName", userMap.get(USER_NAME), metaObject);
}
Object lastUpdDate = getFieldValByName("lastUpdDate", metaObject);
if (lastUpdDate == null) {
setFieldValByName("lastUpdDate", LocalDateTime.now(), metaObject);
}
// 處理帶有 @DecimalScale() 注解的字段
processDecimalFields(metaObject);
}
@Override
public void updateFill(MetaObject metaObject) {
Map<String, String> userMap = getOperator();
setFieldValByName("lastUpdUserCode", userMap.get(USER_CODE), metaObject);
setFieldValByName("lastUpdUserName", userMap.get(USER_NAME), metaObject);
setFieldValByName("lastUpdDate", LocalDateTime.now(), metaObject);
processDecimalFields(metaObject);
}
/**
* 處理帶有 @DecimalScale() 注解的字段
*
* @param metaObject
*/
private void processDecimalFields(MetaObject metaObject) {
Arrays.stream(metaObject.getOriginalObject().getClass().getDeclaredFields())
.filter(field -> field.isAnnotationPresent(DecimalScale.class))
.forEach(field ->
{
try {
field.setAccessible(true);
// 獲取字段值
Object value = field.get(metaObject.getOriginalObject());
if (value != null) {
// 檢查字段值是否為 BigDecimal
if (value instanceof BigDecimal) {
// 是 -> 截取(向下)指定位數(shù)小數(shù)
BigDecimal v = new BigDecimal(value.toString());
// 賦值
field.set(metaObject.getOriginalObject(),
v.setScale(field.getAnnotation(DecimalScale.class).value(), RoundingMode.FLOOR));
} else {
throw new IllegalArgumentException("The field annotated with @DecimalScale must be a Number type.");
}
}
} catch (IllegalAccessException e) {
throw new RuntimeException(e);
}
});
}
}
可以看到上述方法用到了我們一開(kāi)始介紹的截取方法,但是我遇到一個(gè)很奇怪的問(wèn)題,代碼中用的是RoundingMode.FLOOR 向下取整的策略,但入庫(kù)的數(shù)據(jù)仍然是按照四舍五入截取的,本地debug發(fā)現(xiàn)這個(gè)RoundingMode.FLOOR 向下取整的策略配置并沒(méi)有生效,但目前還未到原因。
以上就是Java使用注解實(shí)現(xiàn)BigDecimal的四舍五入的詳細(xì)內(nèi)容,更多關(guān)于Java BigDecimal四舍五入的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
Java使用自定義注解實(shí)現(xiàn)為事件源綁定事件監(jiān)聽(tīng)器操作示例
這篇文章主要介紹了Java使用自定義注解實(shí)現(xiàn)為事件源綁定事件監(jiān)聽(tīng)器操作,結(jié)合實(shí)例形式分析了java自定義注解、注解處理、事件監(jiān)聽(tīng)與響應(yīng)等相關(guān)操作技巧,需要的朋友可以參考下2019-10-10
IDEA自定義Maven倉(cāng)庫(kù)的實(shí)現(xiàn)
使用Maven進(jìn)行Java程序開(kāi)發(fā)時(shí),開(kāi)發(fā)者能夠極大地提高開(kāi)發(fā)效率,本文主要介紹了IDEA自定義Maven倉(cāng)庫(kù)的實(shí)現(xiàn),具有一定的參考價(jià)值,感興趣的可以了解一下2024-03-03
SpringBoot業(yè)務(wù)邏輯異常的處理方法介紹
本篇文章為大家展示了如何在SpringBoot中統(tǒng)一處理邏輯異常,內(nèi)容簡(jiǎn)明扼要并且容易理解,絕對(duì)能使你眼前一亮,通過(guò)這篇文章的詳細(xì)介紹希望你能有所收獲2022-09-09
SpringBoot整合SpringDataRedis的示例代碼
這篇文章主要介紹了SpringBoot整合SpringDataRedis的示例代碼,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2021-05-05
java實(shí)現(xiàn)簡(jiǎn)單貪吃蛇小游戲
這篇文章主要為大家詳細(xì)介紹了java實(shí)現(xiàn)簡(jiǎn)單貪吃蛇小游戲,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-05-05
簡(jiǎn)單學(xué)習(xí)Java+MongoDB
本文給大家介紹的是如何簡(jiǎn)單的使用java+MongoDB實(shí)現(xiàn)數(shù)據(jù)調(diào)用的問(wèn)題,非常的實(shí)用,有需要的小伙伴可以參考下2016-03-03
Java多線程之scheduledThreadPool的方法解析
這篇文章主要介紹了Java多線程之scheduledThreadPool的方法解析,queue是DelayedWorkQueue,但通過(guò)后面的分析可以知道,最大線程數(shù)是不起作用的,最多會(huì)起核心線程數(shù)的數(shù)量,需要的朋友可以參考下2023-12-12
EasyExcel自定義下拉注解的三種實(shí)現(xiàn)方式總結(jié)
使用EasyExcel設(shè)置下拉數(shù)據(jù)時(shí),每次都要?jiǎng)?chuàng)建一個(gè)SheetWriteHandler組件確實(shí)比較繁瑣,為了優(yōu)化這個(gè)過(guò)程,我們可以通過(guò)自定義注解來(lái)簡(jiǎn)化操作,下面就來(lái)看看具體實(shí)現(xiàn)方法吧2024-10-10

