如何让jTextField只允许输入数字?

12

我尝试使用这里所示的示例,但Java显示错误消息:

"AttributeSet无法解析为类型"

这就是为什么我在尝试使用另一种只允许数字的方法:

txtUsername.addKeyListener(new MyKeyListener());

public class MyKeyListener extends KeyAdapter{
  public void keyPressed(KeyEvent ke){
      System.out.println("Key pressed code = "+ke.getKeyCode());
      if (ke.getKeyCode()>=48 && ke.getKeyCode()<=57)
              return true;
      else
              return false;
  }
} 

但是当然它不起作用,因为keyPressed方法是void。那么,为了只在文本字段中打印数字,该怎么办呢?


https://dev59.com/hG035IYBdhLWcg3wC7YR - Jwalin Shah
2
@Bakhtiyor永远不要在JTextComponents中使用KeyListener,应该使用DocumentListener,并且使用DocumentFilter进行过滤。这个论坛上有一些很好的关于Digits的例子,你链接的帖子是最安全的实现方式。 - mKorbel
5个回答

25

可以查看以下代码片段,这是使用DocumentFilter允许JTextField仅接受数字的最有效方法:

import java.awt.*;
import javax.swing.*;
import javax.swing.text.AbstractDocument;
import javax.swing.text.AttributeSet;
import javax.swing.text.BadLocationException;
import javax.swing.text.DocumentFilter;
import javax.swing.text.DocumentFilter.FilterBypass;

public class InputInteger
{
    private JTextField tField;
    private MyDocumentFilter documentFilter;

    private void displayGUI()
    {
        JFrame frame = new JFrame("Input Integer Example");
        frame.setDefaultCloseOperation(JFrame.DISPOSE_ON_CLOSE);

        JPanel contentPane = new JPanel();
        contentPane.setBorder(
            BorderFactory.createEmptyBorder(5, 5, 5, 5));
        tField = new JTextField(10);
        ((AbstractDocument)tField.getDocument()).setDocumentFilter(
                new MyDocumentFilter());        
        contentPane.add(tField); 

        frame.setContentPane(contentPane);
        frame.pack();
        frame.setLocationByPlatform(true);
        frame.setVisible(true);
    }

    public static void main(String[] args)
    {
        Runnable runnable = new Runnable()
        {
            @Override
            public void run()
            {
                new InputInteger().displayGUI();
            }
        };
        EventQueue.invokeLater(runnable);
    }
}

class MyDocumentFilter extends DocumentFilter
{   
    @Override
    public void insertString(DocumentFilter.FilterBypass fp
            , int offset, String string, AttributeSet aset)
                                throws BadLocationException
    {
        int len = string.length();
        boolean isValidInteger = true;

        for (int i = 0; i < len; i++)
        {
            if (!Character.isDigit(string.charAt(i)))
            {
                isValidInteger = false;
                break;
            }
        }
        if (isValidInteger)
            super.insertString(fp, offset, string, aset);
        else
            Toolkit.getDefaultToolkit().beep();
    }

    @Override
    public void replace(DocumentFilter.FilterBypass fp, int offset
                    , int length, String string, AttributeSet aset)
                                        throws BadLocationException
    {
        int len = string.length();
        boolean isValidInteger = true;

        for (int i = 0; i < len; i++)
        {
            if (!Character.isDigit(string.charAt(i)))
            {
                isValidInteger = false;
                break;
            }
        }
        if (isValidInteger)
            super.replace(fp, offset, length, string, aset);
        else
            Toolkit.getDefaultToolkit().beep();
    }
}

或者可以直接使用这种方法,就像@Carlos Heuberger所提供的那样

@Override
public void insertString(FilterBypass fb, int off
                    , String str, AttributeSet attr) 
                            throws BadLocationException 
{
    // remove non-digits
    fb.insertString(off, str.replaceAll("\\D++", ""), attr);
} 
@Override
public void replace(FilterBypass fb, int off
        , int len, String str, AttributeSet attr) 
                        throws BadLocationException 
{
    // remove non-digits
    fb.replace(off, len, str.replaceAll("\\D++", ""), attr);
}

