1. Java 的构造函数 与初始化块:
a. 抽象类的构造函数
若果在父类中(也就是抽象类)中显示的写了有参数的构造函数,在子类是就必须写一个构造函数来调用父类的构造函数
abstract class Person { //定义一个抽象类,必须被继承
Person(int i) {
}
}
public class Student extends Person {
Student() {
super(int i)://必须显示的调用父类构造方法//super代表父类对象
}
}
b. 构造函数
public class Sample{
// 第一:在这a=1
static int a
// 第二:a=2
static{ a=2;}
// 第三:a=4
static{ a=4;}
public static void main(String[] args)...{
// 第四:a=4,执行之后a=5
a++;
// 第五:a=5
System.out.println("a="+a);
}
}
引段网友对类的加载及生命周期的说明:
当类不被用到的时候,什么都不提前加载,
(1)一旦类被使用到(import时),首先加载的是它的静态变量,然后是静态初始化块,静态方法(不调用时不执行)。
(2)当此类要被构造一个对象(new 时)的时候,即被new,或者用反射生成其对象的时候,加载其成员部分。依次为成员变量(即非静态变量)、非静态初始化块、非静态方法、最后才是构造器(不创建实例时不执行)。
成员部分除方法外,每个对象都有其一份副本,其中,非静态方法并不是每个对象都有一个,而是所有对象共享一个副本,这点要注意。
静态部分一旦被加载,就不会被销毁,直到程序结束,关闭虚拟机。
非静态的,即,被各个对象所拥有的东西,当对象不被引用时,这个类就结束了他的生命周期。
但是,它还要在内存中驻留一段时间,等垃圾处理器来将它清除。
在失去引用后,被垃圾处理前的这段时间,虽然它驻留在内存中,但是没法被再次引用。
请注意上面说明的加载顺序:
首先加载 静态变量 然后是静态初始化块,最后是静态方法。
为了验证这个顺序,将上面的代码稍微改了下,添加了个静态方法。
如下
package test1;
public class ClassLoadTest
{
static int a=2;
static{ a=3;}
static{a=4;}
static void init() {a=10;}
/** * @param args
*/
public static void main(String[] args)
{
System.out.println("a="+a);//+为连字符
}
}
结果输出为:a=4
1.构造函数
(1)任何一个类不管它是抽象的还是具体的,都拥有一个构造函数,即使程序员不键入它,Java也会提供一个默认的无参的构造函数。构造函数必须要与类同名,构造函数一定不能够有返回类型,切记void也是一种返回类型!
如果在类中没有创建任何构造函数,那么系统将使用默认的构造函数,如果程序员定义了一个构造函数,那么默认的构造函数将不存在!
public class Book {
private String id;
private String title;
private String author;
//我们自己定义一个构造函数
public Book(String idIn,String titleIn,String authorIn){
id=idIn;
title=titleIn;
author=authorIn;
}
public String toString(){
return "The info of the book:\n"+
"Title:"+title+"\n"+
"Author:"+author+"\n";
}
}
public class Test {
public static void main(String[]args){
//Book book=new Book(); 使用默认的构造函数将出现编译错误
Book book=new Book("0101001","Thinking in Java","Bruce Eckel");
System.out.println(book);
}
}
(2)构造函数的执行方式:
首先调用其超类的构造函数,超类构造函数又调用其超类构造函数,直至到达Object构造函数为止,然后Object()构造函数执行,直到所有的构造函数完成
public class Animal {
public Animal(){
System.out.println("This is the animal constructor");
}
}
public class Snake extends Animal{
public Snake(){
System.out.println("This is snake constructor");
}
}
public class Cobra extends Snake{
public Cobra(){
System.out.println("This is the cobra constructor");
}
}
public class Test {
public static void main(String[]args){
new Cobra();
}
}
执行结果:
This is the animal constructor
This is snake constructor
This is the cobra constructor
(3)默认构造函数是一个无变元的构造函数,隐式的包含了一个对super()的调用
如果一个子类的超类没有无参的构造函数,那么其子类必须程序员实现构造函数,而无法调用默认的构造函数
public class Rpg {
private int hp;
private int mp;
private int grade;
private int exp;
public Rpg(int hpIn,int mpIn,int gradeIn,int expIn){
hp=hpIn;
mp=mpIn;
grade=gradeIn;
exp=expIn;
}
}
public class Magician extends Rpg{
//public Magician(){ 不可以使用默认的构造函数!
//}
public Magician(int hpIn,int mpIn,int gradeIn,int expIn){
super(hpIn,mpIn,gradeIn,expIn);
}
}
(4)构造函数可以重载,如果在同一个类中一个构造函数需要调用另一个重载的构造函数,可以使用this(),this()的变元列表决定了调用哪个具体的构造函数
注意:this()和super()必须出现在构造函数的第一行!!!而且this()和super()函数不能位于同一个构造函数中!!!
抽象类的构造函数在实例化具体子类时被调用
接口是没有构造函数的!
2.初始化块:
Java类中执行操作的地方有三个:
构造函数、方法和初始化块
Java初始化块分为静态初始化块和实例初始化块:
首次加载类时,会运行一次静态初始化块,每次创建一个新实例时,都会运行一次实例初始化块,类中允许出现多个初始化块,它们所执行的顺序与它们在代码中所出现的顺序相同(程序执行时默认是从上到下的)
总体的执行顺序:静态初始化块->super()->实例初始化块->构造函数的其它部分,通过一个例子来说明:
public class Father {
public Father(){
System.out.println("This is super class!");
}
}
public class Test extends Father{
static{
System.out.println("This is static block!");//静态块
}
public Test(){
System.out.println("This is test constructor");//构造函数
}
public static void main(String[]args){
System.out.println("Hello,Java!");
Test test=new Test();
}
{
System.out.println("Common init block!");
}
}
上面的例子的输出结果是:
This is static block!
Hello,Java!
This is super class!
Common init block!
This is test constructor
分享到:
相关推荐
派生类构造函数和析构函数的执行顺序 在面向对象编程中,继承是非常重要的一个概念。派生类可以继承基类的成员,实现代码的重复利用。但是,基类的构造函数和析构函数不能被继承。如果在派生类中需要对新增加的成员...
在Java中,枚举类型可以有构造函数,可以有多个实例变量,可以实现接口,可以有抽象方法等。枚举类型的实例变量可以是final的,也可以不是final的。 在上面的代码中,我们定义了一个枚举类型returnEnum,它有两个...
在C++编程中,"通讯录 c++ 构造函数 类"这一主题涉及到对象导向编程中的核心概念,如类的设计、构造函数的使用以及数据成员和成员函数的管理。下面将详细解释这些知识点: 1. **类(Class)**: 类是C++中的一个...
在C#编程语言中,构造函数(Constructor)是类的一个特殊方法,它的主要作用是在创建对象时初始化新实例。构造函数的名称必须与类名完全相同,并且没有返回类型,即使是void也不行。本篇文章将深入探讨C#中的构造...
如命名空间(namespace)、类成员(Class members)、成员函数(Member ...抽象类(Abstract Classes)、访问控制(Controlling Access to Class Members)、友元(Friends)、构造函数(Constructors)、析构函数...
2. **允许构造函数**:虽然抽象类不能直接实例化,但它们仍然可以拥有构造函数。这些构造函数在子类实例化时被调用,帮助初始化抽象类中可能存在的非静态成员。 3. **抽象类可以没有任何抽象方法**:抽象类不一定...
7. **接口和抽象类**:C#中的接口(Interface)和抽象类(Abstract Class)也是实现多态性的重要手段。接口强制实现一组方法,而抽象类可以包含部分实现。这两种方式都可以用来定义规范,让其他类遵循。 8. **设计...
抽象类可以有构造函数,也可以包含非抽象方法和成员变量,这使得抽象类可以提供一些默认的行为。 接口(Interface)则是另一种形式的抽象,它是完全抽象的,不包含任何方法的实现。接口只定义了方法的签名,而具体...
总的来说,理解和掌握构造函数、析构函数(以及在Java中的`finalize()`方法)、函数的重载、类的继承以及函数覆盖,对于提升Java编程能力至关重要。同时,设计模式的运用可以提高代码的可维护性和可扩展性,使得大型...
在Java语言中,抽象类与接口是实现抽象概念的两种基础机制,它们对于项目的结构设计和扩展性有着深远的影响。要准确地掌握它们各自的用途和特点,对于任何Java开发者而言都是一项基础而重要的技能。 首先,让我们来...
在Java中,抽象类是一种不能被实例化的特殊类,它通常包含抽象方法,即没有具体实现的方法。面试中,对于抽象类的理解和应用是评估开发者技能的关键指标。以下是对一些高难度Java抽象类面试题的详细解析: 1. **...
枚举类使用 enum 关键字定义,隐含了所创建的类型都是 Java.lang.Enum 类的子类(java.lang.Enum 是一个抽象类)。枚举类型符合通用模式 Class Enum<E extends Enum <E>>,而 E 表示枚举类型的名称的每一个值都将...
构造函数可以重载,这意味着一个类中可以有多个构造函数,只要它们的参数列表不同即可。 例如,考虑`Location`类,它定义了三个构造函数:无参构造函数、一个接受单个整数参数的构造函数以及一个接受两个整数参数的...
抽象类you构造函数 抽象类只能用作基类,无法实例化(new) 抽象类不能被实例化。 抽象类存在的意义: 1.抽象类不能被实例化,只能被其他类继承 2.继承抽象类的子类必须把抽象类中的所有抽象成员都重写(实现)(除非...
在C++编程语言中,类(Class)...总之,类、构造函数和析构函数是C++面向对象编程中的关键概念,它们共同实现了对象的生命周期管理、数据封装和行为抽象。理解并熟练运用这些概念对于编写高效、健壮的C++程序至关重要。
C++中的类与对象是面向对象编程的核心概念,它们提供了数据封装和抽象的能力,使得代码更具有可维护性和可扩展性。在这个课件中,我们将深入理解类与对象的定义、使用,以及相关的构造函数、析构函数、访问权限、...
在这个【IT十八掌徐培成】的课程中,我们重点关注了Java基础知识的第四天内容,主要讲解了构造代码块、构造函数以及JavaBean和POJO的概念。下面我们将深入探讨这些关键知识点。 首先,构造代码块是在类中定义的一段...
本案例展示了如何在Java中定义一个简单的`Circle`类,并利用这个类来计算和输出圆的周长和面积。同时,还实现了基本的输入验证功能,确保了程序的健壮性和实用性。通过这种方式,可以帮助初学者更好地理解面向对象...
Java 8之后,接口可以包含默认方法和静态方法,但仍不能有实例字段或构造函数。 - 抽象类可以包含成员变量,而接口只能有常量(final static)。 4. **访问权限** - 抽象类的访问修饰符可以是public、protected、...
抽象类在C#或Java等语言中通常用`abstract`关键字来声明,它们不能被实例化,只能作为其他类的基类来使用。 描述中提到的"窗体应用程序"表明这是一个基于图形用户界面(GUI)的项目,可能使用了.NET Framework的...