1_基础知识_chapter03_对象的共享_4_不变性

  • 不可变对象

    (1) 满足条件

    1° 对象创建以后状态不能更改

    2° 对象的所有域都是final类型

    3° 对象被正确创建(构造函数中没有发生this引用逸出)

    (2) 不可变对象一定是线程安全的

    (3) 不可变对象的状态可以更新, 更新的方式是通过一个保存新状态的实例来替换原有的不可变对象(例如String, Integer)

  • final域

    (1) 如果final域所引用的对象可变, 那么这些被引用的对象是可以修改的

    (2) final域在Java内存模型中有特殊的语义:

    final域能够确保初始化过程的安全性, 从而在共享这些对象时无须同步(保证了可见性)

    (3) 即使一个对象是可变的, 但是可以将某些域声明为final, 减少可能的状态数

    (4) 除非需要更高的可见性, 否则所有的域都应该是private的;

    除非需要某个域真正可变, 否则所有的域都应该是final的

  • 一种操作是将类中的所有的状态变量封装在一起组成一个新的类,这个类写成不可变类的形式, 在原有类中有volatile声明这个新类的对象, 这样如果更新类的状态, 则将这个新类的对象整体进行更新, 无需显式使用synchronized也可以保证线程安全

    示例

    OneValueCache.java

      @Immutable
      public class OneValueCache {
    
          private final BigInteger lastNumber;
          private final BigInteger[] lastFactors;
    
          public OneValueCache(BigInteger i,
                               BigInteger[] factors) {
    
              lastNumber = i;
              lastFactors = Arrays.copyOf(factors, factors.length);
          }
    
          public BigInteger[] getFactors(BigInteger i) {
    
              if (lastNumber == null || !lastNumber.equals(i)) {
                  return null;
              } else {
                  return Arrays.copyOf(lastFactors, lastFactors.length);
              }
          }
      }
    

    VolatileCachedFactorizer.java

      @ThreadSafe
      public class VolatileCachedFactorizer extends GenericServlet implements Servlet {
    
          private volatile OneValueCache cache = new OneValueCache(null, null);
    
          public void service(ServletRequest req, ServletResponse resp) {
    
              BigInteger i = extractFromRequest(req);
              BigInteger[] factors = cache.getFactors(i);
    
              if (factors == null) {
    
                  factors = factor(i);
                  cache = new OneValueCache(i, factors);
              }
              encodeIntoResponse(resp, factors);
          }
    
          private void encodeIntoResponse(ServletResponse resp, BigInteger[] factors) {
          }
    
          private BigInteger extractFromRequest(ServletRequest req) {
    
              return new BigInteger("7");
          }
    
          private BigInteger[] factor(BigInteger i) {
    
              // Doesn't really factor
              return new BigInteger[]{i};
          }
      }
    

    VolatileCachedFactorizer原本有两个状态变量: lastNumber和lastFactors。现在把它们抽出来组合成一个不可变类OneValueCache。 OneValueCache是一个不可变类, 因为它完全满足不可变对象的三个条件(注意使用了Arrays.copyOf传出lastFactors避免了状态被改变)。

    现在, 在VolatileCachedFactorizer声明了一个不可变对象cache, 并且用volatile修饰保证了可见性, 需要更改它的状态时则new一个新的OneValueCache对象,这样就做到了无需使用synchronized也能保证线程安全。

©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 多线程之 Final变量 详解 原文: http://www.tuicool.com/articles/2Yjmq...
    朦胧蜜桃阅读 1,428评论 0 2
  • 同步的作用 确保复合操作的原子性(复合操线程间作互斥) 内存可见性 volatile 作用:将当前线程对volat...
    WFitz阅读 444评论 0 0
  • 1、一个".java"源文件中是否可以包括多个类(不是内部类)?有什么限制?答:可以有多个类,但只能有一个publ...
    岳小川阅读 981评论 0 2
  • 小编费力收集:给你想要的面试集合 1.C++或Java中的异常处理机制的简单原理和应用。 当JAVA程序违反了JA...
    八爷君阅读 4,731评论 1 114
  • 在微博看到的一句话,他说会为你遮风挡雨,可后来你的大风大浪都是他给的 我一直憧憬那种执子之手与之偕老的老师,羡慕...
    木木逗比君阅读 219评论 0 2