在Model JList中更新ArrayList时,应在View中更新

时间:2012-10-30 14:57:50

标签: java swing model-view-controller arraylist jlist

我现在已经玩了一段时间但没有成功。我已经查看了ListModel,但很难将其实现到我当前的项目中。

我有一个Producer类线程,它将元素添加到Model中的ArrayList。这很好,并且ArrayList正在运行时更新。我的问题是,然后我希望添加到ArrayList的新对象被添加到View类中的JList。我无法看到如何将ListModel或DefaultListModel合并到我当前的设置中。非常感谢。

public class Person
{
    private String name;
    private int    age;

    public Person(String name, int age)
    {
        this.name = name;
        this.age = age;
    }

    public String getName()
    {
        return name;
    }

    public int getAge()
    {
        return age;
    }

    public String toString()
    {
        return this.name;
    }
}

public class Producer extends Thread
{
    private Model model;

    public Producer(Model model)
    {
        this.model = model;
    }

    public void run()
    {
        Person fred     = new Person("Fred Flintstone", 37);
        Person wilma    = new Person("Wilma Flintstone", 18);
        Person pebbles  = new Person("Pebbles Flintstone", 15);
        Person dino     = new Person("Dino Flintstone", 45);
        Person barney   = new Person("Barney Rubble", 76);
        Person betty    = new Person("Betty Rubble", 76);
        Person bamm     = new Person("Bamm-Bamm Rubble", 76);

        try
        {
            model.addPerson(fred);
            Thread.sleep(1500);
            model.addPerson(wilma);
            Thread.sleep(1500);
            model.addPerson(pebbles);
            Thread.sleep(1500);
            model.addPerson(dino);
            Thread.sleep(1500);
            model.addPerson(barney);
            Thread.sleep(1500);
            model.addPerson(betty);
            Thread.sleep(1500);
            model.addPerson(bamm);
        }
        catch(Exception e)
        {
            System.out.println("Error adding Person object to Model.people
                                ArrayList" + e);
        }
    }
}

public class Model 
{
    private List <Person> people;

    public Model()
    {
        people = new ArrayList<Person>();
    }

    public List<Person> getPeople()
    {
        return people;
    }

    public void addPerson(Person aPerson)
    {
        people.add(aPerson);
        System.out.println("Person object added to people list:" + aPerson);
    }

    public void removePerson(Person aPerson)
    {
        people.remove(aPerson);
    }
}

public class View extends JFrame
{
    private JPanel              topPanel, botPanel;
    private JList               peopleList;
    private JScrollPane         scrollPane;
    private Model               model;

    public View(Model model)
    {
        this.model = model;
        setSize(200, 220);
        setTitle("View");
        setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        topPanel = new JPanel();
        botPanel = new JPanel();
        peopleList = new JList(model.getPeople().toArray());
        scrollPane = new JScrollPane(peopleList);
        topPanel.setLayout(new GridLayout(1, 1));
        topPanel.add(scrollPane);
        topPanel.setBorder(BorderFactory.createTitledBorder
       (BorderFactory.createEtchedBorder(), "People list"));

        Container cp = getContentPane();
        cp.add(topPanel, BorderLayout.NORTH);
        cp.add(botPanel, BorderLayout.SOUTH);
    }
}

public class Main 
{
    public static void main(String[] args)
    {
        Model model = new Model();
        View theView = new View(model);
        theView.setVisible(true);
        Producer producer = new Producer(model);
        producer.start();
    }
}

1 个答案:

答案 0 :(得分:3)

在此处使用适当的MVC模式:视图侦听模型中的更改并根据模型中的通知进行更新:

