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