转自:https://blog.csdn.net/qq_41620160/article/details/79443524

使用构造器时需要记住:

1.构造器必须与类同名(如果一个源文件中有多个类,那么构造器必须与公共类同名)

2.每个类可以有一个以上的构造器

3.构造器可以有0个、1个或1个以上的参数

4.构造器没有返回值

5.构造器总是伴随着new操作一起调用

6.不添加任何构造器会默认有空的构造器

继承与构造器

使用super调用父类构造器的语句必须是子类构造器的第一条语句

为什么加上super?

Java的构造器并不是函数,所以他并不能被继承,这在我们extends的时候写子类的构造器时比较的常见,即使子类构造器参数和父类的完全一样,我们也要写super就是因为这个原因。

如果子类构造器没有显式地调用父类的构造器,则将自动调用父类的默认(没有参数)的构造器。如果父类没有不带参数的构造器,并且在子类的构造器中又没有显式地调用父类的构造器,则java编译器将报告错误

构造方法和实例方法的区别:

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

1、和实例方法一样,构造器可以有任何访问的修饰符,public、private、protected或者没有修饰符   ,都可以对构造方法进行修饰。不同于实例方法的是构造方法不能有任何非访问性质的修饰符修饰,例如static、final、synchronized、abstract等都不能修饰构造方法。

解释:构造方法用于初始化一个实例对象,所以static修饰是没有任何意义的;多个线程不会同时创建内存地址相同的同一个对象,所以synchronized修饰没有意义;

构造方法不能被子类继承,所以final和abstract修饰没有意义。

2、返回类型是非常重要的,实例方法可以返回任何类型的值或者是无返回值(void),而构造方法是没有返回类型的,void也不行。

3、至于命名就是构造方法与类名相同,当然了实例方法也可以与类名相同,但是习惯上我们为实例方法命名的时候通常是小写的,另一方面也是与构造方法区分开。

而构造方法与类名相同,所以首字母一般大写。

下面看几个例子熟悉一下:

