一、重载
Java允许重载任何方法,而不只是构造器方法。因此,要完整地描述一个方法,许哟啊指定方法名以及参数类型,也就是方法的签名(signature)。
例如:String类有4个名为indexOf的公共方法,它们的签名是
indexOf(int)
indexOf(int, int)
indexOf(String)
indexOf(String, int)
返回类型不是方法签名的一部分,也就是说:
二、无参构造
如果写一个类时没有编写构造器,就会为你提供一个无参数构造器,这个构造器将所有的实例字段设置为默认值(0、false、null)。
如果在类中提供了至少一个构造器,但是没有提供无参构造器,那么使用无参构造器构造对象就是不合法的:
会报错:
三、this关键字
this指示隐式参数,也就是所构造的对象。下面是一个实例:
// this给字段赋值
public A(String name, int id) {
this.name = name;
this.id = id;
}
使用this关键字还可以调用另一个构造器:当调用new A(“张三”, 1, 100)时,A(int)构造器将调用A(String, int)构造器,采用这种方式使用this关键字非常有用,这样对公共的构造器代码只需要编写一次即可。
// this调用另一个构造器
public A(int money) {
// calls A(String, int)
this(A + money);
}
四、默认字段初始化
如果在构造器中没有显式地为字段设置初值,字段就会被自动的赋默认值:
然而方法中的局部变量必须明确的初始化。这是字段与局部变量的一个重要区别。
五、继承的加载顺序
由于static块会在首次加载类的时候执行,因此下面的例子就是用static块来测试类的加载顺序。
所有的变量初始化完,才会执行初始化块、构造方法。
在类的加载过程中,只有内部的变量创建完,才会去执行这个类的构造方法。
在类的加载过程中,静态成员类的对象,会优先加载;而普通成员类的对象则是使用的时候才回去加载。
// this调用另一个构造器
package com.example.demo.test;
class Father {
public Father() { // 父类无参构造函数
System.out.println("Father init block");
}
{ // 父类初始化块
System.out.println("I'm Father class");
}
static { // 父类静态代码块
System.out.println("static Father");
}
}
public class Son extends Father {
public Son() { // ...
System.out.println("Son init block");
}
{ // ...
System.out.println("I'm Son class");
}
static { // ...
System.out.println("static Son");
}
public static void main(String[] args) {
new Son();
}
}
输出:
static Father // 执行父类静态代码
static Son // 执行子类静态代码块
I'm Father class // 执行父类初始化块
Father init block // 执行父类无参构造函数
I'm Son class // ..
Son init block // ..
六、Java new一个对象过程中发生了什么
1.确认类元信息是否存在。
当 JVM 接收到 new 指令时,
首先在 metaspace 内检查需要创建的类元信息是否存在。
若不存在,那么在双亲委派模式下,使用当前类加载器以 ClassLoader + 包名+类名为 Key 进行查找对应的 class 文件。 如果没有找到文件,则抛出 ClassNotFoundException 异常 , 如果找到,则进行类加载(加载 - 验证 - 准备 - 解析 - 初始化),并生成对应的 Class 类对象。
2.分配对象内存。
首先计算对象占用空间大小,如果实例成员变量是引用变量,仅分配引用变量空间即可,即 4 个字节大小,接着在堆中划分—块内存给新对象。 在分配内存空间时,需要进行同步操作,比如采用 CAS (Compare And Swap) 失败重试、 区域加锁等方式保证分配操作的原子性。
3.设定默认值。
成员变量值都需要设定为默认值, 即各种不同形式的零值。
4.设置对象头。
设置新对象的哈希码、 GC 信息、锁信息、对象所属的类元信息等。这个过程的具体设置方式取决于 JVM 实现。
5.执行 init 方法。
初始化成员变量,执行实例化代码块,调用类的构造方法,并把堆内对象的首地址赋值给引用变量。