PropertyChangeSupport的使用
PropertyChangeSupport的使用,有这么几个类是相关的PropertyChangeEvent、PropertyChangeListener;
?
看一个小的例子:
?
public class A_listViewerExample {public static void main(String[] args) {Display display = Display.getDefault();Shell shell = new Shell(display);shell.setSize(400, 400);shell.setLayout(new FillLayout());ListViewer viewer = new ListViewer(shell, SWT.BORDER);viewer.setLabelProvider(new LabelProvider());viewer.setContentProvider(new ContentProvider());viewer.setInput(getData());((ListModel)viewer.getInput()).printInfo();shell.open();viewer.add(new UserModel("3","Ascos"));((ListModel)viewer.getInput()).add(new UserModel("4","Jack"));((ListModel)viewer.getInput()).printInfo();//viewer.refresh();while (!shell.isDisposed())if (!display.readAndDispatch())display.sleep();}private static Object getData() {ListModel input = new ListModel();input.add(new UserModel("1","Chris"));input.add(new UserModel("2","Leon"));return input;}}class LabelProvider implements ILabelProvider {public Image getImage(Object element) {return Display.getCurrent().getSystemImage(SWT.ICON_WORKING);}public String getText(Object element) {if(element instanceof UserModel)return ((UserModel)element).getName();return element.toString();}public boolean isLabelProperty(Object element, String property) {return false;}public void dispose() {}public void addListener(ILabelProviderListener listener) {}public void removeListener(ILabelProviderListener listener) {}}class ContentProvider implements IStructuredContentProvider, PropertyChangeListener {private ListViewer viewer;public Object[] getElements(Object inputElement) {if(inputElement instanceof ListModel)return ((ListModel)inputElement).elements();return null;}public void inputChanged(Viewer viewer, Object oldInput, Object newInput) {this.viewer = (ListViewer) viewer;if (oldInput instanceof ListModel)((ListModel) oldInput).removePropertyChangeListener(this);if (newInput instanceof ListModel)((ListModel) newInput).addPropertyChangeListener(this);}public void propertyChange(PropertyChangeEvent event) {if(event.getPropertyName().equals(ListModel.ADD_ELEMENT))viewer.add(event.getNewValue());if(event.getPropertyName().equals(ListModel.REMOVE_ELEMENT))viewer.remove(event.getNewValue());}public void dispose() {}}class ListModel {public static final String ADD_ELEMENT = "add Element";public static final String REMOVE_ELEMENT = "remove Element";private PropertyChangeSupport delegate;private Vector<Object> content;public ListModel() {content = new Vector<Object>();delegate = new PropertyChangeSupport(this);}public void addPropertyChangeListener(PropertyChangeListener listener) {delegate.addPropertyChangeListener(listener);}public void removePropertyChangeListener(PropertyChangeListener listener) {delegate.removePropertyChangeListener(listener);}public void firePropertyChange(PropertyChangeEvent event) {delegate.firePropertyChange(event);}public void add(Object element) {if (content.add(element))firePropertyChange(new PropertyChangeEvent(this, ADD_ELEMENT, null,element));}public void remove(Object element) {if (content.remove(element))firePropertyChange(new PropertyChangeEvent(this, REMOVE_ELEMENT, null, element));}public Object[] elements() {return content.toArray();}public void printInfo() {for(Object o : content) {if(o instanceof UserModel) {UserModel user = (UserModel)o;System.out.println("The user id is "+user.getId() + ", and name is " + user.getName());}}}}class UserModel {private String id;private String name;public UserModel(String id, String name) {this.id = id;this.name = name;}public void setId(String id) {this.id = id;}public void setName(String name) {this.name = name;}public String getId() {return id;}public String getName() {return name;}?

?
一个简单的Jface的ListViewer的例子,这是一个PropertyChangeSupport的典型的在GUI的中的应用,在GUI中有一个普遍的问题,就是数据和图形同步的问题,如上面的例子ListViewer,【viewer.add(new UserModel("3","Ascos"));】这个就是仅仅在图像中的增加了数据项,而在对应的数据原型上面listModel上面根本没有增加数据项,这个带来了一个问题,就是图形显示的和底层数据的不一致的问题,这样的话,要是GUI一但刷新的画,就会出现问题了,应该通常的数据刷新都是GUI根据底层数据来刷新,就发现刷新完的GUI数据项“Ascos”不见了。上面的例子,可以将【viewer.refresh();】这个注释的语句打开就可以测试一下。
?
为了保持数据和GUI的同步,只有Data.add同时View.add,还要Data.remove的同时View.remove;这个办法能够解决上面的数据和GUI显示同步的问题,但是对于后期的维护和功能的扩展是非常不方便的,如考虑GUI的常常出现的一种扩展的需求,如上面的List的例子,现在老板觉得是不是table的显示或者tree或者tabletree或者graphics的显示是不是更好一些呢,于是有了view1、view2、view3、在没有一个数据add的地方,加上对view的add,remove的地方也是。这样的修改,常常出现遗漏和错误。
?
应该PropertyChangeSupport,就可以解决上面的问题,数据看成是属性的提供方,所有的view都看着做是,这个属性的监听方,这样不管你加入多少的view,在创建view的时候,就注册到属性提供方中去,然后数据的变化就会。
?
其实,这个也很好的实现了GUI中的基本思想就MVC的思想,M就是基本的数据单元,V就是各种View,C就是逻辑业务单元的处理,这个负责M的创建也就是Data的创建,增加,删除等操作,V自动对应到Data模型,是多么完美的事情啊。
?
不过,这种属性监听的模式,和观察者的模式,非常的相似,观察者=监听者;被观察者=背监听者=管理者;
?
?