ClassCastExcpetion после использования Double.intValue () в лямбда-выражении - PullRequest
0 голосов
/ 01 декабря 2019

Итак, у меня странная проблема:

У меня есть JSlider и JSpinner, ползунок использует значения int, а счетчик принимает двойные значения. Теперь я хочу «соединить» их, используя прослушиватели изменений:

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$29(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$28(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$29(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.access$500(EventQueue.java:97)
at java.awt.EventQueue$3.run(EventQueue.java:709)
at java.awt.EventQueue$3.run(EventQueue.java:703)
at java.security.AccessController.doPrivileged(Native Method)

Я заметил следующее:

  • Использование System.out.println(spinner.getValue().getClass()) Я получаю class java.lang.Double.

  • Я проверил параметры метода spinner.setValue(), и он принимает примитив int.

  • Использование кода spinner.addChangeListener(e -> System.out.println(((Double) spinner.getValue()).intValue())); работает без исключения и печатает целое число.

  • Я также пробовал различные решения, такие как Math.round() или(int) Double.doubleValue(), до сих пор ничего не получалось.

Если у кого-то есть идея, почему это происходит, пожалуйста, скажите мне!

Редактировать: Пробуя еще немного, я обнаружил, что slider.setValue(((Number) spinner.getValue()).intValue())работает, но мне все равно хотелось бы сейчас почему? Я снова проверил возвращаемый класс, и он по-прежнему: java.lang.Double! Любая помощь приветствуется

Ответы [ 2 ]

1 голос
/ 01 декабря 2019

Первый слушатель

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

устанавливает значение счетчика на Integer - slider.getValue(), возвращает 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, как предложил metallurg , так как модель предназначена для работыс Number.

1 голос
/ 01 декабря 2019

Spinner return Integer. Слайдер ожидает дубль. Вы можете преобразовать Integer в Doble следующим образом: spinner.addChangeListener(e -> slider.setValue(((Number)spinner).doubleValue()));

Integer является подклассом Number. Вот почему разрешено приводить его к номеру. Номер имеет метод doubleValue().

...