今天一个同学问了这样的问题,现有如下代码:
class Student {
private int age;
public Student(int v) {
this.age = v;
}
public Student() {
this.age = 20;
}
public void showAge() {
System.out.println(this.age);
}
}
public class Test {
public static void main(String[] args) {
Student s1 = new Student(1);
Student s2 = new Student(2);
Student le = new Student();
le = s1;
s1 = s2;
le.showAge();
}
}
请问:为什么输出的是1,而不是2?
我的理解是:当我们在说le、s1、s2的时候,他们都有两层含义,即变量和对象,我们可以说le是一个变量,那么,这个变量存的值是le作为一个对象在内存中的地址。变量和对象需要区别对待,对于变量而言,我们关心的是它里面存的值,对于对象而言,我们关心的是它的地址,当解释器在执行le = s1;这条语句时,是将变量s1中存的值(即对象s1的地址)放入变量le中,同理,执行s1 = s2;语句时,也是将变量s2中存的值(即对象s2的地址)放入变量s1中,执行完这两条语句后,变量le中存放的是对象s1的地址,而对象s1中的age变量在整个程序执行的过程中没有被改变,依然是初始化的值,即1,所以最后打印的结果是1。
执行完下面三条语句后,变量与对象的关系如图1所示:
Student s1 = new Student(1);
Student s2 = new Student(2);
Student le = new Student();
图1
执行完下面两条语句后,变量与对象的关系如图2所示:
le = s1;
s1 = s2;
图2
由上图可见,程序在执行le.showAge()时变量le对应的对象是new Student(1),因此输出的结果是1。
其实,这在C++中,是可以通过引用让le与s1对应同一对象的,代码如下所示:
#include <iostream>
using namespace std;
class Student {
private:
int age;
public:
Student(int v);
Student();
void showAge();
};
Student::Student(int v) {
this->age = v;
}
Student::Student() {
this->age = 20;
}
void Student::showAge() {
cout << this->age << endl;
}
int main(void) {
Student s1 = Student(1);
Student s2 = Student(2);
Student &le = s1;
s1 = s2;
le.showAge();
}
该程序的输出结果是2。