呵呵奇怪,但对于DocumentFilter +1。 - mKorbel
@mKorbel:谢谢,我刚刚在错误的线程中回答了某个人的问题,关于如何使用DocumentFilter处理此问题,我刚学到不需要使用JFormattedField,因为在某些情况下,使用JFormattedField得不到预期结果,所以我用普通的JTextField解决了这个问题。 :-) - nIcE cOw
2
@mKorbel:请注意,您的代码中存在一个错误。如果我将像“a2”这样的字符串粘贴到JTextField中,那么它会将给定的字符串放入JTextField中,因为只有最后一个字符被验证。 - Lajos Arpad
1
@LajosArpad:非常感谢您指出这个__BUG__。我希望这次已经将其排除了 :-) 保持微笑 :-) - nIcE cOw
1
听起来需要花费很多精力,而在更现代的语言中,比如Android,一个简单的textField.SetValidation(TextField.Numeric)或类似的方法就可以解决问题。很遗憾Swing没有这个功能。 - Jono
显示剩余2条评论

10

DocumentFilter是另一种过滤文档内容的方式... +1 - mKorbel

1

这是我用来处理非数字的方法

textField.addKeyListener(new KeyAdapter() {
        public void keyTyped(KeyEvent e) {
            char c = e.getKeyChar();
            if (((c < '0') || (c > '9')) && (c != KeyEvent.VK_BACK_SPACE)) {
                e.consume(); // consume non-numbers
            }
        }
    });

不接受浮点数。同时,可以使用复制粘贴插入字符。 - Bu Saeed
用户要求仅输入数字。你关于复制粘贴的想法是正确的。 - Alde

1

我尝试完全拒绝不需要的字符的版本:

        val2 = new JTextField();
        val2.addKeyListener(new KeyAdapter() {
        @Override
        public void keyTyped(KeyEvent e) {

            try {

                char input = e.getKeyChar(); //checks each key pressed
                // in other words it prevents input of any character other than 0-9 or .

                try {
                    int i = Integer.parseInt(val2.getText());
                    valid1.setText(""); //valid1 is a label for the message

                } catch (Exception e2) {
                    // TODO: handle exception
                    valid1.setText("Invalid Number!");
                }

                if((!Character.isDigit(input)) && !(input == '.')) { //numbers accept "."
                    e.consume();
                    // event won't be sent for any further event listeners
                    try {
                        int i = Integer.parseInt(val2.getText()); // checks if the string can be converted to int
                        double i1 = Double.parseDouble(val2.getText()); // checks if string can be converted to double
                        valid1.setText("");

                    } catch (Exception e2) {
                        // TODO: handle exception
                        // if it is not in a valid format.. it will generate error message
                        valid1.setText("Invalid Number!");
                    }


                }
                else {
                    // if format and characters are fine... no output
                    valid1.setText("");
                }


            }
            catch (Exception e2) {
                // TODO: handle exception
                // for unexpected errors
                valid1.setText("Error");
            }

        }

    });

    // auto generated for design of the textfield
    val2.setColumns(10);
    val2.setBounds(236, 11, 126, 43);
    panel.add(val2);

0

试试这个:

public void replace(DocumentFilter.FilterBypass fb, int offset,
     int length, String text, AttributeSet attrs)throws BadLocationException { 

    if(fb.getDocument().getLength() + text.length() > 10) {
        return;
    }
    fb.insertString(offset, text, attrs);
}

改为:

public void replace(DocumentFilter.FilterBypass fp, int offset
                , int length, String string, AttributeSet aset)
                                    throws BadLocationException
{
    int len = string.length();
    boolean isValidInteger = true;

    for (int i = 0; i < len; i++)
    {
        if (!Character.isDigit(string.charAt(i)))
        {
            isValidInteger = false;
            break;
        }
    }
    if (isValidInteger)
        super.replace(fp, offset, length, string, aset);
    else
        Toolkit.getDefaultToolkit().beep();
}

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