Java中的构造方法总结

java 专栏收录该内容
18 篇文章 1 订阅
构造方法和实例方法的区别

一、主要的区别在于三个方面:修饰符、返回值、命名

1、和实例方法一样,构造器可以有任何访问的修饰符,public、private、protected或者没有修饰符   ,都可以对构造方法进行修饰。不同于实例方法的是构造方法不能有任何非访问性质的修饰符修饰,例如static、final、synchronized、abstract等都不能修饰构造方法
解释:构造方法用于初始化一个实例对象,所以static修饰是没有任何意义的;多个线程不会同时创建内存地址相同的同一个对象,所以synchronized修饰没有意义;
构造方法不能被子类继承,所以final和abstract修饰没有意义。
2、返回类型是非常重要的,实例方法可以返回任何类型的值或者是无返回值(void),而构造方法是没有返回类型的,void也不行。
3、至于命名就是构造方法与类名相同,当然了实例方法也可以与类名相同,但是习惯上我们为实例方法命名的时候通常是小写的,另一方面也是与构造方法区分开。
而构造方法与类名相同,所以首字母一般大写。
下面看几个例子熟悉一下:
  1. public class Sample {  
  2.       
  3.     private int x;  
  4.   
  5.     public Sample() { // 不带参数的构造方法  
  6.         this(1);  
  7.     }  
  8.       
  9.     public Sample(int x) { //带参数的构造方法  
  10.         this.x=x;  
  11.     }  
  12.       
  13.     public int Sample(int x) { //不是构造方法  
  14.         return x++;  
  15.     }  
  16.   
  17. }  
上面的例子即使不通过注释我们也很容易能区分开的,再看下面一个例子
  1. public class Mystery {  
  2.     private String s;  
  3.       
  4.     public void Mystery() { //不是构造方法  
  5.     s = "constructor";  
  6.     }  
  7.       
  8.     void go() {  
  9.     System.out.println(s);  
  10.     }  
  11.       
  12.     public static void main(String[] args) {  
  13.     Mystery m = new Mystery();  
  14.     m.go();  
  15.     }  
  16. }  
程序执行的结果为null,虽然说Mystery m = new Mystery();调用了 Mystery 类的构造方法,但是public void Mystery()并不是构造方法,他只是一个普通的实例方法而已,那该类的构造方法哪去了呢?

二、说到这就得说一下java的默认构造方法
我们知道,java语言中规定每个类至少要有一个构造方法,为了保证这一点, 当用户没有给java类定义明确的构造方法的时候,java为我们提供了一个默认的构造方法,这个构造方法没有参数,修饰符是public并且方法体为空。如果用户有定义构造方法,就不会有默认构造方法!!!
其实默认的构造方法还分为两种,一种就是刚刚说过的隐藏的构造方法,另一种就是显示定义的默认构造方法.
如果一个类中定义了一个或者多个构造方法,并且每一个构造方法都是带有参数形式的,那么这个类就没有默认的构造方法,看下面的例子。
  1. public class Sample1{}  
  2.   
  3. public class Sample2{  
  4.     public Sample2(int a){System.out.println("My Constructor");}  
  5. }  
  6.   
  7. public class Sample3{  
  8.     public Sample3(){System.out.println("My Default Constructor");}  
  9. }  
上面的三个类中Sample1有一个隐式的默认构造方法,下列语句Sample1 s1=new Sample()合法;
Sample2没有默认的构造方法,下列语句Sample2 s2=new Sample2()不合法,执行会编译错误
Sample3有一个显示的默认构造方法,所以以下语句Sample3  s3=new Sample3();合法。

三、实例方法和构造方法中this、super的使用.
"this"的用法
实例方法中可以使用this关键字,它指向正在执行方法的类的实例对象,当然static方法中是不可以使用this对象的,因为静态方法不属于类的实例对象;而构造方法中同样可以使用this关键字,构造器中的this是指向同一个对象中不同参数的另一个构造器。让我们来看下面的一段代码:
  1. public class Platypus {  
  2.     String name;  
  3.   
  4.     Platypus(String input) {  
  5.         name = input;  
  6.     }  
  7.   
  8.     Platypus() {  
  9.         this("John/Mary Doe");  
  10.     }  
  11.   
  12.     public static void main(String args[]) {  
  13.         Platypus p1 = new Platypus("digger");  
  14.         Platypus p2 = new Platypus();  
  15.         System.out.println(p1.name + "----" + p2.name);  
  16.     }  
  17. }  
