java别名问题
“别名”意味着多个句柄都试图指向同一个工具,就象前面的例子展示的那样。若有人向谁人工具里写入一点什么对象,就会发生别名问题。若其他句柄的所有者不但愿谁人工具改变,恐怕就要失望了。这可用下面这个简朴的例子说明:
//: Alias1.java // Aliasing two handles to one object public class Alias1 { int i; Alias1(int ii) { i = ii; } public static void main(String[] args) { Alias1 x = new Alias1(7); Alias1 y = x; // Assign the handle System.out.println("x: " + x.i); System.out.println("y: " + y.i); System.out.println("Incrementing x"); x.i++; System.out.println("x: " + x.i); System.out.println("y: " + y.i); } } ///:~
对下面这行:
Alias1 y = x; // Assign the handle
它会新建一个Alias1句柄,但不是把它分派给由new建设的一个新鲜工具,而是分派给一个现有的句柄。所以句柄x的内容——即工具x指向的地点——被分派给y,所以无论x照旧y都与沟通的工具毗连起来。这样一来,一旦x的i在下述语句中增值:
x.i++;
y的i值也一定受到影响。从最终的输出就可以看出:
x: 7 y: 7 Incrementing x x: 8 y: 8
此时最直接的一个办理步伐就是爽性不这样做:不要有意将多个句柄指向同一个浸染域内的同一个工具。这样做可使代码更易领略和调试。然而,一旦筹备将句柄作为一个自变量或参数通报——这是Java设想的正常要领——别名问题就会自动呈现,因为建设的当地句柄大概修改“外部工具”(在要领浸染域之外建设的工具)。下面是一个例子:
//: Alias2.java // Method calls implicitly alias their // arguments. public class Alias2 { int i; Alias2(int ii) { i = ii; } static void f(Alias2 handle) { handle.i++; } public static void main(String[] args) { Alias2 x = new Alias2(7); System.out.println("x: " + x.i); System.out.println("Calling f(x)"); f(x); System.out.println("x: " + x.i); } } ///:~
输出如下:
x: 7
Calling f(x)
x: 8
要领改变了本身的参数——外部工具。一旦碰着这种环境,必需判定它是否公道,用户是否愿意这样,以及是不是会造成问题。
凡是,我们挪用一个要领是为了发生返回值,可能用它改变为其挪用要领的谁人工具的状态(要领其实就是我们向谁人工具“发一条动静”的方法)。很少需要挪用一个要领来处理惩罚它的参数;这叫作操作要领的“副浸染”(Side Effect)。所以倘若建设一个会修改本身参数的要领,必需向用户明晰地指出这一环境,并告诫利用谁人要领大概会有的效果以及它的潜在威胁。由于存在这些夹杂和缺陷,所以应该只管制止改变参数。
若需在一个要领挪用期间修改一个参数,且不规划修改外部参数,就应在本身的要领内部建造一个副本,从而掩护谁人参数。本章的大大都内容都是环绕这个问题展开的。