从一个小程序看Java的对象构造过程

本文通过一个Java继承的例子,展示了在子类构造过程中,父类构造器如何调用子类已覆盖的方法,导致输出结果与预期不符的情况。解释了类初始化时构造函数的调用顺序及变量的初始化过程。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

 今天在百度知道上看见某人问的一个问题,改装了一下,给人的感觉会更加诡异。下面来看看这个小程序:

class TestA {
    
int x=1;
    
    
public TestA() {
        a1();
    }

    
public void a1() {
        System.out.println(
"x in TestA is " + x);
    }
}

public class TestB extends TestA {
    
int x = 10;

    
public TestB() {
        x 
= 1000;
    }

    
public void a1() {
        System.out.println(
"x in TestB is " + x);
    }

    
public static void main(String[] args) {
        
new TestB(); 
    }
}

现在来猜猜运行结果会是什么?TestA还是TestB?x是1,10还是100?

如果你对上面几个数字念念不忘,那么恭喜你,你提前知道了自己错了。正确的答案是:

x in testB is 0

下面来说说原因:

类初始化时构造函数调用顺序:
(1)初始化对象的存储空间为零或null值;
(2)调用父类构造函数;
(3)按顺序分别调用类成员变量和实例成员变量的初始化表达式;
(4)调用本身构造函数。

首先new TestB()实例化一个TestB的对象

(1)先给自己分配了空间,并赋初始值;

(2)由于TestB继承至TestA,这里使用的是默认的没有参数的构造函数,它会先调用父类的构造函数,父类的构造函数中调用了a1()方法,但我们实例化的是一个子类TestB的对象,在子类中已经覆盖了a1()方法,因此调用的是子类的a1()方法,这时候步骤(3)还并没有被执行,因此x还是初始值,int的默认初始值是0;

(3)执行x=10;

(4)执行TestB中的构造函数中的x=100;

我们的程序是在第(2)步中打印结果的,所以结果会是上面那样。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值