java高并发6.1 创建不可变对象

不可变对象

    需要根据实际对象是否可以做成不可变对象, 如果可以,尽量变成不可变对象 , 这样一来在多线程环境下就不会有线程安全的问题了*

1、不可变对象

有一种对象只要它发布了就是安全的,它就是不可变对象。一个不可变对象需要满足的条件:

  • 对象创建一个其状态不能修改

  • 对象所有域都是final类型

  • 对象是正确创建的(在对象创建期间,this引用没有逸出)

2、创建一个不可变对象的方法

(1)自己定义

这里可以采用的方式包括:

1、将类声明为final,这样它就不能被继承。

2、将所有的成员声明为私有的,这样就不允许直接访问这些成员。

3、对变量不提供set方法,将所有可变的成员声明为final,这样就只能赋值一次。通过构造器初始化所有成员进行深度拷贝。

4、在get方法中不直接返回对象的本身,而是克隆对象,返回对象的拷贝。

(2)使用Java中提供的Collection类中的各种unmodifiable开头的方法

(3)使用Guava中的Immutable开头的类

3、final关键字

final关键字可以修饰类、修饰方法、修饰变量

  • 修饰类:类不能被继承(String, integer ,long)。

基础类型的包装类都是final类型的类。final类中的成员变量可以根据需要设置为final,但是要注意的是,final类中的所有成员方法都会被隐式的指定为final方法

  • 修饰方法

(1)把方法锁定,以防任何继承类修改它的含义

(2)效率:在早期的java实现版本中,会将final方法转为内嵌调用。但是如果方法过于庞大,可能看不见效果。一个private方法会被隐式的指定为final方法

  • 修饰变量

  1. 基本数据类型变量,在初始化之后,它的值就不能被修改了。

  2. 如果是引用类型变量,在它初始化之后便不能再指向另外的对象。(但能修改其中的值*  如Map,所以容易引发线程安全问题)

java高并发6.1 创建不可变对象_第1张图片

从上图我们可见,

(1)对一个被final修饰的变量(Integer a、String b)被赋值时在编译过程中就出现了错误。

(2)(map)在重新被指向一个新的map对象的时候也出现了错误。

那么对被定义为final的map进行赋值呢?我们单独运行map.put(1,3)语句,结果是可以的。被final修饰的引用类型变量,虽然不能重新指向,但是可以修改,这一点尤为要注意。

____________________________________________________________________________

除了final是否还有其他的手段可以定义不可变对象呢?

4.unmodifiable相关方法

使用Java的Collection类的unmodifiable相关方法,可以创建不可变对象。unmodifiable相关方法包含:Collection、List、Map、Set….

java高并发6.1 创建不可变对象_第2张图片

运行结果:

会抛出异常, 由此可见map对象已经成为不可变对象。

5.Guava:Immutable相关类

使用Guava的Immutable相关类也可以创建不可变对象。同样包含很多类型:Collection、List、Map、Set….

java高并发6.1 创建不可变对象_第3张图片

 

你可能感兴趣的:(高并发,并发编程)