  1. 在模型上添加PropertyChangeSupport并在修改模型时触发PropertyChangeEvent
  2. 将视图添加为模型的PropertyChangeListener
  3. 相应地响应通知(如果您的模型中添加了Person,当您收到“添加了新人”的事件时,请将其添加到ListModel peopleList },同样为“删除的人”等...)
  4. http://docs.oracle.com/javase/7/docs/api/java/beans/PropertyChangeSupport.html

    注意:由于您的模型将被另一个线程修改而不是EDT(事件派发线程),因此请确保在EDT上修改您的UI(请查看SwingUtilities.isEventDispatchingThread()SwingUtilities.invokeLater()

    更新:

    这是一个片段,说明了我上面要解释的内容(抱歉,如果它有点冗长,但我尽量保持与原始代码尽可能接近):

    import java.awt.BorderLayout;
    import java.awt.Container;
    import java.awt.GridLayout;
    import java.beans.PropertyChangeEvent;
    import java.beans.PropertyChangeListener;
    import java.beans.PropertyChangeSupport;
    import java.util.ArrayList;
    import java.util.List;
    import java.util.Random;
    
    import javax.swing.BorderFactory;
    import javax.swing.DefaultListModel;
    import javax.swing.JFrame;
    import javax.swing.JList;
    import javax.swing.JPanel;
    import javax.swing.JScrollPane;
    import javax.swing.SwingUtilities;
    
    public class Main {
    
        public class Person {
    
            private String name;
            private int age;
    
            public Person(String name, int age) {
                this.name = name;
                this.age = age;
            }
    
            public String getName() {
                return name;
            }
    
            public int getAge() {
                return age;
            }
    
            @Override
            public String toString() {
                return this.name;
            }
        }
    
        public class Producer extends Thread {
            private Model model;
    
            public Producer(Model model) {
                this.model = model;
            }
    
            @Override
            public void run() {
                Random random = new Random();
                Person fred = new Person("Fred Flintstone", 37);
                Person wilma = new Person("Wilma Flintstone", 18);
                Person pebbles = new Person("Pebbles Flintstone", 15);
                Person dino = new Person("Dino Flintstone", 45);
                Person barney = new Person("Barney Rubble", 76);
                Person betty = new Person("Betty Rubble", 76);
                Person bamm = new Person("Bamm-Bamm Rubble", 76);
                while (true) {
                    try {
                        model.addPerson(fred);
                        Thread.sleep(1500);
                        model.addPerson(wilma);
                        Thread.sleep(1500);
                        model.addPerson(pebbles);
                        Thread.sleep(1500);
                        model.addPerson(dino);
                        Thread.sleep(1500);
                        model.addPerson(barney);
                        Thread.sleep(1500);
                        model.addPerson(betty);
                        Thread.sleep(1500);
                        model.addPerson(bamm);
                        while (model.getPeople().size() > 0) {
                            Person p = model.getPeople().get(random.nextInt(model.getPeople().size()));
                            model.removePerson(p);
                            Thread.sleep(1000);
                        }
                    } catch (Exception e) {
                        e.printStackTrace();
                    }
                }
            }
        }
    
        public class Model {
            private static final String PEOPLE = "people";
    
            private List<Person> people;
    
            private PropertyChangeSupport propertyChangeSupport;
    
            public Model() {
                people = new ArrayList<Person>();
                propertyChangeSupport = new PropertyChangeSupport(this);
            }
    
            public PropertyChangeSupport getPropertyChangeSupport() {
                return propertyChangeSupport;
            }
    
            public List<Person> getPeople() {
                return people;
            }
    
            public void addPerson(Person aPerson) {
                people.add(aPerson);
                System.out.println("Person object added to people list:" + aPerson);
                getPropertyChangeSupport().firePropertyChange(PEOPLE, null, aPerson);
            }
    
            public void removePerson(Person aPerson) {
                people.remove(aPerson);
                getPropertyChangeSupport().firePropertyChange(PEOPLE, aPerson, null);
            }
        }
    
        public class View extends JFrame implements PropertyChangeListener {
            private JPanel topPanel, botPanel;
            private JList peopleList;
            private JScrollPane scrollPane;
            private Model model;
            private DefaultListModel peopleListModel;
    
            public View(Model model) {
                this.model = model;
                setSize(200, 220);
                setTitle("View");
                setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
                topPanel = new JPanel();
                botPanel = new JPanel();
                peopleListModel = new DefaultListModel();
                for (Person p : model.getPeople()) {
                    peopleListModel.addElement(p);
                }
                peopleList = new JList(peopleListModel);
                model.getPropertyChangeSupport().addPropertyChangeListener(Model.PEOPLE, this);
                scrollPane = new JScrollPane(peopleList);
                topPanel.setLayout(new GridLayout(1, 1));
                topPanel.add(scrollPane);
                topPanel.setBorder(BorderFactory.createTitledBorder(BorderFactory.createEtchedBorder(), "People list"));
    
                Container cp = getContentPane();
                cp.add(topPanel, BorderLayout.NORTH);
                cp.add(botPanel, BorderLayout.SOUTH);
            }
    
            @Override
            public void propertyChange(final PropertyChangeEvent evt) {
                if (!SwingUtilities.isEventDispatchThread()) {
                    SwingUtilities.invokeLater(new Runnable() {
                        @Override
                        public void run() {
                            propertyChange(evt);
                        }
                    });
                    return;
                }
                if (evt.getSource() == model) {
                    if (Model.PEOPLE.equals(evt.getPropertyName())) {
                        if (evt.getOldValue() != null && evt.getNewValue() == null) {
                            peopleListModel.removeElement(evt.getOldValue());
                        } else if (evt.getOldValue() == null && evt.getNewValue() != null) {
                            peopleListModel.addElement(evt.getNewValue());
                        }
                    }
                }
            }
        }
    
        public static void main(String[] args) {
            new Main().init();
        }
    
        private void init() {
            final Model model = new Model();
            SwingUtilities.invokeLater(new Runnable() {
                @Override
                public void run() {
                    View theView = new View(model);
                    theView.setVisible(true);
                }
            });
            Producer producer = new Producer(model);
            producer.start();
        }
    }