Oracle官方并发教程之不可变对象

原文链接译文链接,译者:Greenster,校对:郑旭东

一个对象如果在创建后不能被修改,那么就称为不可变对象。在并发编程中,一种被普遍认可的原则就是:尽可能的使用不可变对象来创建简单、可靠的代码。
在并发编程中,不可变对象特别有用。由于创建后不能被修改,所以不会出现由于线程干扰产生的错误或是内存一致性错误。
但是程序员们通常并不热衷于使用不可变对象,因为他们担心每次创建新对象的开销。实际上这种开销常常被过分高估,而且使用不可变对象所带来的一些效率提升也抵消了这种开销。例如:使用不可变对象降低了垃圾回收所产生的额外开销,也减少了用来确保使用可变对象不出现并发错误的一些额外代码。
接下来看一个可变对象的类,然后转化为一个不可变对象的类。通过这个例子说明转化的原则以及使用不可变对象的好处。

一个同步类的例子

SynchronizedRGB是表示颜色的类,每一个对象代表一种颜色,使用三个整形数表示颜色的三基色,字符串表示颜色名称。

01 public class SynchronizedRGB {
02  
03     // Values must be between 0 and 255.
04     private int red;
05     private int green;
06     private int blue;
07     private String name;
08  
09     private void check(int red,
10                        int green,
11                        int blue) {
12         if (red < 0 || red > 255
13             || green < 0 || green > 255
14             || blue < 0 || blue > 255) {
15             throw new IllegalArgumentException();
16         }
17     }
18  
19     public SynchronizedRGB(int red,
20                            int green,
21                            int blue,
22                            String name) {
23         check(red, green, blue);
24         this.red = red;
25         this.green = green;
26         this.blue = blue;
27         this.name = name;
28     }
29  
30     public void set(int red,
31                     int green,
32                     int blue,
33                     String name) {
34         check(red, green, blue);
35         synchronized (this) {
36             this.red = red;
37             this.green = green;
38             this.blue = blue;
39             this.name = name;
40         }
41     }
42  
43     public synchronized int getRGB() {
44         return ((red << 16) | (green << 8) | blue);
45     }
46  
47     public synchronized String getName() {
48         return name;
49     }
50  
51     public synchronized void invert() {
52         red = 255 - red;
53         green = 255 - green;
54         blue = 255 - blue;
55         name = "Inverse of " + name;
56     }
57 }

使用SynchronizedRGB时需要小心,避免其处于不一致的状态。例如一个线程执行了以下代码:

1 SynchronizedRGB color =
2     new SynchronizedRGB(0, 0, 0, "Pitch Black");
3 ...
4 int myColorInt = color.getRGB();      //Statement 1
5 String myColorName = color.getName(); //Statement 2

如果有另外一个线程在Statement 1之后、Statement 2之前调用了color.set方法,那么myColorInt的值和myColorName的值就会不匹配。为了避免出现这样的结果,必须要像下面这样把这两条语句绑定到一块执行:

1 synchronized (color) {
2     int myColorInt = color.getRGB();
3     String myColorName = color.getName();
4 }

这种不一致的问题只可能发生在可变对象上。

定义不可变对象的策略

以下的一些规则是创建不可变对象的简单策略。并非所有不可变类都完全遵守这些规则,不过这不是编写这些类的程序员们粗心大意造成的,很可能的是他们有充分的理由确保这些对象在创建后不会被修改。但这需要非常复杂细致的分析,并不适用于初学者。

  1. 不要提供setter方法。(包括修改字段的方法和修改字段引用对象的方法)
  2. 将类的所有字段定义为final、private的。
  3. 不允许子类重写方法。简单的办法是将类声明为final,更好的方法是将构造函数声明为私有的,通过工厂方法创建对象。
  4. 如果类的字段是对可变对象的引用,不允许修改被引用对象。
    • 不提供修改可变对象的方法。
    • 不共享可变对象的引用。当一个引用被当做参数传递给构造函数,而这个引用指向的是一个外部的可变对象时,一定不要保存这个引用。如果必须要保存,那么创建可变对象的拷贝,然后保存拷贝对象的引用。同样如果需要返回内部的可变对象时,不要返回可变对象本身,而是返回其拷贝。

将这一策略应用到SynchronizedRGB有以下几步:

  1. SynchronizedRGB类有两个setter方法。第一个set方法只是简单的为字段设值(译者注:删掉即可),第二个invert方法修改为创建一个新对象,而不是在原有对象上修改。
  2. 所有的字段都已经是私有的,加上final即可。
  3. 将类声明为final的
  4. 只有一个字段是对象引用,并且被引用的对象也是不可变对象。

经过以上这些修改后,我们得到了ImmutableRGB

01 final public class ImmutableRGB {
02  
03     // Values must be between 0 and 255.
04     final private int red;
05     final private int green;
06     final private int blue;
07     final private String name;
08  
09     private void check(int red,
10                        int green,
11                        int blue) {
12         if (red < 0 || red > 255
13             || green < 0 || green > 255
14             || blue < 0 || blue > 255) {
15             throw new IllegalArgumentException();
16         }
17     }
18  
19     public ImmutableRGB(int red,
20                         int green,
21                         int blue,
22                         String name) {
23         check(red, green, blue);
24         this.red = red;
25         this.green = green;
26         this.blue = blue;
27         this.name = name;
28     }
29  
30     public int getRGB() {
31         return ((red << 16) | (green << 8) | blue);
32     }
33  
34     public String getName() {
35         return name;
36     }
37  
38     public ImmutableRGB invert() {
39         return new ImmutableRGB(255 - red,
40                        255 - green,
41                        255 - blue,
42                        "Inverse of " + name);
43     }
44 }
时间: 2024-10-02 19:28:06