public class Sample {  private int x;  public Sample() { // 不带参数的构造方法  this(1);  }  public Sample(int x) { //带参数的构造方法  this.x=x;  }  public int Sample(int x) { //不是构造方法  return x++;  }  }  

上面的例子即使不通过注释我们也很容易能区分开的,再看下面一个例子

public class Mystery {  private String s;  public void Mystery() { //不是构造方法  s = "constructor";  }  void go() {  System.out.println(s);  }  public static void main(String[] args) {  Mystery m = new Mystery();  m.go();  }  }

程序执行的结果为null,虽然说Mystery m = new Mystery();调用了Mystery 类的构造方法,但是public void Mystery()并不是构造方法,他只是一个普通的实例方法而已,那该类的构造方法哪去了呢?

二、说到这就得说一下java的默认构造方法

我们知道,java语言中规定每个类至少要有一个构造方法,为了保证这一点,当用户没有给java类定义明确的构造方法的时候,java为我们提供了一个默认的构造方法,这个构造方法没有参数,修饰符是public并且方法体为空。

其实默认的构造方法还分为两种,一种就是刚刚说过的隐藏的构造方法,另一种就是显示定义的默认构造方法.

如果一个类中定义了一个或者多个构造方法,并且每一个构造方法都是带有参数形式的,那么这个类就没有默认的构造方法,看下面的例子。

public class Sample1{}  public class Sample2{  public Sample2(int a){System.out.println("My Constructor");}  }  public class Sample3{  public Sample3(){System.out.println("My Default Constructor");}  }

上面的三个类中Sample1有一个隐式的默认构造方法,下列语句Sample1 s1=new Sample()合法;

Sample2没有默认的构造方法,下列语句Sample2 s2=new Sample2()不合法,执行会编译错误

Sample3有一个显示的默认构造方法,所以以下语句Sample3  s3=new Sample3();合法。

三、实例方法和构造方法中this、super的使用.

"this"的用法

实例方法中可以使用this关键字,它指向正在执行方法的类的实例对象,当然static方法中是不可以使用this对象的,因为静态方法不属于类的实例对象;而构造方法中同样可以使用this关键字,构造器中的this是指向同一个对象中不同参数的另一个构造器。让我们来看下面的一段代码:

public class Platypus {  String name;  Platypus(String input) {  name = input;  }  Platypus() {  this("John/Mary Doe");  }  public static void main(String args[]) {  Platypus p1 = new Platypus("digger");  Platypus p2 = new Platypus();  System.out.println(p1.name + "----" + p2.name);  }  }

上面的代码中 类有两个构造器,第一个构造器给类的成员name赋值,第二个构造器调用第一个构造器给类的成员name一个初始值Jonn/Mary Doe

所以程序执行结果:digger----John/Mary Doe

需要注意的两个地方是:

1、构造方法中通过this关键字调用其他构造方法时,那么这句代码必须放在第一行,否则会编译错误。

2、构造方法中只能通过this调用一次其他的构造方法。

"super"的用法:

实例方法和构造方法中的super关键字都用于去指向父类,实例方法中的super关键字是去调用父类当中的某个方法,看下面的代码:

class getBirthInfo {  void getBirthInfo() {  System.out.println("born alive.");  }  }   class Platypus1 extends getBirthInfo  {  void getBirthInfo() {  System.out.println("hatch from eggs");  System.out.println("a mammal normally is ");  super.getBirthInfo();  }  }  public class test1 {  public static void main(String[] args) {  Platypus1 p1=new Platypus1();  p1.getBirthInfo();  }  }

上面的例子使用super.getBirthInfo();调用了它的父类的void getBirthInfo()方法。

构造器中使用super关键字调用父类中的构造器,看下面的代码:

class getBirthInfo {  getBirthInfo(){  System.out.println("auto");  }  void aa() {  System.out.println("born alive.");  }  }  class Platypus1 extends getBirthInfo{  Platypus1() {  super();  System.out.println("hatch from eggs");  System.out.println("a mammal normally is ");  }  }  public class test1 {  public static void main(String[] args) {  Platypus1 p1=new Platypus1();  }  }

执行了代码我们就会看到构造器中的super调用了父类的构造方法。

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

请看实例1:

class SuperClass{   SuperClass()   {   System.out.println("SuperClass constructor");   }   }   public class SubClass extends SuperClass {  SubClass()   {   System.out.println("SubClass constructor");   }   public static void main(String[] args) {  SubClass sub = new SubClass();   }   }

执行结果:SuperClass constructor

SubClass constructor

代码中我们只实例化子类一个对象,但从执行结果上看程序一开始并不是运行子类的构造方法,而是先执行父类的默认构造方法,然后再执行子类的构造方法.所以我们在实例化子类对象时,程序会先调用父类的默认构造方法,然后再执行子类的构造方法。

再看实例2:

class SuperClass{   SuperClass(String str)   {   System.out.println("Super with a string.");   }   }   public class SubClass extends SuperClass   {   SubClass(String str)   {   System.out.println("Sub with a string.");   }   public static void main(String[] args)   {   SubClass sub = new SubClass("sub");   }   }

此程序在JDK下不能编译成功,因为我们在实例化子类对象的时候会先调用其父类默认的构造方法,但是它的父类没有默认的构造方法,所以不能编译成功。

解决办法:

1、在父类中加一个显示的默认构造方法

2、在子类的构造方法中加一句super(str)并且必须在构造器的第一句。

两个办法都可以解决程序编译的问题,但是执行结果是不一样的.

第一种执行结果为:Sub with a string.

第二种执行结果为:Super with a string.

Sub with a string.     第二种方法即使父类中有显示的默认构造方法也不会被调用。

再看实例三:

class One{   One(String str)   {   System.out.println(str);   }   }   class Two{   One one_1 = new One("one-1");   One one_2 = new One("one-2");   One one_3 = new One("one-3");   Two(String str)   {   System.out.println(str);   }   }   public class Test{   public static void main(String[] args)   {   System.out.println("Test main() start");   Two two = new Two("two");   }   }

执行结果:

Test main() start

one-1

one-2

one-3

two

我们在main方法中实例了一个Two的对象,但是程序在实例Two对象时并没有先调用Two的构造方法,而是先初始化Two类的成员变量,Two类中有三个成员变量,他们都是One类的对象,所以要依次执行One类的构造方法,然后再初始化Two类的对象。

既在实例化类的对象时,类中的成员变量会首先进行初始化,如果其中的成员变量有对象,那么它们也会按照顺序执行初始化工作。在所有类成员初始化完成后,才调用对象所在类的构造方法创建对象。构造方法作用就是初始化。

再看实例四:

class One{   One(String str)   {   System.out.println(str);   }   }   class Two{   One one_1 = new One("one-1");   One one_2 = new One("one-2");   static One one_3 = new One("one-3");   Two(String str)   {   System.out.println(str);   }   }   public class Test{   public static void main(String[] args)   {   System.out.println("Test main() start");   Two two_1 = new Two("two-1");   System.out.println("------------");   Two two_2 = new Two("two-2");   }   }

执行结果:

Test main() start

one-3

one-1

one-2

two-1

------------

one-1

one-2

two-2

结论:如果一个类中有静态对象,那么他会在非静态对象初始化前进行初始化,但只初始化一次。而非静态对象每次调用时都要初始化。

再看实例五:

class One{   One(String str)   {   System.out.println(str);   }   }   class Two{   One one_1 = new One("one-1");   One one_2 = new One("one-2");   static One one_3 = new One("one-3");   Two(String str)   {   System.out.println(str);   }   }   public class Test{   static Two two_3 = new Two("two-3");   public static void main(String[] args)   {   System.out.println("Test main() start");   Two two_1 = new Two("two-1");   System.out.println("------------");   Two two_2 = new Two("two-2");   }   }

执行结果:

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,这也说明:如果一个类中有静态对象,那么它会在非静态对象前初始化,但只初始化一次。非静态对象每次调用时都要初始化。

总结初始化顺序:

