书城计算机网络大话设计模式
8183300000099

第99章 4 理解构造器――构造器和方法的区别?

大B:“要学习Java,你必须理解构造器。因为构造器可以提供许多特殊的方法,这个对于初学者经常混淆。但是,构造器和方法又有很多重要的区别。”

小A:“那它的功能和作用的有什么不同?”

大B:“功能和作用的不同下面我就详细给你讲一下。构造器是为了创建一个类的实例。这个过程也可以在创建一个对象的时候用到:Platypus p1=new Platypus();相反,方法的作用是为了执行java代码。修饰符,返回值和命名的不同。”

小A:“构造器和方法在这里有什么区别?”

大B:“修饰符,返回值,命名。和方法一样,构造器可以有任何访问的修饰:public,protected,private或者没有修饰(通常被package 和friendly调用)不同于方法的是,构造器不能有以下非访问性质的修饰:abstract,final,native,static,或者synchronized。返回类型也是非常重要的。方法能返回任何类型的值或者无返回值(void),构造器没有返回值,也不需要void。”

小A:“构造器和方法命名有什么不同?”

大B:“构造器使用和类相同的名字,而方法则不同。按照习惯,方法通常用小写字母开始,而构造器通常用大写字母开始。构造器通常是一个名词,因为它和类名相同;而方法通常更接近动词,因为它说明一个操作。‘this’的用法,构造器和方法使用关键字this有很大的区别。方法引用this指向正在执行方法的类的实例。静态方法不能使用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();

}

}

大B:“在刚才讲的代码中,有2个不同参数列表的构造器。第一个构造器,给类的成员name赋值,第二个构造器,调用第一个构造器,给成员变量name一个初始值‘John/Mary Doe’。在构造器中,如果要使用关键字this,那么,必须放在第一行,如果不这样,将导致一个编译错误。‘super’的用法,构造器和方法,都用关键字super指向超类,但是用的方法不一样。方法用这个关键字去执行被重载的超类中的方法。”

看下面的例子:

class Mammal{

void getBirthInfo(){

System。out。println(“born alive。”);

}

}

class Platypus extends Mammal{

void getBirthInfo(){

System。out。println(“hatch from eggs”);

System。out。print(“a mammal normally is”);

super。getBirthInfo();

}

}

大B:“从我们刚才讲的例子中,使用super。getBirthInfo()去调用超类Mammal中被重载的方法。构造器使用super去调用超类中的构造器。而且这行代码必须放在第一行,否则编译将出错。”

看下面的例子:

public class SuperClassDemo{

SuperClassDemo(){}

}

class Child extends SuperClassDemo{

Child(){

super();

}

}

大B:“在上面这个没有什么实际意义的例子中,构造器Child()包含了super,它的作用就是将超类中的构造器SuperClassDemo实例化,并加到Child类中。”

小A:“编译器怎样自动加入代码?”

大B:“编译器自动加入代码到构造器,对于这个,java程序员新手可能比较混淆。当我们写一个没有构造器的类,编译的时候,编译器会自动加上一个不带参数的构造器。”

例如:public class Example{}

编译后将如下代码:

public class Example{

Example(){}

}

在构造器的第一行,没有使用super,那么编译器也会自动加上,例如:

public class TestConstructors{

TestConstructors(){}

}

编译器会加上代码,如下:

public class TestConstructors{

TestConstructors(){

super;

}

}

仔细想一下,就知道下面的代码

public class Example{}

经过会被编译器加代码形如:

public class Example{

Example(){

super;

}

}

继承

构造器是不能被继承的。子类可以继承超类的任何方法。看看下面的代码:

public class Example{

public void sayHi{

system。out。println(“Hi”);

}

Example(){}

}

public class SubClass extends Example{

}

大B:“我们来注意一下java功能语句。修饰不能用abstract,final,native,static,or synchronized 能返回类型,没有返回值,没有void,有返回值,或者void,命名和类名相同;通常为名词,大写开头,通常代表一个动词的意思,小写开头,this 指向同一个类中另外一个构造器,在第一行指向当前类的一个实例,不能用于静态方法,super调用父类的构造器,在第一行调用父类中一个重载的方法。继承,构造器不能被继承,方法可以被继承,编译器自动加入一个缺省的构造器,自动加入(如果没有)不支持。编译器自动加入一个缺省的调用到超类的构造器,自动加入(如果没有)不支持。”