如何只允许在 jTextField 中引入数字?

Posted

技术标签:

【中文标题】如何只允许在 jTextField 中引入数字?【英文标题】:How to allow introducing only digits in jTextField? 【发布时间】:2012-03-17 15:41:53 【问题描述】:

我尝试使用here 显示的示例,但 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。那么,如何才能在文本字段中只打印数字呢?

【问题讨论】:

***.com/questions/5662651/… @Bakhtiyor 从不将 KeyListener 用于 JTextComponents,使用 DocumentListener 和过滤 DocumentFilter,这个论坛上有一些很好的 Digits 示例,您的链接帖子是最安全的方法 【参考方案1】:

这里检查这个代码 sn-p,这就是你如何通过使用 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 :谢谢,我刚刚在错误的线程中回答了某人,如何使用 DocumentFilter 来做到这一点,刚刚了解到不需要 JFormattedField ,因为人们想要的结果是不可取的使用 JFormattedField,在某些情况下,使用普通的 JTextField 也是如此。 :-) @mKorbel:请注意您的代码中存在错误。如果我将“a2”之类的字符串粘贴到该 JTextField 中,那么它将给定的字符串放入 JTextField 中没有问题,因为只有最后一个字符被验证。 @LajosArpad :非常感谢您指出BUG。我希望这一次,该错误已被删除 :-) 保持微笑 :-) 听起来需要付出很多努力,而在 android 等更现代的语言中,一个简单的 textField.SetValidation(TextField.Numeric) 或类似的方法就可以了。相当可怜的 Swing 没有这个【参考方案2】:

我建议使用JFormattedTextField,方法如下:How to Use Formatted Text Fields

【讨论】:

DocumentFilter 是另一种方式... +1【参考方案3】:

这是我用来消费非数字的

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
            
        
    );

【讨论】:

不接受浮点数。也可以使用复制粘贴来插入字符。 用户只要求输入数字。你说的复制粘贴是对的。【参考方案4】:

我的版本完全拒绝不需要的字符(我试过):

        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);

【讨论】:

【参考方案5】:

试试这个:

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();

【讨论】:

以上是关于如何只允许在 jTextField 中引入数字?的主要内容,如果未能解决你的问题,请参考以下文章

尝试仅允许将数字输入JTextField,将非数字设置为0

JAVA中如何判断JTextField的一些问题?求高手解答……谢谢

如何使用 Action Listener 和 Action Event 让 JButtons 在 JTextField 中打印整数?

如何限制用户只能输入0-50的数字

如何确保 JTextField 只包含字母?

如何在多次输入后更改 JTextField 字体(更新)