上面的代码中 类有两个构造器,第一个构造器给类的成员name赋值,第二个构造器调用第一个构造器给类的成员name一个初始值Jonn/Mary Doe
所以程序执行结果:digger----John/Mary Doe
需要注意的两个地方是:
1、构造方法中通过this关键字调用其他构造方法时,那么这句代码必须放在第一行,否则会编译错误。
2、构造方法中只能通过this调用一次其他的构造方法。

"super"的用法:
实例方法和构造方法中的super关键字都用于去指向父类,实例方法中的super关键字是去调用父类当中的某个方法,看下面的代码:
  1. class getBirthInfo {  
  2.     void getBirthInfo() {  
  3.         System.out.println("born alive.");  
  4.     }  
  5. }  
  6.   
  7. class Platypus1 extends getBirthInfo  
  8. {  
  9.     void getBirthInfo() {  
  10.            System.out.println("hatch from eggs");  
  11.            System.out.println("a mammal normally is ");  
  12.            super.getBirthInfo();  
  13.       }  
  14. }  
  15.   
  16. public class test1 {  
  17.     public static void main(String[] args) {  
  18.         Platypus1 p1=new Platypus1();  
  19.         p1.getBirthInfo();  
  20.     }  
  21. }  
上面的例子使用super.getBirthInfo();调用了它的父类的void getBirthInfo()方法。
构造器中使用super关键字调用父类中的构造器,看下面的代码:
  1. class getBirthInfo {  
  2.     getBirthInfo(){  
  3.         System.out.println("auto");  
  4.     }  
  5.     void aa() {  
  6.         System.out.println("born alive.");  
  7.     }  
  8. }  
  9.   
  10. class Platypus1 extends getBirthInfo  
  11. {  
  12.       Platypus1() {  
  13.         super();  
  14.         System.out.println("hatch from eggs");  
  15.         System.out.println("a mammal normally is ");  
  16.       }  
  17. }  
  18.   
  19. public class test1 {  
  20.     public static void main(String[] args) {  
  21.         Platypus1 p1=new Platypus1();  
  22.     }  
  23. }  
执行了代码我们就会看到构造器中的super调用了父类的构造方法。

类的继承机制使得子类可以调用父类的功能,下面介绍类在继承关系的初始化顺序问题

请看实例1:
  1. class SuperClass   
  2. {   
  3.     SuperClass()   
  4.     {   
  5.         System.out.println("SuperClass constructor");   
  6.     }   
  7. }   
  8. public class SubClass extends SuperClass {  
  9.     SubClass()   
  10.     {   
  11.         System.out.println("SubClass constructor");   
  12.     }   
  13.     public static void main(String[] args) {  
  14.         SubClass sub = new SubClass();   
  15.     }   
  16. }   
执行结果:SuperClass constructor
     SubClass constructor
代码中我们只实例化子类一个对象,但从执行结果上看程序一开始并不是运行子类的构造方法,而是先执行父类的默认构造方法,然后再执行子类的构造方法.所以我们在实例化子类对象时,程序会先调用父类的默认构造方法,然后再执行子类的构造方法。

再看实例2:
  1. class SuperClass   
  2. {   
  3.     SuperClass(String str)   
  4.     {   
  5.     System.out.println("Super with a string.");   
  6.     }   
  7. }   
  8. public class SubClass extends SuperClass   
  9. {   
  10.     SubClass(String str)   
  11.     {   
  12.     System.out.println("Sub with a string.");   
  13.     }   
  14.       
  15.     public static void main(String[] args)   
  16.     {   
  17.     SubClass sub = new SubClass("sub");   
  18.     }   
  19. }   
注意:此程序在JDK下不能编译成功,因为我们在实例化子类对象的时候会先调用其父类默认的构造方法(除非实例化对象的构造函数中有调用任意的父类构造方法),但是它的父类没有默认的构造方法,所以不能编译成功。易错地方!!
解决办法:
1、在父类中加一个显示的默认构造方法
2、在子类的构造方法中加一句super(str)并且必须在构造器的第一句。
两个办法都可以解决程序编译的问题,但是执行结果是不一样的.
第一种执行结果为:Sub with a string.
第二种执行结果为:Super with a string. 
                       Sub with a string.     第二种方法即使父类中有显示的默认构造方法也不会被调用。

再看实例三:
  1. class One   
  2. {   
  3.     One(String str)   
  4.     {   
  5.     System.out.println(str);   
  6.     }   
  7. }   
  8. class Two   
  9. {   
  10.     One one_1 = new One("one-1");   
  11.     One one_2 = new One("one-2");   
  12.     One one_3 = new One("one-3");   
  13.     Two(String str)   
  14.     {   
  15.     System.out.println(str);   
  16.     }   
  17. }   
  18. public class Test   
  19. {   
  20.     public static void main(String[] args)   
  21.     {   
  22.     System.out.println("Test main() start");   
  23.     Two two = new Two("two");   
  24.     }   
  25. }  
