Java 每个操作侦听器的单个类?

Java 每个操作侦听器的单个类?,java,swing,button,listener,action,Java,Swing,Button,Listener,Action,我的应用程序有15个不同的按钮,我一直在想,为每个按钮侦听器创建一个单独的类是否是个好主意? 我目前只有一个类使用switch/case处理所有按钮,但很难维护和阅读 我不太喜欢使用匿名类——同样是因为它的可读性 任何能帮助我解决这个问题的建议都将不胜感激 如果这很重要,我将使用Java Swing。您可以使用一个类,但它不是一个好的设计,因为您需要应用分离关注点的原则。您需要类中的一致性,以便根据您所处理的业务领域,这些方法是有意义和逻辑的。此外,在某些情况下,同一个动作侦听器可以处理许多按钮

我的应用程序有15个不同的按钮,我一直在想,为每个按钮侦听器创建一个单独的类是否是个好主意? 我目前只有一个类使用switch/case处理所有按钮,但很难维护和阅读

我不太喜欢使用匿名类——同样是因为它的可读性

任何能帮助我解决这个问题的建议都将不胜感激


如果这很重要,我将使用Java Swing。

您可以使用一个类,但它不是一个好的设计,因为您需要应用分离关注点的原则。您需要类中的一致性,以便根据您所处理的业务领域,这些方法是有意义和逻辑的。此外,在某些情况下,同一个动作侦听器可以处理许多按钮

示例:假设我正在构建一个计算器。我知道操作符在单击时的行为是相似的。数字按钮也是如此。所以我可以上一些课,比如说

public class OperationActionListener {

    public void actionPerformed(ActionEvent e) {
        // Handle what happens when the user click on +, -, * and / buttons
    }
}


public class DigitActionListener {

    public void actionPerformed(ActionEvent e) {
        // Handle what happens when the user click on a digit button
    }
}
等等

现在,在我的用户界面中,我将添加同一操作侦听器的实例

JButton buttonPlus = new JButton("+")
JButton buttonMinus = new JButton("-");
...

JButton buttonOne = new JButton("1");
JButton buttonTwo = new JButton("2"); 
...

OperationActionListener operationListener = new OperationActionListener();
DigitActionListener digitListener = new DigitsActionListener();

buttonPlus.addActionListener(operationListener);
buttonMinus.addActionListener(operationListener);
....

buttonOne.addActionListener(digitListener);
buttonTwo.addActionListener(digitListener);
....

希望这能有所帮助。

下面是一个同一个侦听器被多个按钮使用的示例:

import java.awt.*;
import java.awt.event.*;
import javax.swing.*;
import javax.swing.border.*;

public class CalculatorPanel extends JPanel
{
    private JTextField display;

    public CalculatorPanel()
    {
        Action numberAction = new AbstractAction()
        {
            @Override
            public void actionPerformed(ActionEvent e)
            {
//              display.setCaretPosition( display.getDocument().getLength() );
                display.replaceSelection(e.getActionCommand());
            }
        };

        setLayout( new BorderLayout() );

        display = new JTextField();
        display.setEditable( false );
        display.setHorizontalAlignment(JTextField.RIGHT);
        add(display, BorderLayout.NORTH);

        JPanel buttonPanel = new JPanel();
        buttonPanel.setLayout( new GridLayout(0, 5) );
        add(buttonPanel, BorderLayout.CENTER);

        for (int i = 0; i < 10; i++)
        {
            String text = String.valueOf(i);
            JButton button = new JButton( text );
            button.addActionListener( numberAction );
            button.setBorder( new LineBorder(Color.BLACK) );
//          button.setPreferredSize( new Dimension(50, 50) );
            buttonPanel.add( button );

            InputMap inputMap = button.getInputMap(JComponent.WHEN_IN_FOCUSED_WINDOW);
            inputMap.put(KeyStroke.getKeyStroke(text), text);
            inputMap.put(KeyStroke.getKeyStroke("NUMPAD" + text), text);
            button.getActionMap().put(text, numberAction);
        }
    }

