Java 面向对象编程之构造方法、构造代码块html
面向对象编程是咱们认识世界,创造事物的一种新的思惟,其实就抓大放小,将全部共性的事物进行封装的过程。编程人员在进行流程做业时,能够方便的进行复用。java
站在面向对象的角度看问题,编程就是对象的动做的调用;站在面向过程的角度,编程就是动做的执行者。“把大象装进冰箱”就能够完美的解释面向对象和面向过程的区别。编程
就如同一个 “Baby” 对象创建以后,那么生孩子的过程就会变得很简单:安全
public class Baby { // 定义一个Baby类 Long IDcard; String name; public void eat() { System.out.println(name+"...eat..."); } public void run(){ System.out.println(name+"...run..."); } }
但是不是全部生下来的孩子都是同样的,他们具备不一样的身份证号码,不一样的名字。jvm
因此,new一个对象很简单,可是要考虑初始化对象,并给予不一样的区别就要对这个类进行构造方法。函数
构造方法(构造函数):给对象进行统一的初始化。如代码:测试
public class Baby { Long IDcard; String name; public void eat() { System.out.println(name+"...eat..."); } public void run(){ System.out.println(name+"...run..."); } public Baby() // 无参数的构造函数 { System.out.println("my name is "+name+";my idcard is :"+IDcard); } public Baby(Long IDcard,String name) // 有参数的构造函数 { System.out.println("my name is "+name+";my idcard is :"+IDcard); } }
1. 构造函数是没有返回值类型的。
2. 构造函数的函数名必需要与类名一致。
3. 构造函数并非由咱们手动调用的,而是在建立对应的对象时,jvm就会主动调用到对应的构造函数。
4. 若是一个类没有显式的写上一个构造方法时,那么java编译器会为该类添加一个无参的构造函数的。
5. 若是一个类已经显式的写上一个构造方法时,那么java编译器则不会再为该类添加 一个无参的构造方法。
6. 构造函数是能够在一个类中以函数重载的形式存在多个的。spa
1. 返回值类型的区别:htm
2. 函数名的区别:对象
3. 调用方式的区别:
4. 做用上的区别:
在Java中,静态代码块、构造代码块、构造函数、普通代码块的执行顺序是一个笔试的考点,经过这篇文章但愿能完全了解它们之间的执行顺序。
(1)静态代码块:
代码结构:在java类中使用static关键字和{}声明的代码块,通常是类中才有这样的结构,方法中不存在。
public class CodeBlock { static{ System.out.println("静态代码块"); } }
执行时机:静态代码块在类被加载的时候就运行了,并且只运行一次,而且优先于各类代码块以及构造函数。若是一个类中有多个静态代码块,会按照书写顺序依次执行。后面在比较的时候会经过具体实例来证实。
静态代码块的做用:通常状况下,若是有些代码须要在项目启动的时候就执行,这时候就须要静态代码块。好比一个项目启动须要加载的不少配置文件等资源,咱们就能够都放入静态代码块中。
静态代码块不能存在任何方法体中:
静态代码块是在类加载的时候就要运行了。咱们分状况讨论:
对于普通方法,因为普通方法是经过加载类,而后new出实例化对象,经过对象才能运行这个方法,而静态代码块只须要加载类以后就能运行了。
对于静态方法,在类加载的时候,静态方法也已经加载了,可是咱们必需要经过类名或者对象名才能访问,也就是说相比于静态代码块,静态代码块是主动运行的,而静态方法是被动运行的。
不论是哪一种方法,咱们须要明确静态代码块的存在在类加载的时候就自动运行了,而放在不论是普通方法仍是静态方法中,都是不能自动运行的。
静态代码块不能访问普通变量:普通变量只能经过对象来调用,是不能放在静态代码块中的。
(2)构造代码块:
代码结构:在java类中使用{}声明的代码块(和静态代码块的区别是少了static关键字):
public class CodeBlock { static{ System.out.println("静态代码块"); } { System.out.println("构造代码块"); } }
执行时机:构造代码块在建立对象时被调用,每次建立对象都会调用一次,可是优先于构造函数执行。须要注意的是,听名字咱们就知道,构造代码块不是优先于构造函数执行,而是依托于构造函数,也就是说,若是你不实例化对象,构造代码块是不会执行的。怎么理解呢?咱们看看下面这段代码:
public class CodeBlock { { System.out.println("构造代码块"); } public CodeBlock(){ System.out.println("无参构造函数"); } public CodeBlock(String str){ System.out.println("有参构造函数"); } }
若是存在多个构造代码块,则执行顺序按照书写顺序依次执行。
构造代码块的做用:
和构造函数的做用相似,都能对对象进行初始化,而且只要建立一个对象,构造代码块都会执行一次。可是反过来,构造函数则不必定每一个对象创建时都执行(多个构造函数状况下,创建对象时传入的参数不一样则初始化使用对应的构造函数)。
利用每次建立对象的时候都会提早调用一次构造代码块特性,咱们能够作诸如统计建立对象的次数等功能。
(3)构造函数:
1.构造函数的命名必须和类名彻底相同。在java中普通函数能够和构造函数同名,可是必须带有返回值;
2.构造函数的功能主要用于在类的对象建立时定义初始化的状态。它没有返回值,也不能用void来修饰。这就保证了它不只什么也不用自动返回,并且根本不能有任何选择。而其余方法都有返回值,即便是void返回值。尽管方法体自己不会自动返回什么,但仍然可让它返回一些东西,而这些东西多是不安全的;
3.构造函数不能被直接调用,必须经过new运算符在建立对象时才会自动调用;而通常的方法是在程序执行到它的时候被调用的;
4.当定义一个类的时候,一般状况下都会显示该类的构造函数,并在函数中指定初始化的工做也可省略,不过Java编译器会提供一个默认的构造函数.此默认构造函数是不带参数的。而通常的方法不存在这一特色;
(4)普函数:
普通代码块和构造代码块的区别是,构造代码块是在类中定义的,而普通代码块是在方法体中定义的。且普通代码块的执行顺序和书写顺序一致。
public void sayHello(){ { System.out.println("普通代码块"); } }
(5)执行顺序:
静态代码块>构造代码块>构造函数>普通代码块
public class CodeBlock { static{ System.out.println("静态代码块"); } { System.out.println("构造代码块"); } public CodeBlock(){ System.out.println("无参构造函数"); } public void sayHello(){ { System.out.println("普通代码块"); } } public static void main(String[] args) { System.out.println("执行了main方法"); new CodeBlock().sayHello();; System.out.println("---------------"); new CodeBlock().sayHello();; } }
(6)父类和子类的执行顺序:
对象的初始化顺序:
首先执行父类静态的内容,父类静态的内容执行完毕后,接着去执行子类的静态的内容,当子类的静态内容执行完毕以后,再去看父类有没有构造代码块,若是有就执行父类的构造代码块,父类的构造代码块执行完毕,接着执行父类的构造方法;父类的构造方法执行完毕以后,它接着去看子类有没有构造代码块,若是有就执行子类的构造代码块。子类的构造代码块执行完毕再去执行子类的构造方法。
总之一句话,静态代码块内容先执行,接着执行父类构造代码块和构造方法,而后执行子类构造代码块和构造方法。
父类:SuperClass.java
public class SuperClass { static{ System.out.println("父类静态代码块"); } { System.out.println("父类构造代码块"); } public SuperClass(){ System.out.println("父类构造函数"); } }
子类:SubClass.java
public class SubClass extends SuperClass{ static{ System.out.println("子类静态代码块"); } { System.out.println("子类构造代码块"); } public SubClass(){ System.out.println("子类构造函数"); } }
测试:
public static void main(String[] args) { SubClass sb = new SubClass(); System.out.println("------------"); SubClass sb1 = new SubClass(); }