手记

JAVA类属性初始化的有趣儿情况

2017-05-12 01:07:087956浏览

Qiujuer

2实战 · 12手记 · 17推荐

这是我之前遇到的一个问题,在这里分享给同僚;希望大家能避免这些错误;少走弯路。

问题

这个情况有些特殊,当时我在弄APP的设置模块;结果发现无论设置如何保存第二次获取值的时候都是初始值,跟踪了一下发现保存的时候的确是保存到文件了,那么这就奇葩了。想了一下问题肯定出在获取值的时候,跟踪了一下获取值的断点,发现获取的时候是之前保存的值,而在类的构造函数之后值却又成了初始值,这个是为啥?

百思不得其姐啊~~~

我喜欢直接一点,来看看代码是怎么回事儿。

代码
package net.qiujuer;

public abstract class People {
    public People() {
        initValue();
    }

    public abstract void initValue();

    public void show() {
        System.out.printf("Show:" + toString());
    }
}

一个父类,在构造函数中调用初始化方法。

package net.qiujuer;

public class Man extends People {
    private String mName = "聂小倩";
    private int mAge = 18;

    @Override
    public void initValue() {
        mName = "Qiujuer";
        mAge = 23;
    }

    @Override
    public String toString() {
        return "Name:" + (mName == null ? "NULL" : mName) + " Age:" + mAge + "\n";
    }
}

实现类,实现其具体的初始化方法,并在该方法中做一些属性操作。

package net.qiujuer;

public class MainClass {
    public static void main(String[] args) {
        Man man = new Man();
        man.show();
    }
}

测试类,在该类中直接 new 并调用方法。
结果是什么呢?

有谁能猜到?

其输出为:

Show:Name:聂小倩 Age:18

怎么样?有想到的么?为什么会这样呢?自己试试看看是不是。

变更

我们输出一些详细的地方。

首先更改父类的构造函数:

    public People() {
        System.out.printf("People-1:" + toString());
        initValue();
        System.out.printf("People-2:" + toString());
    }

然后更改 Man 的构造函数:

    public Man() {
        super();
        System.out.printf("Man:" + toString());
    }

再来运行看看~~

想想一下结果是什么呢?

先想一下,然后再看估计有些用处。

算了,还是直接进入吧~

结果输出:

People-1:Name:NULL Age:0
People-2:Name:Qiujuer Age:23
Man:Name:聂小倩 Age:18
Show:Name:聂小倩 Age:18

想到了么?

结论

我想到了这些:

  • 在类属性初始化的时候,其直接赋值的默认值将会在父类的构造函数之后进行。
  • 在父类操作的时候,其子类的属性并不具备值:mName 属性为 NULL
  • 在父类中可以通过方法对实现类的属性进行赋值操作,但是当其实现类中默认有对值进行赋值的话,那么将会自动进行二次赋值。

一个无心之举,或许会造成一个天大的BUG;且用且珍惜~

34人推荐
随时随地看视频
慕课网APP

热门评论

先父类静态代码块,静态变量,出现顺序执行。然后子类静态代码块,静态变量。然后父类代码块,变量,构造方法,再然后是子类代码块,变量,构造方法

先父类静态代码块,静态变量,出现顺序执行。然后子类静态代码块,静态变量。然后父类代码块,变量,构造方法,再然后是子类代码块,变量,构造方法

man的构造函数没有super(),所以不会initvalue()

查看全部评论