    private static void createAndShowUI()
    {
        UIManager.put("Button.margin", new Insets(10, 10, 10, 10) );

        JFrame frame = new JFrame("Calculator Panel");
        frame.setDefaultCloseOperation( JFrame.EXIT_ON_CLOSE );
        frame.add( new CalculatorPanel() );
        frame.pack();
        frame.setLocationRelativeTo( null );
        frame.setVisible(true);
    }

    public static void main(String[] args)
    {
        EventQueue.invokeLater(new Runnable()
        {
            public void run()
            {
                createAndShowUI();
            }
        });
    }
}
import java.awt.*;
导入java.awt.event.*;
导入javax.swing.*;
导入javax.swing.border.*;
公共类计算器面板扩展了JPanel
{
专用JTextField显示;
公共计算器面板()
{
Action numberAction=new AbstractAction()
{
@凌驾
已执行的公共无效操作(操作事件e)
{
//display.setCaretPosition(display.getDocument().getLength());
display.replaceSelection(例如getActionCommand());
}
};
setLayout(新的BorderLayout());
display=新的JTextField();
display.setEditable(false);
display.setHorizontalAlignment(JTextField.RIGHT);
添加(显示,BorderLayout.NORTH);
JPanel buttonPanel=新的JPanel();
setLayout(新的GridLayout(0,5));
添加(按钮面板、边框布局、中心);
对于(int i=0;i<10;i++)
{
String text=String.valueOf(i);
JButton按钮=新JButton(文本);
addActionListener(numberAction);
button.setOrder(新行边框(颜色为黑色));
//按钮。设置首选尺寸(新尺寸(50,50));
按钮面板。添加(按钮);
InputMap InputMap=button.getInputMap(JComponent.WHEN_IN_FOCUSED_窗口);
inputMap.put(击键.getKeyStroke(文本),文本);
inputMap.put(KeyStroke.getKeyStroke(“NUMPAD”+text),text);
button.getActionMap().put(文本、数字操作);
}
}
私有静态void createAndShowUI()
{
UIManager.put(“按钮边距”,新插图(10,10,10,10));
JFrame=新的JFrame(“计算器面板”);
frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
frame.add(新计算器面板());
frame.pack();
frame.setLocationRelativeTo(空);
frame.setVisible(true);
}
公共静态void main(字符串[]args)
{
invokeLater(新的Runnable()
{
公开募捐
{
createAndShowUI();
}
});
}
}

您研究过lambda表达式吗?它们是函数接口(只有一个方法的接口)的Mous类的替代品。通常,是的,每个JButton都有一个单独的动作监听器是个好主意。然而,如果一组JButtons是逻辑分组的,比如一个3 x 3网格的JButtons用于tic-tac-toe游戏,那么JButtons组有一个动作侦听器是有意义的。所以有多个类不是问题吗?在我的例子中,几乎所有的按钮都必须更改用户界面,我必须将帧的实例传递给侦听器,以便进行更改。类似于:
someButton.addActionListener(newbuttonListener(this))这会导致性能问题吗?没有多个类本身不是问题。但是,设计课程应该总是有目的地进行。在每个按钮上添加操作侦听器时,不要执行
new ButtonListener(this)
,而是使用
ButtonListener listener=new ButtonListener(this)
然后执行
someButton.addActionListener(listener)
和someButton2.addActionListener(listener)。通过这种方式重复使用同一个侦听器。您能解释一下使用匿名对象实例化和创建非匿名对象之间的区别吗?提前谢谢。@kraikov当您确定不需要在其他任何地方使用匿名对象实例时,您通常会使用它。这是一次性交易!您无法在其他地方重用匿名类,因为如果您试图复制相同的创建代码,它将实例化一个新对象。在某些情况下,它是有用的:您不想因为管理一个类而打扰自己。示例:您希望仅一次性使用MouseAdapter中的mouseClicked事件