一位经验丰富的开发人员告诉我,传递一个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内部。

现在,我们可以假设一个名为MyGuimyGui实例具有一个可能要更新GUI的函数,例如:

public void foo(GuiUpdater updater) {
    updater.update("Interfaces are great");
}


您可以像foo(myGui)那样调用此函数,因为MyGui满足该接口。这将foo的实现与MyGui的实现解耦,这意味着任何一方都可以与对另一方的实现的更改隔离。

不管您如何设计软件(MVC,MVP等),对接口进行编码都是一种好习惯。它隐藏了实现细节,因此减少了更改这些细节的影响。

07-24 15:27