奇怪的 Java 强制转换异常。为什么我不能将Long投掷到浮子上?

2022-09-03 04:32:42

为什么我不能将Long投掷到浮子上?

我收到此错误消息:

java.lang.ClassCastException: java.lang.Long 不能 cast to java.lang.Float

为什么这是一个问题?我试图转换的数字是域 [-10.0, 10.0] 中的小数。它们最初是使用 返回的对象实例。但它们必须转换为浮点数。JFormattedTextField.getValue()

堆栈跟踪:

Exception in thread "AWT-EventQueue-0" java.lang.ClassCastException: java.lang.Long cannot be cast to java.lang.Float
    at submodeler.animation.Timeline.setKeyedAttribute(Timeline.java:59)
    at submodeler.ui.attributes.TransformationAttributePanel.actionPerformed(TransformationAttributePanel.java:247)
    at javax.swing.AbstractButton.fireActionPerformed(AbstractButton.java:2028)
    at javax.swing.AbstractButton$Handler.actionPerformed(AbstractButton.java:2351)
    at javax.swing.DefaultButtonModel.fireActionPerformed(DefaultButtonModel.java:387)
    at javax.swing.DefaultButtonModel.setPressed(DefaultButtonModel.java:242)
    at javax.swing.plaf.basic.BasicButtonListener.mouseReleased(BasicButtonListener.java:236)
    at java.awt.Component.processMouseEvent(Component.java:6348)
    at javax.swing.JComponent.processMouseEvent(JComponent.java:3267)
    at java.awt.Component.processEvent(Component.java:6113)
    at java.awt.Container.processEvent(Container.java:2085)
    at java.awt.Component.dispatchEventImpl(Component.java:4714)
    at java.awt.Container.dispatchEventImpl(Container.java:2143)
    at java.awt.Component.dispatchEvent(Component.java:4544)
    at java.awt.LightweightDispatcher.retargetMouseEvent(Container.java:4618)
    at java.awt.LightweightDispatcher.processMouseEvent(Container.java:4282)
    at java.awt.LightweightDispatcher.dispatchEvent(Container.java:4212)
    at java.awt.Container.dispatchEventImpl(Container.java:2129)
    at java.awt.Window.dispatchEventImpl(Window.java:2475)
    at java.awt.Component.dispatchEvent(Component.java:4544)
    at java.awt.EventQueue.dispatchEvent(EventQueue.java:635)
    at java.awt.EventDispatchThread.pumpOneEventForFilters(EventDispatchThread.java:296)
    at java.awt.EventDispatchThread.pumpEventsForFilter(EventDispatchThread.java:211)
    at java.awt.EventDispatchThread.pumpEventsForHierarchy(EventDispatchThread.java:201)
    at java.awt.EventDispatchThread.pumpEvents(EventDispatchThread.java:196)
    at java.awt.EventDispatchThread.pumpEvents(EventDispatchThread.java:188)
    at java.awt.EventDispatchThread.run(EventDispatchThread.java:122)

答案 1

如果您提供堆栈跟踪,这将有所帮助。

否则,标准解决方案是替换为(Float) someLongValuesomeLongValue.floatValue()

如果您正在处理基元类型,则可以从 long 转换为 float,尽管这是一个 5.1.2 加宽基元转换,但可能会丢失精度,所以要小心!显然,你有包装器类型Long,它不能隐式转换,因此你得到classcastexception。这可能是因为自动装箱或显式 Long 对象创建。

一些更不请自来的建议:如果你的有效值是-10到+10范围内的小数,则标准数据类型是(基元)。如果您指的是确切的数字,请避免浮点数。 也不是最优的,因为它不像int那样是完全原子的,它需要2倍的内存。如果允许不同的状态“未分配”,则 (可能取 null) 也是 OK。intlongInteger


答案 2

发生此异常的原因是编译器认识到 Long 和 Float 没有父子依赖项。与C++不同,它将尝试找到重载的强制转换运算符。

您可以在从 Number 派生的任何基元包装器类型上调用 doubleValue(),因此这是执行所需操作的推荐方法:

double val = ((Number)textField.getValue()).doubleValue()

这里需要注意的几件事:首先,我将getValue()的输出转换为Number。这样,您可以改变主意,并在将来实际返回Double值,并且不会中断。

另外,我使用的是双精度值,而不是浮点数。JVM 在内部使用 double,因此除了节省数组中的空间之外,没有理由使用 float。


推荐