执行结果:
Test main() start
one-1
one-2
one-3
two
我们在main方法中实例了一个Two的对象,但是程序在实例Two对象时并没有先调用Two的构造方法,而是先初始化Two类的成员变量,Two类中有三个成员变量,他们都是One类的对象,所以要依次执行One类的构造方法,然后再初始化Two类的对象。
在实例化类的对象时,类中的成员变量会首先进行初始化,如果其中的成员变量有对象,那么它们也会按照顺序执行初始化 工作 。在所有类成员初始化完成后,才调用对象所在类的构造方法创建对象。构造方法作用就是初始化。 

再看实例四:
  1. class One   
  2.     {   
  3.     One(String str)   
  4.     {   
  5.     System.out.println(str);   
  6.     }   
  7. }   
  8. class Two   
  9. {   
  10.     One one_1 = new One("one-1");   
  11.     One one_2 = new One("one-2");   
  12.     static One one_3 = new One("one-3");   
  13.     Two(String str)   
  14.     {   
  15.     System.out.println(str);   
  16.     }   
  17. }   
  18. public class Test   
  19. {   
  20.     public static void main(String[] args)   
  21.     {   
  22.     System.out.println("Test main() start");   
  23.     Two two_1 = new Two("two-1");   
  24.     System.out.println("------------");   
  25.     Two two_2 = new Two("two-2");   
  26.     }   
  27. }   
执行结果:
Test main() start
one-3
one-1
one-2
two-1
------------
one-1
one-2
two-2
结论:如果一个类中有静态对象,那么他会在非静态对象初始化前进行初始化,但只初始化一次。而非静态对象每次调用时都要初始化。

再看实例五:
  1. class One   
  2. {   
  3.     One(String str)   
  4.     {   
  5.     System.out.println(str);   
  6.     }   
  7. }   
  8. class Two   
  9. {   
  10.     One one_1 = new One("one-1");   
  11.     One one_2 = new One("one-2");   
  12.     static One one_3 = new One("one-3");   
  13.     Two(String str)   
  14.     {   
  15.     System.out.println(str);   
  16.     }   
  17. }   
  18. public class Test   
  19. {   
  20.     static Two two_3 = new Two("two-3");   
  21.     public static void main(String[] args)   
  22.     {   
  23.     System.out.println("Test main() start");   
  24.     Two two_1 = new Two("two-1");   
  25.     System.out.println("------------");   
  26.     Two two_2 = new Two("two-2");   
  27.     }   
  28. }  
执行结果:
one-3
one-1
one-2
two-3
Test main() start
one-1
one-2
two-1
------------
one-1
one-2
two-2
结论:程序中主类的静态变量会在main()方法执行前初始化。结果中只输出了一次one-3,这也说明:如果一个类中有静态对象,那么它会在非静态对象前初始化,但只初始化一次。非静态对象每次调用时都要初始化。 


再看实例六:

class One {
	One(String str) {
		System.out.println(str+" in Class One Constror ");
	}
}

class Two {
	One one_1 = new One("Class Two Field one-1");
	One one_2 = new One("Class Two Field one-2");
	static One one_3 = new One("Class Two Static Field one-3");
	public Two() {
		System.out.println("Class Two Defalut Constror ");
	}
	Two(String str) {
		System.out.println(str+" in Class Two Constror +");
	}
}

class TwoSub extends Two{
	One one_4 = new One("Class TwoSub Field one-4");
	static One one_5 = new One("Class TwoSub Static Field one-5");
	public TwoSub(String str) {
		System.out.println(str+" in Class TwoSub Constror ");
	}
}

public class Test {
//	static Two two_3 = new Two("two-3");

	public static void main(String[] args) {
		System.out.println("Test main() start");
		TwoSub twoSub = new TwoSub("twoSub");
//		Two two_1 = new Two("two-1");
//		System.out.println("------------");
//		Two two_2 = new Two("two-2");
	}
}
执行结果:
Test main() start
Class Two Static Field one-3 in Class One Constror 
Class TwoSub Static Field one-5 in Class One Constror 
Class Two Field one-1 in Class One Constror 
Class Two Field one-2 in Class One Constror 
Class Two Defalut Constror 
Class TwoSub Field one-4 in Class One Constror 
twoSub in Class TwoSub Constror 

由此总结初始化顺序:
  1.父类的静态成员
    2.子类的静态成员
    3.父类的非静态成员
  4.父类的默认构造函数被调用。 
  5.子类的非静态对象(变量) 
  6.子类的构造函数。 

©️2021 CSDN 皮肤主题: 大白 设计师:CSDN官方博客 返回首页
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值