Java Observer / Observable更新

时间:2016-12-20 21:54:54

标签: java swing observable

我尝试应用Observable / Observer模式但是当我尝试更改JTextPane的文本字段时,我的代码出现了问题。

我有3个班级,PlayControllerSecondWindow这里是他们代码的示例。

public class Play() {
    Controller c = new Controller();
    SecondWindow sw = new SecondWindow();
    c.addObserver(sw)

    c.setText("blabla");
}

我的班级Controller

public class Controller extends Observable(){

    private String text ="";

    private static Controller getInstance() {
        if (instance == null) {
            instance = new Controller();
        }
        return instance;
    }

    public void setText(String s) {
        text = s;
        setChanged();
        notifyObservers();
    }
}

SecondWindow

public class SecondWindow extends JFrame implements Observer{
    private JPanel contentPane;
    private Controller c;
    private JTextPane txt = new JTextPane();

    public SecondWindow () {
        EventQueue.invokeLater(new Runnable() {
            public void run() {
                try {
                    SecondWindow frame = new SecondWindow();
                    frame.setVisible(true);
                } catch (Exception e) {
                    e.printStackTrace();
                }
            }
        });
    }

    public SecondWindow() {
        initComponents();
        createEvents();
        c = Controller.getInstance();
    }

    public void initComponents() {

        setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        setBounds(1000, 0, 300,500);
        contentPane = new JPanel();
        contentPane.setBorder(new EmptyBorder(5, 5, 5, 5));
        setContentPane(contentPane);
        contentPane.setLayout(null);

        txt.setBounds(0, 0, 280, 460);
        txt.enable(false);
        contentPane.add(txt);
    }

    public void update(Observable arg0 , Object arg1){
        // Things to change here
    }

我无法将变量c放在textField中(就像txt.setText(c.getText)指令一样)。我确信它会读取方法更新,但我不知道如何确保它有效。

1 个答案:

答案 0 :(得分:4)

提示:根据Observerable API,notifyObservers方法有一个重载,它接受任何对象作为参数:

public void notifyObservers(Object arg)

这甚至可以是一个字符串。然后根据Observer API,此对象将被传递到观察者的update方法中,您可以在那里使用它。

void update(Observable o,
        Object arg)
     

arg - 传递给notifyObservers方法的参数。

此处有单独的问题:

  

contentPane.setLayout(空);

对于大多数Swing爱好者来说,看到这就像在黑板上听指甲一样 - 这很痛苦。虽然null布局和setBounds()似乎是Swing新手,比如创建复杂GUI的最简单和最好的方法,但是你创建的Swing GUI越多,你在使用它们时会遇到更严重的困难。 。当GUI调整大小时,他们不会调整组件的大小,他们是增强或维护的皇室女巫,当他们放置在滚动窗格中时,他们完全失败,当他们在所有平台或屏幕分辨率不同时看起来很糟糕原来的。相反,您将需要学习和学习布局管理器,然后嵌套JPanels,每个JPanels都使用自己的布局管理器来创建令人满意的复杂GUI,这些GUI在所有操作系统上都很好看。

第二个问题:您的代码不是Swing线程安全的,因为Swing事件调度线程或EDT的可观察 off 可以很好地通知Swing GUI。虽然这个简单的程序不太可能引起频繁或严重的问题,但一般情况下最好使用SwingPropertyChangeSupport和PropertyChangeListeners而不是Observer / Observable。

下一步旁边问题
这个:

public class Controller extends Observable(){

不可编辑/犹太Java。对于SecondWindow类的重复的无参数构造函数也是如此。是的,我们知道你要做的是什么,但尝试理解其他人的代码已经足够困难了,你真的不想通过发布类型来更难那种无法编译的代码,相信我。

例如,可以使用PropertyChangeListeners在Swing中实现一些简单的东西,如下所示:

import java.util.concurrent.TimeUnit;

public class Play2 {
    public static void main(String[] args) {
        Model2 model2 = new Model2();
        View2 view2 = new View2();
        new Controller2(model2, view2);
        view2.show();
        for (int i = 0; i < 10; i++) {
            try {
                TimeUnit.SECONDS.sleep(1);
            } catch (InterruptedException e) {
                // one of the few times it's OK to ignore an exception
            }
            String text = String.format("Counter Value: %d", i);
            model2.setText(text);
        }
    }
}

import java.beans.PropertyChangeListener;
import javax.swing.event.SwingPropertyChangeSupport;

public class Model2 {
    private SwingPropertyChangeSupport pcSupport = new SwingPropertyChangeSupport(this);
    public static final String TEXT = "text"; // name of our "bound" property
    private String text = "";

    public String getText() {
        return text;
    }

    public void setText(String text) {
        String oldValue = this.text;
        String newValue = text;
        this.text = text;
        pcSupport.firePropertyChange(TEXT, oldValue, newValue);
    }

    public void addPropertyChangeListener(PropertyChangeListener listener) {
        pcSupport.addPropertyChangeListener(listener);
    }

    public void removePropertyChangeListener(PropertyChangeListener listener) {
        pcSupport.removePropertyChangeListener(listener);
    }

    public void addPropertyChangeListener(String name, PropertyChangeListener listener) {
        pcSupport.addPropertyChangeListener(name, listener);
    }

    public void removePropertyChangeListener(String name, PropertyChangeListener listener) {
        pcSupport.removePropertyChangeListener(name, listener);
    }
}

import javax.swing.*;

public class View2 {
    private JPanel mainPanel = new JPanel();
    private JTextField textField = new JTextField(10);

    public View2() {
        textField.setFocusable(false);
        mainPanel.add(new JLabel("Text:"));
        mainPanel.add(textField);
    }

    public JPanel getMainPanel() {
        return mainPanel;
    }

    public void setText(String text) {
        textField.setText(text);
    }

    public void show() {
        SwingUtilities.invokeLater(() -> {
            JFrame frame = new JFrame("View");
            frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
            frame.add(getMainPanel());
            frame.pack();
            frame.setLocationRelativeTo(null);
            frame.setVisible(true);
        });
    }
}

import java.beans.PropertyChangeEvent;
import java.beans.PropertyChangeListener;

public class Controller2 {
    private Model2 model2;
    private View2 view2;

    public Controller2(Model2 model2, View2 view2) {
        this.model2 = model2;
        this.view2 = view2;
        model2.addPropertyChangeListener(Model2.TEXT, new ModelListener());
    }

    private class ModelListener implements PropertyChangeListener {
        @Override
        public void propertyChange(PropertyChangeEvent pcEvt) {
            view2.setText((String) pcEvt.getNewValue());
        }
    }
}