标题可能有点绕。。。。
正常当我们设计一个不可变类时,需要保证实例变量是不可变的,如果当个这个类的实例变量是一个引用类型时,而这个引用类型又是可变的该怎么设计这个不可变类?
class Name
{
private String firstName;
private String lastName;
public Name(){};
public Name(String firstName, String lastName)
{
this.firstName= firstName;
this.lastName= lastName;
}
public void setFirstName(String firstName)
{
this.firstName= firstName;
}
public String getFirstName()
{
return firstName;
}
public String getLastName()
{
return lastName;
}
}
public class Person
{
private Name name ;
public Person(Name name)
{
this.name = name;
}
public Name getName()
{
return name;
}
public static void main(String[] args)
{
Name n = new Name("渠梁" , "嬴");
Person p = new Person(n);
System.out.println(p.getName().getFirstName()); //打印渠梁
n.setFirstName("鞅");
System.out.println(p.getName().getFirstName()); //打印鞅
}
}
从上面代码我们可以看到由于引用类型变量name中的firstName被重新设置,导致Person的实例变量name被改变了,这违背了我们开始想创建不可变类的初衷,想让Person中的name实例中的数据是不可变的,我们可以对Person的构造器做如下修改;
class Name
{
private String firstName;
private String lastName;
public Name(){};
public Name(String firstName, String lastName)
{
this.firstName= firstName;
this.lastName= lastName;
}
public void setFirstName(String firstName)
{
this.firstName= firstName;
}
public String getFirstName()
{
return firstName;
}
public String getLastName()
{
return lastName;
}
}
public class Person
{
private Name name ;
public Person(Name name)
{
this.name = new Name(name.getFirstName() , name.getLastName()); //设置临时实例;
}
public Name getName()
{
return new Name(name.getFirstName() , name.getLastName()); //设置临时实例;
}
public static void main(String[] args)
{
Name n = new Name("渠梁" , "嬴");
Person p = new Person(n);
System.out.println(p.getName().getFirstName()); //打印渠梁
n.setFirstName("鞅");
System.out.println(p.getName().getFirstName()); //打印渠梁
}
}
原理就是在Person初始化时,设置一个临时的实例,实例中只是初始化时用了n中的属性,但没有把n本身return,所以当n变量中的属性在初始化后又被改变的时候也不会导致临时实例中的数据发生变化;
学Java的时候学到这里,反复看了几遍才懂,感觉过段时间搞不好我会遗忘,所以这里用自己的话记录敲一遍,加深记忆,方便以后复习。白板敲的代码、文字中的一些术语可能会有一些问题,欢迎大家指正。
网友评论