- Java 双重比较
- java - 比较器与 Apache BeanComparator
- Objective-C 完成 block 导致额外的方法调用?
- database - RESTful URI 是否应该公开数据库主键?
我正在锻炼Java Puzzlers第二个谜题。
public class Change {
public static void main(String args[]) {
System.out.println(2.00 - 1.10);
}
}
您会认为答案是 0.9。但事实并非如此。如果你锻炼这个,你会得到 0.8999999。给出的解决方案是
System.out.println(new BigDecimal("2.00").subtract(new BigDecimal("1.10")));
现在它将打印 0.9。我明白为什么它会打印 0.89999。但是虽然我好奇地调试 BigDecimal 类,我发现在大多数地方都有许多常量值被替换。我将在下面列出所有这些,并且有兴趣了解其背后的原因。
BigDecimal.java第 394 行,
while (len > 10 && Character.digit(c, 10) == 0) {
offset++;
c = in[offset];
len--;
}
在这里Character.digit (c,10).
public static int digit(char ch, int radix) {
return digit((int)ch, radix);
}
这里 10 作为基数传递。
Q1。为什么 10 在那里传递。??
BigDecimal.java 行号 732
int sign = ((valBits >> 63)==0 ? 1 : -1);
int exponent = (int) ((valBits >> 52) & 0x7ffL);
long significand = (exponent==0 ? (valBits & ((1L<<52) - 1)) << 1
: (valBits & ((1L<<52) - 1)) | (1L<<52));
exponent -= 1075;
Q2。如果你深入查看代码,你可以理解valBits是什么,我无法理解为什么在某些地方使用右移?
Q3。在这里您还可以看到使用了 63、52 等许多常量。为什么?
Q4。我可以理解这里使用的十六进制0x7ffL
会提高执行速度。再次说明为什么 BitWise &
运算符与十六进制常量一起使用。
我希望我的问题很清楚。感谢您的耐心等待。
最佳答案
Q1:当然,基数是计算的基础。您使用的是以 10 为基数的数字 ;)
Q4:7ff 二进制:0111 1111 1111
如果您阅读位移运算符的解释,Java "Bit Shifting" Tutorial?你会看到 >> 63 丢弃前 63 位并保留最后一位。最后一位是符号位(对于有符号类型)。如果它是 1 则整数为负,因此那里有 int 符号。
另外可以引用https://en.wikipedia.org/wiki/Floating_point对于 float 的定义。 52是为了区分指数和尾数。
Q4:当然,对于指数,您不想在其中使用“符号”位,因为它不是指数的一部分,因此掩码为 7ff。 Bitwise & 0x7ff 就是这样做的;它只会在第一位上放置一个 0,而将另一个保留为相同的状态。 (参见“与”运算符的真值表 https://en.wikipedia.org/wiki/Truth_table )
编辑:
Q4 补充:如果指数是 12,那么第一位将是这样的:
0000 0000 1100 ... (positive value) (ex: 1x10^12)
1000 0000 1100 ... (negative value) (ex: -1x10^12)
但是:1000 0000 1100 是十进制的 2060。
如果你应用“面具”:
1000 0000 1100 & 0111 1111 1111 -> 0000 0000 1100 (12 decimal)
AND
0000 0000 1100 & 0111 1111 1111 -> 0000 0000 1100 (12 decimal)
这就是 0x7ff 的用途。
关于java - Java 中的 BigDecimal 类 - 常量值背后的原因,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/31563363/
当你做这样的事情 BigDecimal bigDecimal = BigDecimal.ONE; 为什么bigDecimal成为一个新对象? 最佳答案 在这种情况下,变量(或字段)bigDecimal
我正在使用 BigDecimal 计算一些大实数。虽然我尝试了两种方法:BigDecimal.toString() 或 BigDecimal.stripTrailingZeros().toString
我广泛使用 BigDecimals。我经常需要进行长时间的计算,然后比较结果。由于这些漫长的计算,答案不是 1而是0.9999999... 。这就是为什么我无法精确比较 BigDecimals,而只能
我有一段代码有两个 BigDecimal 变量。这两个变量都被发送到方法格式(工作正常)然后打印。变量“d”被发送到格式方法,然后直接打印而不分配它(因为格式方法返回一个字符串)。问题出现在变量 a
我遇到过 java.math.BigDecimal 和 android.icu.math.BigDecimal 因为我需要在项目中使用 BigDecimal . 我意识到 Android BigDec
我在整个应用程序中使用 BigDecimal 来处理金钱 和百分比。但是,我需要一种方法来区分两者之间的用法(为了呈现它们,即在 JTable 中)。因此,我最初的想法是编写两个行为与 BigDeci
这应该很简单,但它正在爆炸。有什么想法吗? d = BigDecimal.new("2.0") YAML::load({:a => d}.to_yaml) TypeError: BigDecimal
我有两个 BigDecimal,我想根据它们的 significant digits 确定它们是否接近。 . 例如,请考虑以下内容: BigDecimal million = new BigDecim
我需要一种方法将其表达为有效的java代码:我有2个BigDecimals,我想知道较小的BigDecimal是否可以(当添加到较大的BigDecimals时) BigDecimal 一次)更改较大
我正在争论是使用 BigDecimal 和 BigInteger 还是仅使用 BigDecimal 来让我的生活更轻松,减少来回转换。在资源方面仅使用 BigDecimal 有缺点吗? 仅使用原始数据
我有以下代码: Welcome to Scala version 2.10.3 (Java HotSpot(TM) 64-Bit Server VM, Java 1.7.0_45). scala> v
我有一个 java 客户端,它通过 GRPC 调用 Clojure 服务。问题是,当我们从他们的服务收到响应时,它会在大十进制的末尾添加字母。当我们尝试在 java 中将其转换为大十进制时,我们会收到
我仍在学习 Java,并且一直在阅读多个站点上的文章。我在 Java Code Geeks 找到了一篇文章我有一个问题。该文章正在解释开放/封闭原则。本文以对公司产品应用折扣的场景为例。第一部分代码如
我正在尝试对 List 中的多个 BigDecimals 求和.目前,我正在使用两个流,但如果可能的话,我希望只有一个流。我不确定如何以高效的方式重写下面的内容。 BigDecimal totalCh
下面两行代码有什么区别? BigDecimal one = new BigDecimal("1"); BigDecimal two = BigDecimal.ONE; 两条线是否相同? 谢谢! 最佳答
这个问题在这里已经有了答案: BigDecimal from Double incorrect value? (4 个答案) Convert double to BigDecimal and set
在 Java 中,来自另一个 bigDecimal.toString() 的新 BigDecimal 是否总是等于?例如 BigDecimal a = new BigDecimal("1.23
对于新变量的比较或初始化,您使用其中的哪一个会有所不同吗? 我知道 BigDecimal.ZERO 是 1.5 的功能,所以这是一个问题,但假设我使用的是 1.5,这有关系吗? 谢谢。 最佳答案 Bi
情况: public static double pi(int a) { return (BigDecimal.valueOf(53360*sqrt(640320))).divide(co
我正在尝试创建一个可以在 diesel 中用于插入的结构。具体来说,我正在使结构可插入。编译时出现此错误。 我有一个结构,我正试图通过 derive 属性使 Insertable 成为可能。我有一个名
我是一名优秀的程序员,十分优秀!