Java使用注解實(shí)現(xiàn)BigDecimal的四舍五入
在開始寫代碼之前,我們需要先了解下BigDecimal
有哪幾種小數(shù)位的截取方式
關(guān)于“截取”那些事兒
想要了解 BigDecimal
是如何處理小數(shù)位截取的,我們需要看下 RoundingMode
這個(gè)類:
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
表示該值無需做截取位數(shù)的操作,這個(gè)可以用來檢測(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ù) }
來簡(jiǎn)單介紹下上述代碼的具體含義和作用(了解的同學(xué)可以跳過這p)
@Target({ElementType.FIELD})
:表示這個(gè)注解可以應(yīng)用于字段(FIELD
)元素@Retention(RetentionPolicy.RUNTIME)
:表明這個(gè)注解在運(yùn)行時(shí)仍然保留(可以在運(yùn)行時(shí)通過反射等機(jī)制獲取和使用)@interface DecimalScale
:定義了這個(gè)注解的名字為DecimalScale
(使用的時(shí)候就是@DecimalScale
)int value() default 2;
:定義了這個(gè)注解有一個(gè)名為value
的屬性,類型為整數(shù),并且默認(rèn)值是 2(定義這個(gè)屬性是為了使這個(gè)注解更加靈活,可以動(dòng)態(tài)傳入數(shù)字以指定具體保留的小數(shù)位)
創(chuàng)建 MetaObjectHandler 接口的實(shí)現(xiàn)類
我們先來看看MetaObjectHandler是什么,為什么要實(shí)現(xiàn)這個(gè)接口
大家應(yīng)該經(jīng)常會(huì)碰到一些字幾乎所有的表都有,而且填充方式也是一樣的(例如例如創(chuàng)建時(shí)間、更新時(shí)間、創(chuàng)建人、更新人等),這些也就是我們常說的“公共字段”,而為了簡(jiǎn)化這一操作,MyBatis-Plus 就提供了 MetaObjectHandler 接口,允許用戶可以定義自動(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(); // 如果用戶不為數(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); } }); } }
可以看到上述方法用到了我們一開始介紹的截取方法,但是我遇到一個(gè)很奇怪的問題,代碼中用的是RoundingMode.FLOOR
向下取整的策略,但入庫(kù)的數(shù)據(jù)仍然是按照四舍五入截取的,本地debug發(fā)現(xiàn)這個(gè)RoundingMode.FLOOR
向下取整的策略配置并沒有生效,但目前還未到原因。
以上就是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)聽器操作示例
這篇文章主要介紹了Java使用自定義注解實(shí)現(xiàn)為事件源綁定事件監(jiān)聽器操作,結(jié)合實(shí)例形式分析了java自定義注解、注解處理、事件監(jiān)聽與響應(yīng)等相關(guān)操作技巧,需要的朋友可以參考下2019-10-10IDEA自定義Maven倉(cāng)庫(kù)的實(shí)現(xiàn)
使用Maven進(jìn)行Java程序開發(fā)時(shí),開發(fā)者能夠極大地提高開發(fā)效率,本文主要介紹了IDEA自定義Maven倉(cāng)庫(kù)的實(shí)現(xiàn),具有一定的參考價(jià)值,感興趣的可以了解一下2024-03-03SpringBoot業(yè)務(wù)邏輯異常的處理方法介紹
本篇文章為大家展示了如何在SpringBoot中統(tǒng)一處理邏輯異常,內(nèi)容簡(jiǎn)明扼要并且容易理解,絕對(duì)能使你眼前一亮,通過這篇文章的詳細(xì)介紹希望你能有所收獲2022-09-09SpringBoot整合SpringDataRedis的示例代碼
這篇文章主要介紹了SpringBoot整合SpringDataRedis的示例代碼,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2021-05-05java實(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)用的問題,非常的實(shí)用,有需要的小伙伴可以參考下2016-03-03Java多線程之scheduledThreadPool的方法解析
這篇文章主要介紹了Java多線程之scheduledThreadPool的方法解析,queue是DelayedWorkQueue,但通過后面的分析可以知道,最大線程數(shù)是不起作用的,最多會(huì)起核心線程數(shù)的數(shù)量,需要的朋友可以參考下2023-12-12EasyExcel自定義下拉注解的三種實(shí)現(xiàn)方式總結(jié)
使用EasyExcel設(shè)置下拉數(shù)據(jù)時(shí),每次都要?jiǎng)?chuàng)建一個(gè)SheetWriteHandler組件確實(shí)比較繁瑣,為了優(yōu)化這個(gè)過程,我們可以通過自定義注解來簡(jiǎn)化操作,下面就來看看具體實(shí)現(xiàn)方法吧2024-10-10