Java中SpinnerNumberModel的奇怪行为

5
我正在尝试使用SpinnerNumberModel设置JSpinner,其中value = 0.0005,min = 0.0,max = 1.0,step size = 0.0005。然而,当我使用这些参数创建一个微调器时,我观察到非常奇怪的行为。它不是从0.0005开始,并且每次单击向上箭头时增加0.0005,而是似乎停留在0。
为了确保这不仅仅是格式问题,我在每个更改事件后打印了微调器的值。毫无疑问,对于每个向上箭头的单击,控制台都显示值为0,然后是0.0005,无论微调器被单击多少次。
下面是我用来测试这个问题的代码,包括一个具有稍微不同值的微调器,可以用于比较目的。
import javax.swing.*;
import javax.swing.event.ChangeEvent;
import javax.swing.event.ChangeListener;
import java.awt.*;

public class Main {

    public static void main(String[] args) {
        JFrame frame = new JFrame();
        frame.setSize(200, 80);
        JPanel panel = new JPanel();
        panel.setLayout(new BorderLayout());
        final JSpinner spinner1 = new JSpinner(new SpinnerNumberModel(0.0005, 0.0, 1.0, 0.0005));
        final JSpinner spinner2 = new JSpinner(new SpinnerNumberModel(0.05, 0.0, 1.0, 0.05));
        panel.add(spinner1, BorderLayout.NORTH);
        panel.add(spinner2, BorderLayout.SOUTH);
        frame.add(panel);
        spinner1.addChangeListener(new ChangeListener() {
            public void stateChanged(ChangeEvent e) {
                System.out.println("spinner1: " + ((Number) spinner1.getValue()).doubleValue());
            }
        });
        spinner2.addChangeListener(new ChangeListener() {
            public void stateChanged(ChangeEvent e) {
                System.out.println("spinner2: " + ((Number) spinner2.getValue()).doubleValue());
            }
        });
        frame.setVisible(true);
        frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
    }

}

有人能解释一下为什么spinner2的工作正常,但spinner1却不行吗?

谢谢。

编辑:感谢Peter指出,微调器文本字段的格式实际上会影响存储的值。为了使第一个微调器按预期工作,我添加了以下代码:spinner1.setEditor(new JSpinner.NumberEditor(spinner1, "0.0000"));


感谢您的输入。我最初遇到这个问题的实际项目比这个复杂得多。我在这里发布的代码仅是一个非常简单的示例,用于演示该问题。 - David Thielke
好的,我已经编辑了这个例子,加入了一个JPanel来容纳这两个微调框。不幸的是,第一个微调框仍然得到了相同的奇怪结果。 - David Thielke
我已经使用了JPanel,问题仍然存在。我尝试使用(0.001, 0.0001)作为JSpinner模型的(value, step),也尝试了(0.0001, 0.001),但都不起作用。但是(0.001, 0.001)却可以正常工作。这是否意味着JSpinner存在精度问题? - xorange
1个回答

9
问题在于微调器使用的JFormattedTextField。 在该编辑器中,您可以找到:
public void commitEdit() throws ParseException {
    AbstractFormatter format = getFormatter();

    if (format != null) {
        setValue(format.stringToValue(getText()), false, true);
    }
}

您的微调控件将0.0005显示为0.0,如果您先将0.0放入模型中,然后再将其增加到0.0005,则事件实际上会捕获它,但是您的文本字段仅看到0.0。

因此解决方案是更改微调控件文本字段的格式。

您可以覆盖编辑器。

final JSpinner spinner1 = new JSpinner(new SpinnerNumberModel(0.0005, 0.0, 1.0, 0.0005)) {
      @Override
      protected JComponent createEditor( SpinnerModel model )
      {
        return new NumberEditor(this, "0.0000");//needed decimal format
      }
    };

有趣的是...我以为无论文本字段如何格式化,旋转器都会保持相同的双精度值。谢谢! - David Thielke
谢谢Peter。我编辑了我的问题,加入了一行类似的内容。 - David Thielke

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接