在 lambda 表达式中使用 Double.intValue() 后的 ClassCastExcpetion

ClassCastExcpetion after using Double.intValue() in lambda expression

所以,我有一个奇怪的问题:

我有一个 JSlider 和一个 JSpinner,Slider 使用 int 值,而 spinner 接受 double 值。现在我想 'connect' 他们通过使用更改侦听器:

slider.addChangeListener(e -> spinner.setValue(slider.getValue()));
spinner.addChangeListener(e -> slider.setValue(((Double) spinner.getValue()).intValue()));

但是由于一些奇怪的原因我得到了一个 ClassCastException:

Exception in thread "AWT-EventQueue-0" java.lang.ClassCastException: java.lang.Integer cannot be cast to java.lang.Double
at rt.Main.lambda$rotateTo(Main.java:318)
at javax.swing.JSpinner.fireStateChanged(JSpinner.java:458)
at javax.swing.JSpinner$ModelListener.stateChanged(JSpinner.java:386)
at javax.swing.AbstractSpinnerModel.fireStateChanged(AbstractSpinnerModel.java:119)
at javax.swing.SpinnerNumberModel.setValue(SpinnerNumberModel.java:447)
at javax.swing.JSpinner.setValue(JSpinner.java:354)
at rt.Main.lambda$rotateTo(Main.java:317)
at javax.swing.JSlider.fireStateChanged(JSlider.java:432)
at javax.swing.JSlider$ModelListener.stateChanged(JSlider.java:350)
at javax.swing.DefaultBoundedRangeModel.fireStateChanged(DefaultBoundedRangeModel.java:364)
at javax.swing.DefaultBoundedRangeModel.setRangeProperties(DefaultBoundedRangeModel.java:302)
at javax.swing.DefaultBoundedRangeModel.setValue(DefaultBoundedRangeModel.java:168)
at javax.swing.JSlider.setValue(JSlider.java:531)
at rt.Main.lambda$rotateTo(Main.java:318)
at javax.swing.JSpinner.fireStateChanged(JSpinner.java:458)
at javax.swing.JSpinner$ModelListener.stateChanged(JSpinner.java:386)
at javax.swing.AbstractSpinnerModel.fireStateChanged(AbstractSpinnerModel.java:119)
at javax.swing.SpinnerNumberModel.setValue(SpinnerNumberModel.java:447)
at javax.swing.JSpinner.setValue(JSpinner.java:354)
at javax.swing.plaf.basic.BasicSpinnerUI$ArrowButtonHandler.actionPerformed(BasicSpinnerUI.java:654)
at javax.swing.AbstractButton.fireActionPerformed(AbstractButton.java:2022)
at javax.swing.AbstractButton$Handler.actionPerformed(AbstractButton.java:2348)
at javax.swing.DefaultButtonModel.fireActionPerformed(DefaultButtonModel.java:402)
at javax.swing.DefaultButtonModel.setPressed(DefaultButtonModel.java:259)
at javax.swing.plaf.basic.BasicButtonListener.mouseReleased(BasicButtonListener.java:252)
at java.awt.AWTEventMulticaster.mouseReleased(AWTEventMulticaster.java:289)
at java.awt.AWTEventMulticaster.mouseReleased(AWTEventMulticaster.java:289)
at java.awt.Component.processMouseEvent(Component.java:6539)
at javax.swing.JComponent.processMouseEvent(JComponent.java:3324)
at java.awt.Component.processEvent(Component.java:6304)
at java.awt.Container.processEvent(Container.java:2239)
at java.awt.Component.dispatchEventImpl(Component.java:4889)
at java.awt.Container.dispatchEventImpl(Container.java:2297)
at java.awt.Component.dispatchEvent(Component.java:4711)
at java.awt.LightweightDispatcher.retargetMouseEvent(Container.java:4904)
at java.awt.LightweightDispatcher.processMouseEvent(Container.java:4535)
at java.awt.LightweightDispatcher.dispatchEvent(Container.java:4476)
at java.awt.Container.dispatchEventImpl(Container.java:2283)
at java.awt.Window.dispatchEventImpl(Window.java:2746)
at java.awt.Component.dispatchEvent(Component.java:4711)
at java.awt.EventQueue.dispatchEventImpl(EventQueue.java:760)
at java.awt.EventQueue.access0(EventQueue.java:97)
at java.awt.EventQueue.run(EventQueue.java:709)
at java.awt.EventQueue.run(EventQueue.java:703)
at java.security.AccessController.doPrivileged(Native Method)

我注意到以下几点:

如果有人知道为什么会发生这种情况,请告诉我!

编辑: 在尝试更多时,我发现 slider.setValue(((Number) spinner.getValue()).intValue()) 确实有效,但我现在仍然想知道为什么? 我检查了再次返回的class,它仍然是:java.lang.Double! 任何帮助表示赞赏

微调器return 整数。滑块需要 Double。您可以按如下方式将 Integer 转换为 Doble: spinner.addChangeListener(e -> slider.setValue(((Number)spinner).doubleValue()));

整数是数字的子类。这就是允许将其转换为 Number 的原因。 Number 有方法 doubleValue().

第一个听众

slider.addChangeListener(e -> spinner.setValue(slider.getValue()));

正在将微调器的值设置为 Integer - slider.getValue() returns 一个 int 被装箱,因为 spinner.setValue() 需要 Object.

可能* 您正在使用一个 SpinnerNumberModel 接受任何 Number 的属性(参见 javadoc)。由于微调器的值已更改(由第一个侦听器以编程方式更改),它的侦听器也被调用,导致异常,因为它现在具有 Integer 作为值。

显示微调器获得 Integer:

的简单测试
JSlider slider = new JSlider(0, 0, 9, 5);
JSpinner spinner = new JSpinner(new SpinnerNumberModel(5.0, 0.0, 9.0, 1.0));

slider.addChangeListener(e -> spinner.setValue(slider.getValue()));
spinner.addChangeListener(e -> slider.setValue(((Number) spinner.getValue()).intValue()));

JOptionPane.showMessageDialog(null, new Object[] { slider, spinner });
System.out.println(spinner.getValue().getClass());

移动滑块并按 OK - 它会将微调器值设置为 Integer

* 缺少有问题的最小完整示例


正确的解决方案,恕我直言:

slider.addChangeListener(e -> spinner.setValue((double)slider.getValue()));
spinner.addChangeListener(e -> slider.setValue(((Number) spinner.getValue()).intValue()));

我将演员表留给 Number,因为该模型旨在与 Number 一起使用。