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

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

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

java本身提供了Collection<E>提供了创建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都有对应的影子对象

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

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

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

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

    WritableMap<String,Binding> widgetBindings=new WritableMap<String,Binding>();
    IObservableMap<String, Binding> unmodifiableWidgetBindings=Observables.unmodifiableObservableMap(widgetBindings);

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏LanceToBigData

OOAD-设计模式(三)之创建型设计模式(5种)

前言   前面介绍了OOAD的基础知识,现在我们来详细的说明一下GOF设计模式中的23种模式,希望大家能够学到东西! 一、工厂方法模式(Factory Meth...

2215
来自专栏大数据钻研

让你分分钟学会 javascript 闭包

闭包,是 javascript 中重要的一个概念,对于初学者来讲,闭包是一个特别抽象的概念,特别是ECMA规范给的定义,如果没有实战经验,你很难从定义去理解它。...

2624
来自专栏大数据架构

Java进阶(六)从ConcurrentHashMap的演进看Java多线程核心技术

2035
来自专栏7号代码

面向对象设计原则

面向对象设计的目标之一在于支持可维持性复用,一方面需要实现设计方案或者源代码的复用,另一方面要确保系统能够易于扩展和修改,具有较好的灵活性。

1354
来自专栏我是攻城师

理解Java里面的代理模式

代理模式是23种设计模式中非常经典的一种模式,在日常生活中到处充满了代理模式的痕迹,常见的比如火车代售点买票,各种公共服务大厅,以及各种网上购物平台其实都可以看...

1031
来自专栏有趣的Python

6-Java面向对象-单例模式

设计模式的官方解释: 一套被反复使用,经过分类编目,多数人知晓的,代码设计经验的总结。

552
来自专栏KK的小酒馆

手绘设计模式结构图

GoF的设计模式一共23个,可以分为3大类:创建型、结构型和行为型,这篇文章主要讨论创建型。

741
来自专栏Java技术分享圈

杨老师课堂_Java教程第五篇之函数运用

今天主要是讲解以下知识点: 1、方法基础知识 2、方法高级内容 3、方法案例

732
来自专栏有趣的Python

8-C++远征之继承篇-学习笔记

公有继承时,public 的继承到public。 protected的继承到protected。

791
来自专栏Java帮帮-微信公众号-技术文章全总结

24(02)多线程锁,线程通讯,线程组,线程池,多线程三种方式,匿名内部类,定时器,设计模式,单例模式,Runtime

(6)多线程实现的第三种方案 package cn.itcast_09;(1) import java.util.concurrent.Callable; //...

3904

扫码关注云+社区