美文网首页
使得可变性最小化

使得可变性最小化

作者: 没走过的二丁目 | 来源:发表于2018-04-18 15:14 被阅读0次

不可变类是指:其实例不能被修改的类。每个实例中包含的所有信息都必须在创建该实例的时候就提供,并在对象的整个生命周期内固定不变。

为使类成为不可变类,要遵循下面五条规则:

  • 不要提供任何会修改对象状态的方法
  • 保证类不会被扩展。为了防止子类化,一般做法是使这个类成为final的 。
  • 使所有的域都是final的。一是为了表明意图,二是为了在多线程间确保对对象使用正确的行为。
  • 使所有的域都成为私有的。这样可以防止客户端获得访问被域引用的可变对象的权限,并防止客户端直接修改这些对象。
  • 确保对于任何可变组件的互斥访问。如果类具有指向可变对象的域,则必须确保该类的客户端无法获得指向这些对象的引用。并且,永远不要用客户端提供的对象引用来初始化这样的域,也不要从任何方法中返回该对象引用。

稍微复杂的例子:

public final class Complex {
private final double re;
private final double im;

public Complex(double re, double im){
    this.re = re;
    this.im = im;
}

public double realPart(){return re;}
public double imaginaryPart(){return im;}

public Complex add(Complex c){
    return new Complex(re + c.re, im + c.im);
}

public Complex subtract(Complex c){
    return new Complex(re - c.re, im - c.im);
}

public Complex multiply(Complex c){
    return new Complex(re * c.re - im * c.im, re * c.im + im * c.re);
}

public Complex divide(Complex c){
    double tmp = c.re * c.re + c.im * c.im;
    return new Complex((re * c.re + im * c.im) / tmp, (im * c.re - re * c.im) / tmp);
}

@Override
public boolean equals(Object o){
    if (o == this)
        return true;
    if (!(o instanceof Complex))
        return false;
    Complex c = (Complex)o;
    return Double.compare(re, c.re) == 0 && Double.compare(im, c.im) == 0;
}

@Override
public int hashCode(){
    int result = 17;
    result = 31 * result + hashDouble(re);
    result = 31 * result + hashDouble(im);
    return result;
}

private int hashDouble (double val) {
    long longBits = Double.doubleToLongBits(re);
    return (int) (longBits ^ (longBits >>> 32));
}

这个类表示一个复数(具有实部和虚部)。除了标准的Object方法之外,它还提供了针对实部和虚部的访问方法,以及4种基本的算术运算:加法,减法,乘法和除法。注意这些算术运算是创建并返回新的Complex实例,而不是修改这个实例。大多数的不可变类都使用了这种模式,它被称作函数做法,因为这些方法返回了一个函数的结果,这些函数对操作数进行运算,但是并不修改它。与之对应的更常见的过程是命令式的做法

不可变类的优点

  • 不可变对象比价简单,它只有一种状态,即被创建时候的状态
  • 不可变对象本质上是线程安全的,他们不要求同步,当多个线程并发访问这样的对象时候,他们不会遭到破坏
  • 不可变对象可以被自由的共享,不可变类应该充分利用这种优势,鼓励客户端尽可能的重用现有的实例

缺点

如果执行一个多步骤的操作,并且每个步骤都会产生一个新的对象,除了最后的结果之外,其他的对象都会被丢弃,此时性能问题就会显露出来。处理这种问题有两个办法,一个是先预测一下会经常用到哪些多步骤的操作,将他们作为基本类型提供,如果无法预测,则提供一个公有的可变配套类,如String类的可变配套类StringBuilder

总之,不要为每个get方法写一个set方法。除非有很好的理由要让类可变,否则都应该让类不可变

相关文章

  • 使得可变性最小化

    不可变类是指:其实例不能被修改的类。每个实例中包含的所有信息都必须在创建该实例的时候就提供,并在对象的整个生命周期...

  • 第四章类和接口

    目录 使类和成员的可访问性最小化 在公共类中使用访问方法而不是公共属性 最小化可变性 组合优于继承 如果使用继承则...

  • Effective-java 3 中文翻译系列 (Item 17

    原文链接 文章也上传到的 github (欢迎关注,欢迎大神提点。) ITEM 17 使可变性最小化 一个不可变的...

  • ITEM 17:最小化可变性

    ITEM 17: MINIMIZE MUTABILITY  不可变类就是一个实例不能被修改的类。每个实例中包含的所...

  • 提示十七

    今天来看提示十七:使可变性最小化。 不可变类简单来说是其实例不能被修改的类。 包含在每个实例中的所有信息在对象的生...

  • Rust 编程语言-3-通用编程概念

    3. 通用的编程概念 3.1变量和可变性 变量默认都是immutable mut关键字使得变量可变,但是带来了额外...

  • 第15条:使可变性最小化

    不可变类是指:其实例不能被修改的类。每个实例中包含的所有信息都必须在创建该实例的时候就提供,并在对象的整个生命周期...

  • 使可变性最小化--EffectiveJava小结(15)

    不可变类:每个实例的所有属性都必须再创建该实例时就提供,并且再对象的整个生命周期内固定不变。 不可变类的五个规则:...

  • 批量梯度下降和随机梯度下降

    1.概念 (1)批量梯度下降---最小化所有训练样本的损失函数,使得最终求解的是全局的最优解,即求解的参数是使得风...

  • 第十七条:使可变性最小化

    不可变类是指其实例不可被修改的类。每个实例中包含的所有信息都必须在创建该实例的时候就提供,并在对象的整个声明周期内...

网友评论

      本文标题:使得可变性最小化

      本文链接:https://www.haomeiwen.com/subject/fncpkftx.html