“发布(Publish)”一个对象的意思是指,使对象能够在当前作用域之外的代码中使用
当某个不应该发布的对象被发布时,这种情况称之为“逸出(Escape)”
发布就是把对象的引用传到在当前作用域之外,发布是基于JAVA的引用传递机制,所以这里指的是对象,不包括基本类型,发布本身是一种技术或者机制,并无好坏之分。不“希望”发布的对象却被发布了,那么就说这个对象逸出了。所以说这种现象是根据实际情况决定的,如果一个对象你“希望”他发布且发布了,那么就是正常发布,如果一个对象你不“希望”被发布却发布了,那么就是逸出。针对逸出情况,可能会对程序运行造成影响,特别是多线程处理,也有可能并不影响程序运行。
下面我们看下几种发布对象的方式
- 将一个指向该对象的引用保存到其他代码可以访问的地方--方法体内的对象引用保存到类的公有静态变量
例子中发布的对象为set,如果set添加元素,则这些元素也被发布。在这里发布的意思就是在当前作用域中(intitialize方法体中)set对象的引用保存在knownSecret中,这样就可以在当前作用域外通过knownSecret引用操纵set对象,包括set中的元素。
- 将一个指向该对象的引用保存到其他代码可以访问的地方--类的私有对象的引用通过公有方法返回
例子中states对象本来只作用于类内,但是现在其他对象可以通过getStates方法得到states的引用并修改状态。
- 将一个指向该对象的引用保存到其他代码可以访问的地方--类对象传递给外部方法
例子中描述的Study类的studyself方法中,因调用了Enjoy类的watchTV方法导致自身person数据状态发生改变。
以上三种方式都是把对象的引用发布到外部,导致外部在持有对象引用时可以修改数据状态。
以下两种发布方式发布的对象为this。
- 内部类的方式
person是Outer的内部属性,但是其他类可以通过内部类Inner去访问修改。只要是Outer的属性,即可通过this访问的属性都发布出去了。
- 构造时发布this引用
输出结果:
构造前:Person [age=0, name=buff]
构造后:Person [age=10, name=oba]
线程修改后::Person [age=10, name=oba]
这种情况主要是在对象还没有构造完成就对其引用,并修改了数据状态。
使用封装的主要原因:封装能够使得对程序的正确性进行分析变得可能,并使得无意中破坏设计约束条件变得更难。
阅读自此,有种豁然开朗的感觉,才知道在设计类时对于权限修饰符的选择是以此为出发点的的。以上的发布情况,都可以通过权限修饰符进行合理的处理,但这样的话就无法进行正常的发布,无法实现线程间的数据共享。所以本书后面的章节会提出解决方式。