完美解決java double數(shù)相加和相減的方案
我就廢話不多說(shuō)了,大家還是直接看代碼吧~
/**
* double的計(jì)算不精確,會(huì)有類(lèi)似0.0000000000000002的誤差,正確的方法是使用BigDecimal或者用整型
* 整型地方法適合于貨幣精度已知的情況,比如12.11+1.10轉(zhuǎn)成1211+110計(jì)算,最后再/100即可
* 以下是摘抄的BigDecimal方法:
*/
public class DoubleUtils implements Serializable {
private static final long serialVersionUID = -3345205828566485102L;
// 默認(rèn)除法運(yùn)算精度
private static final Integer DEF_DIV_SCALE = 2;
/**
* 提供精確的加法運(yùn)算。
*
* @param value1 被加數(shù)
* @param value2 加數(shù)
* @return 兩個(gè)參數(shù)的和
*/
public static Double add(Double value1, Double value2) {
BigDecimal b1 = new BigDecimal(Double.toString(value1));
BigDecimal b2 = new BigDecimal(Double.toString(value2));
return b1.add(b2).doubleValue();
}
/**
* 提供精確的減法運(yùn)算。
*
* @param value1 被減數(shù)
* @param value2 減數(shù)
* @return 兩個(gè)參數(shù)的差
*/
public static double sub(Double value1, Double value2) {
BigDecimal b1 = new BigDecimal(Double.toString(value1));
BigDecimal b2 = new BigDecimal(Double.toString(value2));
return b1.subtract(b2).doubleValue();
}
/**
* 提供精確的乘法運(yùn)算。
*
* @param value1 被乘數(shù)
* @param value2 乘數(shù)
* @return 兩個(gè)參數(shù)的積
*/
public static Double mul(Double value1, Double value2) {
BigDecimal b1 = new BigDecimal(Double.toString(value1));
BigDecimal b2 = new BigDecimal(Double.toString(value2));
return b1.multiply(b2).doubleValue();
}
/**
* 提供(相對(duì))精確的除法運(yùn)算,當(dāng)發(fā)生除不盡的情況時(shí), 精確到小數(shù)點(diǎn)以后10位,以后的數(shù)字四舍五入。
*
* @param dividend 被除數(shù)
* @param divisor 除數(shù)
* @return 兩個(gè)參數(shù)的商
*/
public static Double divide(Double dividend, Double divisor) {
return divide(dividend, divisor, DEF_DIV_SCALE);
}
/**
* 提供(相對(duì))精確的除法運(yùn)算。 當(dāng)發(fā)生除不盡的情況時(shí),由scale參數(shù)指定精度,以后的數(shù)字四舍五入。
*
* @param dividend 被除數(shù)
* @param divisor 除數(shù)
* @param scale 表示表示需要精確到小數(shù)點(diǎn)以后幾位。
* @return 兩個(gè)參數(shù)的商
*/
public static Double divide(Double dividend, Double divisor, Integer scale) {
if (scale < 0) {
throw new IllegalArgumentException("The scale must be a positive integer or zero");
}
BigDecimal b1 = new BigDecimal(Double.toString(dividend));
BigDecimal b2 = new BigDecimal(Double.toString(divisor));
return b1.divide(b2, scale,RoundingMode.HALF_UP).doubleValue();
}
/**
* 提供指定數(shù)值的(精確)小數(shù)位四舍五入處理。
*
* @param value 需要四舍五入的數(shù)字
* @param scale 小數(shù)點(diǎn)后保留幾位
* @return 四舍五入后的結(jié)果
*/
public static double round(double value,int scale){
if(scale<0){
throw new IllegalArgumentException("The scale must be a positive integer or zero");
}
BigDecimal b = new BigDecimal(Double.toString(value));
BigDecimal one = new BigDecimal("1");
return b.divide(one,scale, RoundingMode.HALF_UP).doubleValue();
}
}
javascript double相互運(yùn)算出現(xiàn)小數(shù)問(wèn)題解決
//加法
function numAdd(num1, num2) {
var baseNum, baseNum1, baseNum2;
try {
baseNum1 = num1.toString().split(".")[1].length;
} catch (e) {
baseNum1 = 0;
}
try {
baseNum2 = num2.toString().split(".")[1].length;
} catch (e) {
baseNum2 = 0;
}
baseNum = Math.pow(10, Math.max(baseNum1, baseNum2));
var precision = (baseNum1 >= baseNum2) ? baseNum1 : baseNum2;//精度
return ((num1 * baseNum + num2 * baseNum) / baseNum).toFixed(precision);;
};
//減法運(yùn)算
function numSub(num1, num2) {
var baseNum, baseNum1, baseNum2;
try {
baseNum1 = num1.toString().split(".")[1].length;
} catch (e) {
baseNum1 = 0;
}
try {
baseNum2 = num2.toString().split(".")[1].length;
} catch (e) {
baseNum2 = 0;
}
baseNum = Math.pow(10, Math.max(baseNum1, baseNum2));
var precision = (baseNum1 >= baseNum2) ? baseNum1 : baseNum2;
return ((num1 * baseNum - num2 * baseNum) / baseNum).toFixed(precision);
};
// 乘法運(yùn)算
function numMulti(num1, num2) {
var baseNum = 0;
try {
baseNum += num1.toString().split(".")[1].length;
} catch (e) {
}
try {
baseNum += num2.toString().split(".")[1].length;
} catch (e) {
}
return Number(num1.toString().replace(".", "")) * Number(num2.toString().replace(".", "")) / Math.pow(10, baseNum);
};
// 除法運(yùn)算,避免數(shù)據(jù)相除小數(shù)點(diǎn)后產(chǎn)生多位數(shù)和計(jì)算精度損失。
function numDiv(num1, num2) {
var baseNum1 = 0, baseNum2 = 0;
var baseNum3, baseNum4;
try {
baseNum1 = num1.toString().split(".")[1].length;
} catch (e) {
baseNum1 = 0;
}
try {
baseNum2 = num2.toString().split(".")[1].length;
} catch (e) {
baseNum2 = 0;
}
with (Math) {
baseNum3 = Number(num1.toString().replace(".", ""));
baseNum4 = Number(num2.toString().replace(".", ""));
return (baseNum3 / baseNum4) * pow(10, baseNum2 - baseNum1);
}
};
補(bǔ)充:java中double、float 類(lèi)型相同數(shù)相減不為0
大家在java編程的時(shí)候經(jīng)常會(huì)遇到這種情況,兩個(gè)float或者double類(lèi)型的相減,但是他們結(jié)果不為0,那我們?cè)撛趺唇鉀Q呢,下面小編告訴大家一種方法,希望對(duì)大家有用。
1、大家先看看這種情況,兩個(gè)相同的浮點(diǎn)數(shù)相減不為0,

