一.构造器
1.构造器最大的用处就是在创建对象时执行初始化,每个java类必须包含一个或一个以上的构造器。一般系统会提供一个无参的构造器,但是如果我们自己定义了一个构造器后它就不会再起作用了。如果我们为一个类编写了有参构造器,那么最好还是为该类提供一个无参的构造器。
2.构造器的重载
同一个类中可以包含有多个构造器,多个构造器的形参列表不同,即被称为构造器的重载。系统通过new调用构造器时,系统将根据传入的实参列表来决定调用哪个构造器。示例代码如下:
<span style="font-size:18px;">package test;
public class ConstructorOverload {
public String name;
public int count;
//提供无参构造器
public ConstructorOverload(){
}
//提供带两个参数的构造器
public ConstructorOverload(String name,int count){
this.name=name;
this.count=count;
}
public static void main(String[] args){
//通过无参构造器创建ConstructorOverload对象
ConstructorOverload oc1 = new ConstructorOverload();
//通过有参构造器创建ConstructorOverload对象
ConstructorOverload oc2 = new ConstructorOverload("longkaili",21);
System.out.println(oc1.name+" "+oc1.count);
System.out.println(oc2.name+" "+oc2.count);
}
}
</span>
如果一个构造器中完全包含另一个构造器的执行体,那么可以通过this关键字进行调用,这样可以增加代码的可维护性。示例代码如下:
<span style="font-size:18px;">package test;
public class Apple {
public String name;
public String color;
public double weight;
public Apple(){
}
public Apple(String name,String color){
System.out.println("别的构造器调用了我(0.0)");
this.name=name;
this.color=color;
}
public Apple(String name,String color,double weight){
//通过this调用另一个构造器的初始化代码。
this(name,color);
this.weight=weight;
}
public static void main(String[] args){
Apple ap = new Apple("longnkaili","red",110);
System.out.println(ap.name+" "+ap.color+" "+ap.weight);
}
}
</span>
二.java中的继承
java的继承和C++最大的不同就是:java是单继承,每个子类只能有一个直接的父类(当然间接的父类就随便了)。
java的继承通过关键字extends实现;子类继承父类后可以获得父类的所有方法和File,但是不能获得父类的构造器。
下面从几个方向说明继承需要注意的地方。
1.重写父类的方法
如果父类的某个方法对子类是可见的(不为private),然后又在子类中定义了一个与其完全一样的方法,则称为方法的重写,也称为方法覆盖。方法的重写要遵从“两同两小一大”的原则。需要注意的是,覆盖方法和被覆盖方法必须要么都是类方法要么都是实例方法,不能一个是类方法,一个是实例方法。下面是一份示例代码:
package test;
public class Bird {
public void fly(){
System.out.println("我在天空自由自在的飞翔...");
}
}
package test;
public class Ostrich extends Bird{
public void fly(){
System.out.println("我只能在地上奔跑....");
}
public static void main(String[] args){
Ostrich Os = new Ostrich();
Os.fly();
}
}
2.super限定
当子类方法覆盖了父类方法后,子类对象是无法访问父类中被覆盖的方法的,当可以在子类的方法中调用父类中被覆盖的方法。如果需要在子类方法中调用父类方法,可以使用super(被覆盖的是实例方法)或父类类名(被覆盖的是类方法)作为调用者来调用父类中被覆盖的方法。
当程序创建一个子类对象时,系统不仅会为该类中定义的实例变量分配内存空间,还会为其从父类继承的所有实例变量分配内存(不管是不是被子类覆盖)。 示例代码如下;
package test;
public class BaseClass {
public int a = 5;
}
package test;
public class SubClass extends BaseClass{
public int a=7;
public void accessOwer(){
System.out.println(a);
}
public void accessBase(){
//通过super来访问父类中被覆盖的变量
System.out.println(super.a);
}
}
3.子类调用父类的构造器
可以使用super来显式调用父类的构造器,super调用的格式和this一样;super调用的时候必须放在第一行,所以super与this是不能同时出现的。如果不用super显式调用,子类也必定会在其自己的构造器执行前调用父类的无参构造器,而且一定是从其最开始的父类开始调用。示例代码如下:
package test1;
public class Creature {
public Creature(){
System.out.println("Creature的无参构造器");
}
}
package test1;
public class Animal extends Creature{
public Animal(String name){
System.out.println("Animal带一个参数的构造器,"+
"该动物的名字为:"+name);
}
public Animal(String name,int age){
this(name);
System.out.println("Animal带两个参数的构造器,"+
"其age为:"+ age);
}
}
package test1;
public class Wolf extends Animal{
public Wolf(){
super("灰太狼",3);
System.out.println("Wolf的无参构造器");
}
public static void main(String[] args){
new Wolf();
}
}