  1.主类的静态成员首先初始化。

  2.主类的父类的构造方法被调用。

  3.主类的非静态对象(变量)初始化。

  4.调用主类的构造方法。

Java的Constructor(构造器)的理解相关推荐

  1. [JAVA基础] 了解构造器Constructor———你真的会用构造器吗?

    引言 相信大家对于java里的构造器应该都是有了解的,这次我们来了解一些构造器的不同使用方式,了解构造器的调用顺序,最后可以灵活的在各种情况下定义使用构造器,进一步优化我们的代码: 构造器简介 还是简 ...

  2. Java反射(自己的理解)

    动态语言:运行是代码可以根据某些条件改变自身结构,像js和php,python等,但是我们不像c++,是一门静态语言,可以准确的说我们是一门准动态语言,因为反射让我们具有动态性. 我来直接用我所理解的 ...

  3. 一文了解 Java 中的构造器

    摘要:Java 也采用了构造器,并且还提供了一个垃圾收集器(garbage collector),当不再使用内存资源的时候,垃圾收集器会自动将其释放. 本文分享自华为云社区<一文带你了解 Jav ...

  4. 零基础Java学习之构造器

    构造器(Constructor) 构造器又称为构造方法,那是因为它长的很像方法.但是和方法还有有所区别的. 构造器的作用 ​ 要创建一个类的实例对象,必须调用一个对象的构造器,来完成类的实例初始化过程 ...

  5. Java反射 Constructor类

    Java反射 Constructor类 Java通过反射可以获取构造方法,和使用构造方法创造对象. 在网上找了一个比较好理解的例子来弄清楚Constructor类. public Constructo ...

  6. JAVA 枚举类的初步理解

    JAVA 枚举类的初步理解 现在Java的枚举类在真实项目中已经用的比较频繁,比静态常量更好用,也更有限定性,enum类可以用来表示有限的类对象,比如星期.月份.性别或者项目中的产品类型 像诸如此类的 ...

  7. Java 面向对象:static的理解

    Java 面向对象:static的理解以及各个代码块的运行顺序 一.static关键字 static修饰方法--称为静态方法,可以直接通过类名调用 static修饰变量--称为静态变量,能被类中所有实 ...

  8. python网络编程视频教程_Java网络开发视频教程 – 一站式学习Java网络编程视频教程 全面理解BIO(无密)...

    Java网络开发视频教程 – 一站式学习Java网络编程视频教程 全面理解BIO(无密) 全面理解BIO/NIO/AIO 网络层编程,是每一个开发者都要面对的技术.课程为解决大家学习网络层知识的难题, ...

  9. 关于java多态中覆盖的理解

    在前面一篇文章中,我已经介绍了关于java多态中重载的理解与规则 所以这篇文章主要介绍多态中的覆盖机制. 首先重载与覆盖除了表现方式不同之外,还有运行时间上的不同,重载是在编译期间就已经可以确定好调用 ...

最新文章

  1. HBase中的时间维度
  2. 2017.12.18
  3. AspNetCore微服务下的网关-Kong(一)
  4. 凌动服务器系列,凌动也能造服务器?超微又出怪异新品
  5. RabbitMQ:计划邮件传递
  6. bupt summer training for 16 #3 ——构造
  7. 实验大作业c语言实现简单ftp,C 语言实现 FTP 服务器
  8. Python稳基修炼的经典案例7(计算机二级、初学者必须掌握的例题)
  9. pmod ad2 digilent 提供的pmodad2.c和pmodad2.h
  10. talib python文档_TALib中文文档代码实现
  11. 怎么理解知行之桥EDI系统的端口?
  12. Cocos2dx-lua组件tableView的简单用法
  13. JAVA实现身份证号码的分析
  14. 轻松学会分布式事务算法
  15. 谈《西游记》和泛项目
  16. oracle招聘ocp认证,OracleOCP认证要通过哪些考试?
  17. Kubernetes----Job控制器
  18. java 双屏显示_Android双屏异显的实现
  19. 密战——惊心动魄的网络犯罪追踪小说
  20. 正则虐我千百遍,我要反抗了

热门文章

  1. Hyperledger Fabric学习笔记(二)- Fabric 2.2.1环境搭建
  2. 使用validation校验
  3. python求列表的平均值的用法_python如何求列表平均值?_后端开发
  4. JavaScript(四)——具体对象(Math、字符串对象、Date对象、Number对象及Boolean对象)
  5. Java基本数据类型和String类型的转化
  6. Java 身份证号码识别系统
  7. sihpostreboot关闭没有权限_安卓保护隐私,给APP一次性的权限
  8. 今年考了N3也不知道能不能过
  9. Racket 6.11提供了稳定的细化类型和依赖函数特性
  10. ISACA:网络安全人员短缺仍是老大难问题