一位经验丰富的开发人员告诉我,传递一个GUI实例是一个坏主意。
基本上,我有一个构建和显示GUI的类。在actionListener中,我创建一个对象,该对象执行一些耗时的任务,并且希望在任务的某些里程碑完成时显示状态。
这是该类的非常简化的版本:
public class MyGui extends JFrame {
private static final long serialVersionUID = 1L;
private JPanel mainPanel;
private JPanel selectionPanel;
private JPanel activityPanel;
private JPanel executePanel;
private JButton connectButton;
private JButton disconnectButton;
private JButton abortButton;
private JList aList;
private JComboBox comboBox;
private JRadioButton primaryButton;
private JRadioButton secondaryButton;
private static JTextArea activityTextArea;
MyGui() {
this.setTitle("My Tool");
mainPanel = new JPanel();
mainPanel.setLayout(new GridLayout(3, 1));
mainPanel.setBorder(BorderFactory.createEmptyBorder(10, 10, 10, 10));
mainPanel.setBackground(Color.darkGray);
this.add(mainPanel);
createMainSelectionArea();
createNodeSelectionArea();
createStatusArea();
createExecuteArea();
mainPanel.add(selectionPanel);
mainPanel.add(activityPanel);
mainPanel.add(executePanel);
this.add(mainPanel);
this.setResizable(false);
addActivity("test1");
addActivity("test2");
addActivity("test3");
addActivity("test4");
addActivity("test5");
addActivity("test6");
this.setSize(600, 400);
this.setVisible(true);
this.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
}
private void createMainSelectionArea() {
RadioButtonListener radioButtonListener = new RadioButtonListener();
primaryButton = new JRadioButton("Primary");
primaryButton.setBackground(Color.darkGray);
primaryButton.setForeground(Color.white);
primaryButton.addActionListener(radioButtonListener);
secondaryButton = new JRadioButton("Secondary");
secondaryButton.setBackground(Color.darkGray);
secondaryButton.setForeground(Color.white);
secondaryButton.addActionListener(radioButtonListener);
ButtonGroup buttonGroup = new ButtonGroup();
buttonGroup.add(primaryButton);
buttonGroup.add(secondaryButton);
JPanel buttonGroupPanel = new JPanel(new GridLayout(3, 1));
buttonGroupPanel.setBorder(BorderFactory.createEmptyBorder(10, 10, 10, 10));
buttonGroupPanel.setOpaque(true);
buttonGroupPanel.setBackground(Color.darkGray);
buttonGroupPanel.setForeground(Color.white);
buttonGroupPanel.add(primaryButton);
buttonGroupPanel.add(secondaryButton);
selectionPanel = new JPanel(new GridLayout(1, 2));
selectionPanel.setBorder(BorderFactory.createEmptyBorder(10, 10, 10, 10));
selectionPanel.setBackground(Color.darkGray);
selectionPanel.add(buttonGroupPanel);
}
private void createNodeSelectionArea() {
String[] data1 = {"one", "two", "three", "4", "5", "6"};
String[] data2 = {"four", "five", "six", "seven", "eight"};
ComboBoxListener comboBoxListener = new ComboBoxListener();
comboBox = new JComboBox(data1);
comboBox.setBorder(BorderFactory.createLineBorder(Color.white));
comboBox.setPreferredSize(new Dimension(150, 20));;
comboBox.setBackground(Color.white);
comboBox.setForeground(Color.black);
comboBox.addActionListener(comboBoxListener);
JPanel comboBoxPanel = new JPanel();
comboBoxPanel.setBorder(BorderFactory.createEmptyBorder(10, 10, 10, 10));
comboBoxPanel.setBackground(Color.darkGray);
comboBoxPanel.add(comboBox, BorderLayout.CENTER);
ListBoxListener listBoxListener = new ListBoxListener();
aList = new JList(data2);
aList.setBackground(Color.black);
aList.setForeground(Color.white);
aList.addListSelectionListener(listBoxListener);
JScrollPane scrollPane = new JScrollPane(aList);
scrollPane.setBorder(BorderFactory.createEmptyBorder(5, 5, 5, 5));
scrollPane.setBackground(Color.darkGray);
JPanel listBoxPanel = new JPanel(new GridLayout(1,1));
TitledBorder border = BorderFactory.createTitledBorder("A Selection");
border.setTitleColor(Color.white);
border.setBorder(BorderFactory.createLineBorder(Color.white));
listBoxPanel.setBorder(border);
listBoxPanel.setBackground(Color.darkGray);
listBoxPanel.setForeground(Color.white);
listBoxPanel.add(scrollPane);
selectionPanel.add(comboBoxPanel);
selectionPanel.add(listBoxPanel);
}
private void createStatusArea() {
activityTextArea = new JTextArea();
activityTextArea.setEditable(false);
activityTextArea.setBorder(BorderFactory.createEmptyBorder(5, 5, 5, 5));
activityTextArea.setBackground(Color.black);
activityTextArea.setForeground(Color.white);
JScrollPane scrollPane = new JScrollPane(activityTextArea);
scrollPane.setBorder(BorderFactory.createEmptyBorder(10, 10, 10, 10));
scrollPane.setBackground(Color.darkGray);
TitledBorder activityTitle = BorderFactory.createTitledBorder("Status");
activityTitle.setTitleColor(Color.white);
activityTitle.setBorder(BorderFactory.createLineBorder(Color.white));
activityTitle.setTitlePosition(TitledBorder.CENTER);
activityPanel = new JPanel(new GridLayout(1, 1));
activityPanel.setBackground(Color.darkGray);
activityPanel.setBorder(activityTitle);
activityPanel.add(scrollPane);
}
public void addActivity(String activity) {
activityTextArea.append(activity + "\n");
activityTextArea.setCaretPosition(activityTextArea.getDocument().getLength());
}
public void createExecuteArea() {
ButtonListener buttonListener = new ButtonListener();
connectButton = new JButton("Connect");
connectButton.setPreferredSize(new Dimension(115, 30));
connectButton.setBorder(BorderFactory.createCompoundBorder(BorderFactory.createRaisedBevelBorder(), BorderFactory.createLineBorder(Color.lightGray)));
connectButton.setBackground(Color.white);
connectButton.addActionListener(buttonListener);
disconnectButton = new JButton("Disconnect");
disconnectButton.setPreferredSize(new Dimension(115, 30));
disconnectButton.setBorder(BorderFactory.createCompoundBorder(BorderFactory.createRaisedBevelBorder(), BorderFactory.createLineBorder(Color.lightGray)));
disconnectButton.setBackground(Color.white);
disconnectButton.addActionListener(buttonListener);
abortButton = new JButton("Abort");
abortButton.setPreferredSize(new Dimension(115, 30));
abortButton.setBorder(BorderFactory.createCompoundBorder(BorderFactory.createRaisedBevelBorder(), BorderFactory.createLineBorder(Color.lightGray)));
abortButton.setBackground(Color.white);
abortButton.addActionListener(buttonListener);
executePanel = new JPanel(new GridBagLayout());
executePanel.setBackground(Color.darkGray);
GridBagConstraints c = new GridBagConstraints();
JPanel buttonPanel = new JPanel();
buttonPanel.setBackground(Color.darkGray);
buttonPanel.add(connectButton);
buttonPanel.add(disconnectButton);
buttonPanel.add(abortButton);
executePanel.add(buttonPanel, c);
}
private class ButtonListener implements ActionListener {
@Override
public void actionPerformed(ActionEvent e) {
if(e.getSource().equals(connectButton)) {
System.out.println("Connect Button");
}
if(e.getSource().equals(disconnectButton)) {
System.out.println("Disconnect Button");
}
if(e.getSource().equals(abortButton)) {
System.out.println("Abort Button");
}
}
}
private class RadioButtonListener implements ActionListener {
@Override
public void actionPerformed(ActionEvent e) {
if(e.getSource().equals(primaryButton)) {
System.out.println("Primary Selected");
}
if(e.getSource().equals(secondaryButton)) {
System.out.println("Secondary Selected");
}
}
}
private class ComboBoxListener implements ActionListener {
@Override
public void actionPerformed(ActionEvent e) {
if(e.getSource().equals(comboBox)) {
System.out.println(comboBox.getSelectedItem());
DataClass dataClass = new DataClass(MyGui.this, otherStuff);
dataClass.doStuff(); // This class was calling the addActivity() method.
}
}
}
private class ListBoxListener implements ListSelectionListener {
@Override
public void valueChanged(ListSelectionEvent e) {
if(e.getSource().equals(aList)) {
System.out.println(aList.getSelectedValue());
}
}
}
}
因此,我正在将(假定类名为MyGui)
MyGui.this
传递给有问题的DataClass
,然后使用addActivity("status update");
因此,我没有执行上述操作,而是创建了以下接口:
public interface GuiUpdater {
void update(MyGui MyGui, String update);
}
并修改了上面的
MyGui
类以实现该接口并以这种方式调用它: @Override
public void update(MyGui myGui, String update) {
// TODO Auto-generated method stub
myGui.addActivity(update);
}
因此我可以从也实现了此接口的其他类中对其进行更新。因此我将MyGui的实例从main传递给其他类。
我认为这和以前做的一样(只是有所不同)。
这是使用接口的正确方法吗?如果没有,那么在不传递GUI实例的情况下从其他类更新GUI的正确方法是什么?
最佳答案
接口用于定义对象可以支持的方法,而无需接口的使用者对对象或对象希望如何实现该接口有很多了解。根据您的情况,您希望在MyGui
上提供一个更新方法,该方法不需要MyGui
类与您的其他代码之间的紧密耦合。如果您这样定义接口,则可以编写所有可能需要更新GUI以仅与GuiUpdater
类型的对象进行交互的代码:
public interface GuiUpdater {
void update(String update);
}
然后,您将修改您的
MyGui
类声明以实现此接口:public class MyGui extends JFrame implements GuiUpdater {
您还必须在
MyGui
内部实现此方法:@Override
public void update(String update) {
this.addActivity(update);
}
这与您所遇到的问题类似,但有一个重要区别。因为它是作为
MyGui
中的方法实现的,所以您可以访问MyGui
实例的所有内部状态(例如this
)。换句话说,您无需将MyGui
实例作为参数传递,因为此方法位于MyGui
内部。现在,我们可以假设一个名为
MyGui
的myGui
实例具有一个可能要更新GUI的函数,例如:public void foo(GuiUpdater updater) {
updater.update("Interfaces are great");
}
您可以像
foo(myGui)
那样调用此函数,因为MyGui
满足该接口。这将foo
的实现与MyGui
的实现解耦,这意味着任何一方都可以与对另一方的实现的更改隔离。不管您如何设计软件(MVC,MVP等),对接口进行编码都是一种好习惯。它隐藏了实现细节,因此减少了更改这些细节的影响。