对象的发布与逸出

Posted 飞鸟集

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了对象的发布与逸出相关的知识,希望对你有一定的参考价值。

介绍发布与逸出的定义及代码示范,以及如何使用工厂方法避免this引用在构造方法中逸出

发布:

“发布”一个对象是指,使对象能够在当前作用域之外的代码中使用。

例如:将一个指向该对象的引用保存到其他代码可以访问的地方,或者在某一个非私有的方法中返回该对象的引用,或者将引用传递到其他类的方法中。

代码示范:将一个指向该对象的引用,保存到其他代码可以访问的地方。

// 保存在一个共有的静态变量中
public static Set<Secret> knowSecrets;

public void initialize(){
    knowSecrets = new HashSet<Secret>();
}

逸出:

“逸出”是指,某个不应该发布的对象被发布出去。

当发布某个对象时,可能会间接地发布其他对象。例如上述代码,若将一个 Secret 对象添加到 knownSecret 中,那么会发布这个 Secret 对象;因为任何代码都可以遍历这个集合,并获得对  Secret 对象的引用。同样,如果从非私有方法中返回一个引用,则会发布返回对象。

代码示范:某一个非私有的方法中返回私有对象的引用。

class UnsafeStates{
    private String[] states = new String[]{"AK","AL"};
    public String[] getStates(){
        return states;
    }
}

上述代码中,数组 states 已经逸出了它所在的作用域,因为这个本应是私有的变量,已经被发布了。

逸出范围:当一个对象A被发布时,在A的非私有域中引用的所有对象同样会被发布。也就是,一个已经发布的对象A,能够通过非私有的变量引用和方法调用到达其他的对象,那么所能够到达的对象,均会跟随A一起发布。

此处给出一个定义:“外部方法”。假如有一个类C,对于C来说,“外部方法”是指行为不完全由C来规定的方法,包括其他类中定义的方法以及类C中可以被改写的方法(既不是[private]方法,也不是[final]方法)。

当把一个对象传递给外部方法时,则该对象就会面临一定的危险,因为你不知道外部方法会对该对象做些什么,因此我们需要使用封装。封装能够使得对程序的正确性进行分析变得可能,并使得无意中破坏设计约束条件变得困难。

工厂方法避免this引用在构造方法中逸出:

首先了解 this 引用是如何在构造方法中逸出的。

先看一段代码:发布一个内部类的实例。

public class ThisEscape{
    public ThisEscape(EventSource source){
        source.registerListener(
            new EventListener(){
                public void onEvent(Event e){
                    doSomething(e);
                }
            });
    }
}

上述代码中,当 ThisEscape 发布 EventListener 时,也隐含发布了 ThisEscape 实例本身,因为这个内部类的实例中包含了对 ThisEscapse 实例的隐含引用。只要其他线程在ThisEscape未构造之前(构造返回状态)调用这个类,那么this就会被新建线程共享并识别它(线程溢出)。

下面的代码对上面的示例进行解释:

public class ThisEscape {
     int i = 100; 
     public ThisEscape(int j){
            new Thread(new Runnable() {
                @Override
                public void run() {
                    System.out.println(i + j);
                }
            }).start();
        }
     
     public static void main(String[] args) {
        ThisEscape thisE = new ThisEscape(100);
    }
}

上述代码给出了逸出的一个特殊示例,即 this 引用在构造方法中逸出。当内部的 new Thread  发布时,在外部封装的 ThisEscape 实例也逸出了,于是可以取到 i 值与 j 进行计算。因此当从对象的构造方法中发布对象时,只是发布了一个尚未构造完成的对象。

在构造方法中使用 this 引用逸出的常见错误:在构造方法中启动一个线程;在构造方法中调用一个可改写的实例方法。

如果想在构造方法中注册一个事件监听器或启动线程,那么可以使用一个私有的构造方法和一个公共的工厂方法,从而避免不必要的构造过程。如以下程序所示:

public class SafeListener{
    private final EventListener listener;
    
    private SafeListener(){
        listener = new EventListener(){
            public void onEvent(Event e){
                doSomething(e);
            }
        }
    }
    
    public static SafeListener newInstance(EventSource source){
        SafeListener safe = new SafeListener();
        source.registerListener(safe.listener);
        return safe;
    }
}

上面的代码为在构造方法中注册一个事件监听器,新建的线程无法在构造方法之前共享和识别 safe。

这边有一个用上述方法构造 JDBC 连接池的例子:http://blog.csdn.net/luozhonghua2014/article/details/40622207 。虽然上述方法构造的连接池比较安全,但在实际应用中,如:org.apache.tomcat.jdbc.pool.ConnectionPool ,并没有用到链接中给出的样式。

以上是关于对象的发布与逸出的主要内容,如果未能解决你的问题,请参考以下文章

Java并发

如何安全发布对象

Java线程安全性中的对象发布和逸出

java 的知识要多多了解。(下文为转发)

Java多线程——volatile关键字发布和逸出

《java并发编程实战》