Oracle官方并发教程之不可变对象的相关文章

Oracle官方并发教程之不可变对象(定义不可变对象的策略)

原文链接,译文链接,译者:Greenster,校对:郑旭东 以下的一些规则是创建不可变对象的简单策略.并非所有不可变类都完全遵守这些规则,不过这不是编写这些类的程序员们粗心大意造成的,很可能的是他们有充分的理由确保这些对象在创建后不会被修改.但这需要非常复杂细致的分析,并不适用于初学者. 不要提供setter方法.(包括修改字段的方法和修改字段引用对象的方法) 将类的所有字段定义为final.private的. 不允许子类重写方法.简单的办法是将类声明为final,更好的方法是将构造函数声明为私

Oracle官方并发教程

计算机的使用者一直以为他们的计算机可以同时做很多事情.他们认为当其他的应用程序在下载文件,管理打印队列或者缓冲音频的时候他们可以继续在文字处理程序上工作.甚至对于单个应用程序,他们任然期待它能在在同一时间做很多事情.举个例子,一个流媒体播放程序必须能同时完成以下工作:从网络上读取数字音频,解压缩数字音频,管理播放和更新程序显示.甚至文字处理器也应该能在忙于重新格式化文本和刷新显示的情况下同时响应键盘和鼠标事件.这样的软件就被称为并发软件. 通过Java语言和Java类库对于基础并发的支持,JAV

Oracle官方并发教程之一个同步类的例子

原文链接,译文链接,译者:Greenster,校对:郑旭东 SynchronizedRGB是表示颜色的类,每一个对象代表一种颜色,使用三个整形数表示颜色的三基色,字符串表示颜色名称. 01 public class SynchronizedRGB { 02   03     // Values must be between 0 and 255. 04     private int red; 05     private int green; 06     private int blue;

Oracle官方并发教程之高级并发对象

原文地址,译文地址 译者:李任 目前为止,该教程重点讲述了最初作为Java平台一部分的低级别API.这些API对于非常基本的任务来说已经足够,但是对于更高级的任务就需要更高级的API.特别是针对充分利用了当今多处理器和多核系统的大规模并发应用程序. 本节,我们将着眼于Java 5.0新增的一些高级并发特征.大多数特征已经在新的java.util.concurrent包中实现.Java集合框架中也定义了新的并发数据结构. 锁对象提供了可以简化许多并发应用的锁的惯用法. Executors为加载和管

Oracle官方并发教程之线程对象

原文链接,译文链接,译者:郑旭东 在Java中,每个线程都是Thread类的实例.并发应用中一般有两种不同的线程创建策略. 直接控制线程的创建和管理,每当应用程序需要执行一个异步任务的时候就为其创建一个线程 将线程的管理从应用程序中抽象出来作为执行器,应用程序将任务传递给执行器,有执行器负责执行. 这一节,我们将讨论Thread对象,有关Executors将在高级并发对象一节中讨论. 定义并启动一个线程 应用程序在创建一个线程实例时,必须提供需要在线程中运行的代码.有两种方式去做到这一点: 提供

Oracle官方并发教程之锁对象

原文地址,译文地址,译者:李任,校对:郑旭东 同步代码依赖于一种简单的可重入锁.这种锁使用简单,但也有诸多限制.java.util.concurrent.locks包提供了更复杂的锁.我们不会详细考察这个包,但会重点关注其最基本的接口,锁. 锁对象作用非常类似同步代码使用的隐式锁.如同隐式锁,每次只有一个线程可以获得锁对象.通过关联Condition对象,锁对象也支持wait/notify机制. 锁对象之于隐式锁最大的优势在于,它们有能力收回获得锁的尝试.如果当前锁对象不可用,或者锁请求超时(如

Oracle官方并发教程之同步

原文链接,译文链接,译者:蘑菇街-小宝,Greenster,李任  校对:丁一,郑旭东,李任 线程间的通信主要是通过共享域和引用相同的对象.这种通信方式非常高效,不过可能会引发两种错误:线程干扰和内存一致性错误.防止这些错误发生的方法是同步. 不过,同步会引起线程竞争,当两个或多个线程试图同时访问相同的资源,随之就导致Java运行时环境执行其中一个或多个线程比原先慢很多,甚至执行被挂起,这就出现了线程竞争.线程饥饿和活锁都属于线程竞争的范畴.关于线程竞争的更多信息可参考活跃度一节. 本节内容包括

Oracle官方并发教程之进程和线程

原文链接,译文链接,译者:bjsuo,校对:郑旭东 在并发编程中,有两个基本的执行单元:进程和线程.在java语言中,并发编程最关心的是线程,然而,进程也是非常重要的. 即使在只有单一的执行核心的计算机系统中,也有许多活动的进程和线程.因此,在任何给定的时刻,只有一个线程在实际执行.处理器的处理时间是通过操作系统的时间片在进程和线程中共享的. 现在具有多处理器或有多个执行内核的多处理器的计算机系统越来越普遍,这大大增强了系统并发执行的进程和线程的吞吐量–但在不没有多个处理器或执行内核的简单的系统

Oracle官方并发教程之执行器(Executors)

原文链接,译文链接,译者:Greenster,校对:郑旭东 在之前所有的例子中,Thread对象表示的线程和Runnable对象表示的线程所执行的任务之间是紧耦合的.这对于小型应用程序来说没问题,但对于大规模并发应用来说,合理的做法是将线程的创建与管理和程序的其他部分分离开.封装这些功能的对象就是执行器,接下来的部分将讲详细描述执行器. 执行器接口定义了三种类型的执行器对象. 线程池是最常见的一种执行器的实现. Fork/Join是JDK 7中引入的并发框架. 文章转自 并发编程网-ifeve.