去评论
就爱江湖 Www.92Jh.Cn

【笔记】Java基础面试题

q3196355
2019/08/22 11:05:32
本帖最后由 q3196355 于 2019-8-22 11:11 编辑

类型一:初始化
what:类内部初始化顺序,静态成员、非静态成员、构造器方法是以怎样的顺序完成初始化的?
how? 以下程序会有怎样的输出结果

public class Animal {    static {        System.out.println("静态代码块执行");    }    Other other  = new Other("非静态");    static Other staticOther = new Other("静态");    {        System.out.println("非静态代码块执行");    }    public void sayHi() {        System.out.println("你好,我是 Animal 类");    }    public Animal() {        System.out.println("兽类被初始化");    }    public static void main(String[] args) {        Animal animal = new Animal();    }}

输出:
静态代码块执行静态Other 类初始化非静态Other 类初始化非静态代码块执行兽类被初始化

在同一类中,先初始静态成员(包括静态代码块)——> 非静态成员 ——> 执行构造器内部操作。同一级别中,按照代码先后顺序执行。
why?
what:继承关系,子类的初始化和父类的初始化有关吗?
how? 以下程序会有怎样的输出结果

public class Animal {    public Animal() {        System.out.println("兽类被初始化");    }    public static void main(String[] args) {        Golden golden = new Golden();    }}class Dog extends Animal {    public Dog() {        System.out.println("Dog 类被初始化");    }}class Golden extends Dog{    public Golden(){        System.out.println("Golden 类被初始化");    }}

执行结果:
兽类被初始化Dog 类被初始化Golden 类被初始化

在调用子类构造器创建子类实例时,父类无参构造器会先执行。
why?
子类继承了父类所有成员和方法(包括 private 成员),在子类构造器中可以通过 super 显式调用父类的成员和方法。为确保子类在使用时,父类成员可用,要先调用父类构造器,完成父类初始化。
扩展
在子类构造器中会隐式调用父类无参构造器,这是我们未意识到父类构造器的原因(编译器简化我们的代码)。所以,当父类不存在无参构造器时,我们需要显式调用父类构造器,且必须为子类构造器第一条执行语句。


class Dog extends Animal {    String name;    public Dog(String name) {        this.name = name;        System.out.println("Dog 类被初始化");    }}class Golden extends Dog{    public Golden(){        super("金毛");        System.out.println("Golden 类被初始化");    }}

类型二:封装与多态的结合
what: 实例化父类对象却调用子类的构造函数,那执行方法时,调用的是父类方法,还是子类方法,那方法中所调用的成员呢?
how:
class Test1 {    private int id = 12;    public void printId() {        System.out.println(IDAdd());    }    public int idAdd(){        return id+1;    }}public class Test2 extends Test1 {    private int id = 1;    public static void main(String[] args) {        Test1 test = new Test2();        test.printId();    }}
输出:13
why?
Java 中,调用谁的构造方法,就访问谁的作用域。但是,父类的方法只能访问父类成员。当 子类重写父类方法时,子类作用域调用的就是子类的方法。
修改 Test2 代码
public class Test2 extends Test1 {    private int id = 1;    public int idAdd(){        return id+1;    }    public static void main(String[] args) {        Test1 test = new Test2();        test.printId();    }}
此时的输出的答案就是 2。
此时父类 public void printId()  调用的是子类重写的 idAdd(),而子类的 idAdd() 访问的 id 是其本身的成员变量。由于子类方法的重写,二者的答案截然不同,这就是「多态」的一种体现。
类型三:值传递还是引用传递?
what:参数传递,会改变原有变量的值或引用吗?
how? 以下程序会有怎样的输出结果
public void update(String str,char[] chars){        str = "新字符串";        chars[0] = '0';    }    public static void main(String[] args) {        Test test = new Test();        String str = "字符串";        char[] chars = {'1','2','3'};        test.update(str, chars);        System.out.println("str:" + str);        System.out.println("chars:" + Arrays.toString(chars));    }
输出结果:
str:字符串chars:[0, 2, 3]
String 的引用未改变,但是 chars[0] 的值被改变
why?
Java 中所有的参数都是「值传递」,简单说就是「拷贝」(复制)。对于基本类型,就是复制其值,引用类型则复制其引用。
方法中的 str = "新字符串"; 实际上是将传入的引用地址改变,但是原有的 str 引用是没有发生改变的,而 chars 虽然也是复制引用,但是通过下标访问到到的是 chars[0] 实际内存地址,实际修改了它的值。
为了方便理解我将代码改造,并做了标注,方便理解。

类似于成语「刻舟求剑」所讲述的故事,虽然在舟上做了丢剑处的标记,但是舟下的那片江却不是丢剑的江了。
三道典型的 Java 基础面试讲完了,没准你有什么不一样的看法或者技巧,欢迎你给出评论,或是默默点点赞给个好看,这都是对我的一种认可。