jface databinding: 创建readonly(只读)可观察对象(observable)

java与C++有一点不同:C++有const关键字,使用const关键字,用于指定一个参数、成员变量或函数是只读不可修改的,通过const参数让对象成为readonly的,可以严格禁止外部调用修改对象的内容。而java没有类似的限制,只有一个final指定变量不可以被赋值,但还是可以通过调用变量的方法任意修改变量指向的对象内部状态。也就是说,java没有像C++那样有绝对readonly的限制。
但有的时候,对象向外部提供返回的值,并不希望外部调用者修改,怎么办呢?
对于以java.util.Collection为基类的对象(Map,Set,List….),java在java.util.Collections提供了一系列名字以unmodifiable为前缀的静态方法,可以创建指定Collection对象的一个副本,这个副本与原对象拥有相同的内容,但是只能读取,不可以修改,任何试图调用修改原对象内容的方法,都会抛出UnsupportedOperationException异常,这就有点像C++的const常量了。
以List为例:

    List list=new ArrayList();
    list.add(122);
    list.add(5);
    List listReadOnly=java.util.Collections.unmodifiableList(list);
    for(Integer i:listReadOnly)
        System.out.println(i); // 可正常读取
    listReadOnly.add(100)// 抛出 UnsupportedOperationException异常

java本身提供了Collection提供了创建readonly对象的方式,但是限制于java本身的机制,java并没有为普通的Object提供类似的普适的方法来创建readonly对象。

因为jface databinding本身的现实需求,jface 数据绑定技术中提供了对Observable对象的影子对象的创建方法。
比如,我们有时需要向外部提供一个Observable对象,以用于调用者观察这个对象的改变,但又不允许调用者修改对象,这时候,就不能把Observable对象直接提供给调用者,而是希望提供一个只读(readonly)的副本,或者叫影子对象,调用者可以通过这个影子对象感知真正的Observable对象的所有数据改变,但不能修改它,说白了—就是只许看不许摸。
下面是DataBindingContext的构造函数代码,就有这种影子对象的应用:

    public DataBindingContext(Realm validationRealm) {
        Assert.isNotNull(validationRealm, "Validation realm cannot be null"); //$NON-NLS-1$
        this.validationRealm = validationRealm;

        ObservableTracker.setIgnore(true);
        try {
            bindings = new WritableList(validationRealm);
            // 为bindings成员变量创建一个影子对象
            unmodifiableBindings = Observables
                    .unmodifiableObservableList(bindings);

            validationStatusProviders = new WritableList(validationRealm);
            // 为validationStatusProviders成员变量创建一个影子对象
            unmodifiableStatusProviders = Observables
                    .unmodifiableObservableList(validationStatusProviders);

            validationStatusMap = new ValidationStatusMap(validationRealm,
                    bindings);
        } finally {
            ObservableTracker.setIgnore(false);
        }
    }
    // getBindings实际返回的是替身unmodifiableBindings,而不是bindings真身
    public final IObservableList getBindings() {
        return unmodifiableBindings;
    }
    // getValidationStatusProviders实际返回的是替身unmodifiableStatusProviders,而不是validationStatusProviders真身
    public final IObservableList getValidationStatusProviders() {
        return unmodifiableStatusProviders;
    }

从下面的类型层次结构图中可以看到,对于Map,Set,List以及普通的ObservableValue对象,jface都有对应的影子对象
jface databinding: 创建readonly(只读)可观察对象(observable)_第1张图片

调用org.eclipse.core.databinding.observable.Observables中对应的静态方法,就可以为Observable对象创建对应的影子对象:

jface databinding: 创建readonly(只读)可观察对象(observable)_第2张图片

如下为一个String类型Observable对象name创建了对应的影子对象unmodifiableName

            WritableValue name = new WritableValue();
            IObservableValue unmodifiableName = Observables.unmodifiableObservableValue(name);

如下为一个ObservableMap对象widgetBindings创建了对应的影子对象unmodifiableWidgetBindings

    WritableMap widgetBindings=new WritableMap();
    IObservableMap unmodifiableWidgetBindings=Observables.unmodifiableObservableMap(widgetBindings);

你可能感兴趣的:(java,ui,databinding)