前几天,网上找了套java题目,给项目组成员做了一次考试,意图是让大家知道很多基础概念还不一定清楚,于是应该good good study,day day up。
考试之后,有同事问起一个“by value”知识点相关的题目:
Given the following code:
public class Test{
public static void main(String args[])
{
String str=new String(World);
char ch[]={'H','e','l','l','o'};
change(str,ch);
//System.out.println(str + " and " + ch);
//这里,根据二楼、三楼兄弟指出的问题修改了题目。(多谢指正)
System.out.print(str + " and ");
System.out.println(ch);
}
public static void change(String str, char ch[])
{
str="Changed"; ch[0]='C';
}
}
What is the result after execution?
A. World and Hello
B. World and Cello
C. Changed and Hello
D. Changed and Cello
其他同事回信解释,涉及到“final class”和“不变类”两个概念。看来非常基础的概念也不是所有人都清楚,此次我把我回信的内容放在博客中,做个备份。可能会对非常非常初学的初学者有用。
String类是final的没错,这只是说明它不能够被其他类extends。跟这里的题目是没有关系的。
同时,String是不变类。没有任何一个语句可以在已经生成的String对象上做改变。比如"CC"是一个String对象,不可能有语句把"CC"的,比方说,第二个字符由"C"改成,比方说,"A";"CC"自从它被初始化,就永远是"CC","CA"是另一个对象。是不是不变类,跟这里的题目也没关系。
再比如另一段代码:
public static void main(String[] args) {
Person p = new Person("me");
change(p);
System.out.print(p.getName());//应该打印什么?
}
public static void change(Person p) {
p = new Person("another one");
p.setName("third one");
}
class Person {
private String name;
Person(String aName) {
name = aName;
}
void setName(String aName) {//可以看出,Person不是不变类。
name = aName;
}
String getName() {
return name;
}
}
原来的题目中,change方法体中,str="Change";这句话是把str这个原本指向"World"对象的String类引用,指向了另一个对象"Change"。
而main方法体中,调用change方法时候,JVM新做了一个str对象的拷贝、发往change方法、作为入参,就像int类型参数一样(这一点大家容易理解吧);change函数体内将入参str的值改变(指向另一个对象),不会影响到main方法中的str对象的值,即它的指向。
这里,by value的意思,不知道大家真正理解了没有。
str的value是对"World"这个对象的引用。