Java构造器就是这么简单!

前言

理解构造器之前,首先我们需要了解Java中为什么要引入构造器,以及构造器的作用。

在很久之前,程序员们编写C程序总会忘记初始化变量(这真的是一件琐碎但必须的事),C++引入了 构造器(constructor) 的概念,这是一个在创建对象时被自动调用的特殊方法。Java也采用了构造器。

一、构造器的引入

引入构造器帮助我们解决了哪些问题呢?假设我们每定义一个类都必须定义一个initialize()方法,该方法提醒你,每次使用对象之前都要执行一次该方法,这意味着用户每次都必须记得自己去调用此方法,这和上文提到的C程序员一样,很容易就忘记了。Java构造器的出现很好的规避掉了这种问题,创建对象时,java会在使用对象之前调用相应的构造器,保证对象正确初始化。

我们来看一个简单实例:

 public class TestMain {
 ​
     TestMain() {    //默认构造器
         System.out.println("默认构造器");
     }
 ​
     public static void main(String[] args) {
         new TestMain();
     }
 }
 ​
 //输出
 默认构造器

 

从这个例子我们看到了,构造器为 TestMain() ,创建对象时,会分配内存并调用对应的构造方法,可以看到输出结果为 默认构造器 ,它已经被正确地初始化了。

二、构造器命名规则

从上面那个例子中或许已经观察到了:类名和构造器名必须相同,所以”每个方法首字母小写“的编码风格并不适用于构造器。

三、注意事项

  • 构造器必须与主类同名

  • 构造器可以有参数

  • 构造器可以重载

  • 没有返回值

  • 不添加构造器编译器生成默认构造器

四、默认构造器

默认构造器(又名无参构造器)是没有形式参数的,它创建的是”默认对象“。举个栗子:

 public class TestMain {
     
     //没有指定构造器,Java编译器会自动生成默认构造
public static void main(String[] args) {
         new TestMain();
     }
 }
 ​
 //输出

 

new TestMain()创建了一个新对象,并调用了默认构造——虽然我们并没有主动定义它。Java规定了,如果没有构造会生成默认构造,如果存在了一个及以上的构造便不会自动生成。

 public class TestMain {
     
     TestMain(int i) {}
     TestMain(float f) {}
 ​
     public static void main(String[] args) {
         TestMain t1 = new TestMain();    //会报错,没有对应的构造方法
         TestMain t2 = new TestMain(1);
         TestMain t3 = new TestMain(2.0f);
     }
 }
 ​
 //输出

 

new TestMain()编译器会报错,因为我们没有定义对应的无参构造方法,编译器无法顺利创建对象。如果你没有定义构造器,编译器会认为”你需要一个构造器,我帮你造一个“;如果你自己写了一个构造器,编译器会认为”你已经有构造器了,你知道自己在做什么,我不帮你生成“。

五、构造方法重载

有默认无参构造,就有带参构造;有带参构造也就会发生方法重载。为了满足不同的初始化需求,我们通常会需要定义多个带参构造器,由于都是构造器,它们的名称必须相同,为了让方法名相同而参数不同的方法存在,我们就必须使用 方法重载 。它是构造器所必须的。

 public class TestMain {
 ​
     TestMain() {
         System.out.println("默认构造");
     }
 ​
     TestMain(int i) {
         System.out.println("int带参构造");
     }
 ​
     TestMain(float f) {
         System.out.println("float带参构造");
     }
 ​
     public static void main(String[] args) {
         TestMain t1 = new TestMain();
         TestMain t2 = new TestMain(1);
         TestMain t3 = new TestMain(2.0f);
     }
 }
 ​
 //输出
 默认构造
 int带参构造
 float带参构造

 

从上述代码中我们可以看到,类中定义了三个不同的构造方法,main方法中,在括号里传递不同的参数,编译器会根据参数的类型寻找对应的构造方法,从而初始化三个不同的对象,这就是构造方法的重载。

涉及基本类型的重载

在使用构造方法的重载时,我们经常会遇到将基本类型传递给重载方法时的一些问题。基本类型可以从一个 较小(窄类型) 类型自动提升(转型)为一个 较大(宽类型) 类型,当涉及到方法重载时便会造成一些混淆。举个栗子:

 public class TestMain {
 ​
     TestMain(int i) {
         System.out.println("int带参构造");
     }
 ​
     TestMain(long l) {
         System.out.println("long带参构造");
     }
 ​
     TestMain(double d) {
         System.out.println("double带参构造");
     }
 ​
     public static void main(String[] args) {
         TestMain t1 = new TestMain(2.0f);
         TestMain t2 = new TestMain('菌');
     }
 }
 ​
 //输出
 double带参构造
 int带参构造

 

首先,我们来看一看t1对象,创建对象时传递的参数是一个float类型的数据,但是结果却显示调用了double带参构造,这是咋回事?其实在创建对象时,编译器会根据传递参数的类型自动寻找参数类型对应的构造方法,如果没有一模一样的构造方法,就会寻找类型更“宽”的构造方法。t1就是典型的例子,double类型比float更“宽”(float占4字节,double占8字节),所以会把传递的参数提升。

再来看t2对象,这个对象传递的是char类型数据,对于char类型略有不同,如果无法找到恰好接受char参数的方法,就会把char直接提升为int

更多学习内容请阅读我的知乎专栏:打造全网Java高级工程师资料库(总目录)看完学的更加快,知识更牢固。你值得拥有(持续更新)~

你可能感兴趣的:(Java构造器就是这么简单!)