2、再說(shuō)下BigDecimal,Java在java.math包中提供的API類(lèi)BigDecimal,用來(lái)對(duì)超過(guò)16位有效位的數(shù)進(jìn)行精確的運(yùn)算。雙精度浮點(diǎn)型變量double可以處理16位有效數(shù)。在實(shí)際應(yīng)用中,需要對(duì)更大或者更小的數(shù)進(jìn)行運(yùn)算和處理。這時(shí)我們就要用BigDecimal,用的時(shí)候?qū)氚黬ava.math.BigDecimal;

3、我們既然知道要用BigDecimal就行轉(zhuǎn)化,下面就試試

4、比較之后我們很容易知道BigDecimal對(duì)double、float 類(lèi)型相同數(shù)相減不為0是個(gè)有效的方法了
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教。
相關(guān)文章
手把手帶你實(shí)現(xiàn)第一個(gè)Mybatis程序
這篇文章主要介紹了mybatis實(shí)現(xiàn)過(guò)程詳解,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2021-07-07
SpringBoot定時(shí)任務(wù)動(dòng)態(tài)擴(kuò)展ScheduledTaskRegistrar詳解
這篇文章主要為大家介紹了SpringBoot定時(shí)任務(wù)動(dòng)態(tài)擴(kuò)展ScheduledTaskRegistrar類(lèi)示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-01-01
SpringBoot之RestTemplate在URL中轉(zhuǎn)義字符的問(wèn)題
這篇文章主要介紹了SpringBoot之RestTemplate在URL中轉(zhuǎn)義字符的問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-06-06
myeclipse中使用maven前常見(jiàn)錯(cuò)誤及解決辦法
這篇文章主要介紹了myeclipse中使用maven前常見(jiàn)錯(cuò)誤及解決辦法 的相關(guān)資料,需要的朋友可以參考下2016-05-05
spring security登錄成功后通過(guò)Principal獲取名返回空問(wèn)題
這篇文章主要介紹了spring security登錄成功后通過(guò)Principal獲取名返回空問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-03-03

