目录
继承性(inheritance)
为什么要有继承?
作用:
继承举例
方法的重写
重写举例
四种访问权限修饰符
关键字—super
关键字super举例
调用父类的构造器
调用父类构造器举例
子类对象的实例化过程
多态性
概念
使用
多态性应用举例
虚拟方法调用(Virtual Method Invocation)
方法的重载与重写
多态小结
总结
注意:不要仅为了获取其他类中某个功能而去继承
子类继承了父类,就继承了父类的方法和属性。
- 在子类中,可以使用父类中定义的方法和属性,也可以创建新的数据和方法。
- 在Java 中,继承的关键字用的是“extends”,即子类不是父类的子集, 而是对父类的“扩展”。
关于继承的规则:子类不能直接访问父类中私有的(private)的成员变量和方法
Java只支持单继承和多层继承,不允许多重继承
// Student类继承了父类Person的所有属性和方法,并增加了一个属性school。Person中的属性和方法,Student都可以使用
class Person {public String name;public int age;public Date birthDate;public String getInfo() {// ...}
}
class Student extends Person {public String school;
}
定义:在子类中可以根据需要对从父类中继承来的方法进行改造,也称 为方法的重置、覆盖。在程序执行时,子类的方法将覆盖父类的方法。
要求:
注意:
子类与父类中同名同参数的方法必须同时声明为非static的(即为重写),或者同时声明为 static的(不是重写)。因为static方法是属于类的,子类无法覆盖父类的方法。
public class Person {public String name;public int age;public String getInfo() {return "Name: "+ name + "\n" +"age: "+ age;}
}
public class Student extends Person {public String school;public String getInfo() { //重写方法return "Name: "+ name + "\nage: "+ age +"\nschool: "+ school;
}
public static void main(String args[]){Student s1=new Student();s1.name="Bob";s1.age=20;s1.school="school2";System.out.println(s1.getInfo()); //Name:Bob age:20 school:school2}
}
对于class的权限修饰只可以用public和default(缺省)。 public类可以在任意地方被访问。 default类只可以被同一个包内部的类访问
访问控制分析
父类Parent和子类Child在同一包中定义时在Java类中使用super来调用父类中的指定操作:
注意:
尤其当子父类出现同名成员时,可以用super表明调用的是父类中的成员
super的追溯不仅限于直接父类
super和this的用法相像,this代表本类对象的引用,super代表父类的内存空间的标识
class Person {protected String name = "张三";protected int age;public String getInfo() {return "Name: " + name + "\nage: " + age;}
}
class Student extends Person {protected String name = "李四";private String school = "New Oriental";public String getSchool() {return school;}public String getInfo() {return super.getInfo() + "\nschool: " + school;}}
public class StudentTest {public static void main(String[] args) {Student st = new Student();System.out.println(st.getInfo());
}}
public class Person {private String name;private int age;private Date birthDate;public Person(String name, int age, Date d) {this.name = name;this.age = age;this.birthDate = d;
}
public Person(String name, int age) {this(name, age, null);
}
public Person(String name, Date d) {this(name, 30, d);
}
public Person(String name) {this(name, 30);}
}public class Student extends Person {private String school;public Student(String name, int age, String s) {super(name, age);school = s;
}
public Student(String name, String s) {super(name);school = s;
}
// 编译出错: no super(),系统将调用父类无参数的构造器。
public Student(String s) { school = s;}
}
class Creature {public Creature() {System.out.println("Creature无参数的构造器");
}}
class Animal extends Creature {public Animal(String name) {System.out.println("Animal带一个参数的构造器,该动物的name为" + name);
}
public Animal(String name, int age) {this(name);System.out.println("Animal带两个参数的构造器,其age为" + age);
}}
public class Wolf extends Animal {public Wolf() {super("灰太狼", 3);System.out.println("Wolf无参数的构造器");
}
public static void main(String[] args) {new Wolf();
}}
多态性,是面向对象中最重要的概念,在Java中的体现:
对象的多态性:父类的引用指向子类的对象 可以直接应用在抽象类和接口上
Java引用变量有两个类型:编译时类型和运行时类型。编译时类型由声明 该变量时使用的类型决定,运行时类型由实际赋给该变量的对象决定。简称:编译时,看左边;运行时,看右边。
若编译时类型和运行时类型不一致,就出现了对象的多态性(Polymorphism)
Person p = new Student(); Object o = new Person();//Object类型的变量o,指向Person类型的对象 o = new Student(); //Object类型的变量o,指向Student类型的对象子类可看做是特殊的父类,所以父类类型的引用可以指向子类的对象:向上转型(upcasting)。 一个引用类型变量如果声明为父类的类型,但实际引用的是子类对象,那么该变量就不能再访问子类中添加的属性和方法
Student m = new Student(); m.school = “pku”; //合法,Student类有school成员变量 Person e = new Student(); e.school = “pku”; //非法,Person类没有school成员变量属性是在编译时确定的,编译时e为Person类型,没有school成员变量,因而编译错误。
方法声明的形参类型为父类类型,可以使用子类的对象作为实参调用该方法
public class Test {public void method(Person e) {// ……e.getInfo();
}
public static void main(Stirng args[]) {Test t = new Test();Student m = new Student();t.method(m); // 子类的对象m传送给父类类型的参数e}
}
正常的方法调用
Person e = new Person(); e.getInfo(); Student e = new Student(); e.getInfo();
虚拟方法调用(多态情况下)
子类中定义了与父类同名同参数的方法,在多态情况下,将此时父类的方法称为虚拟方法,父 类根据赋给它的不同子类对象,动态调用属于子类的该方法。这样的方法调用在编译期是无法确定的。Person e = new Student(); e.getInfo(); //调用Student类的getInfo()方法编译时类型和运行时类型 编译时e为Person类型,而方法的调用是在运行时确定的,所以调用的是Student类的getInfo()方法。——动态绑定
重载,是指允许存在多个同名方法,而这些方法的参数不同。编译器根据方法不 同的参数表,对同名方法的名称做修饰。对于编译器而言,这些同名方法就成了 不同的方法。它们的调用地址在编译期就绑定了。
Java的重载是可以包括父类和子类的,即子类可以重载父类的同名不同参数的方法。 所以:对于重载而言,在方法调用之前,编译器就已经确定了所要调用的方法, 这称为“早绑定”或“静态绑定”;
而对于多态,只有等到方法调用的那一刻,解释运行器才会确定所要调用的具体方法,这称为“晚绑定”或“动态绑定”。
引用一句Bruce Eckel的话:“不要犯傻,如果它不是晚绑定,它就不是多态。”
package com.jyc.p2;
/*
① 面向对象的特征之 继承性一.继承性的好处减少了代码的冗余,提高了代码的复用性便于功能扩展为了多态性的使用提供了前提二.继承性的格式
class A extends B{}
A:子类 派生类 subclss
B :父类 超类 基类 superclass体现::1. 一但子类a继承 子类b后 子类a中获取了b类中声明的结构属性方法特别的父类中声明为private的属性或方法 子类继承父类以后,仍任为获取了父类的私有结构只是因为封装性的影响 使得子类不能够直接调用父类的结构而已2.子类继承父类以后,还可以声明自己特有的属性和方法三 java中关于继承的规定1.一个类可以有多个子类继承2. java类中的单继承性:一个类只能有一个父类3.字符类是相对的概念4.子类直接继承的父类 称为直接父类 间接继承的父类:间接父类5.子类继承父类以后,就获得了直接父类以及间接父类中声明的属性和方法四 如果我们没有显示的定义一个类的父类的话则此类继承于java.lang.object类所以的java类(除java.lang.object)都直接或者间接的继承了java.lang.object类意味着 所以的java类具有 java.lang.object类声明的功能② 方法的重写重写:子类继承父类以后,可以对父类中同名同参数的方法进行覆盖应用 重写以后,当创建子类对象以后,通过子类对象调用父类中同名的参数方法时,实际执行的是子类重写的方法重写的规定方法声明:权限修饰符 返回值 方法名(形参列表)throws异常类型{方法体}约定俗称:子类中叫重写的方法,父类中叫被重写的方法1.子类重写的方法的方法名和形参列表与父类被重写的方法的法名和形参列表相同2.子类重写的方法修饰符不小于父类被重写的方法的权限修饰符特殊情况 子类不能重写父类中声明为private权限符3.返回值类型:父类被重写的方法返回值是void,则子类重写的方法的返回值只能是void父类被重写的返回值类型是A类型 则子类重写的方法的返回值类型可以是a类或者a类的子类父类被重写的返回值类型是基本数据类型(int),则类重写的方法的返回值类型必须是相同的基本数据类型(int)4.子类重写的方法抛出的异常类型不大于父类被重写的方法抛出的异常类型5.子类和父类的同名参数的方法要么都声明为非static的(考虑重写),要么都声明为static的(非重写)③四种不同权限修饰符同一个包中的其他类不可以调用私有的属性方法(private)在不同包的子类中,不能调用声明为 private && 缺省 权限的结构在不同包下的普通类(非子类)不可以调用声明为 private && 缺省 && protected 权限的属性和方法④ super关键字的使用super 理解为父类的 可以用来调用 属性 方法 构造器super的使用一.调用属性和和方法1. 我们可以在再子类的方法或者构造器中,通过super.属性 或者super.方法 可以显示的调用父类中声明的属性和方法2. 单数通常情况下我们习惯省略“super”3. 特殊情况,当子类和父类中定义了同名的属性时,我们要想在子类中调用父类声明的属性,则必须显示的使用super.属性的方式表明的是父类的属性4.特殊情况当子类重写了父类方法以后,我们想在子类的方法中调用父类被重写的方法时,则必须显示的使用super.方法的方式表明的是父类的方法二.调用构造器1.我们可以在子类的构造器中显示的使用super(形参列表)的方式,嗲用父类中声明的指定构造器2.super(形参列表)必须声明在子类构造的首行3.我们在类的构造器中针对于“this(形参列表)”或 super(形参列表)只能二选一4.在构造器的首行,没有显示的声明“this(形参列表)”或 super(形参列表) 则默认调用父类中空参的构造器5.在类的构造器中,至少有一个类的构造器使用了“super(形参列表),调用父类中的构造器⑤ 子类实例化的全过程1.结构上来看子类继承父类以后就获取了父类中声明的属性和方法创建子类的对象,在堆空间中就会加载所有父类中声明的属性和方法从过程上来看当我们通过子类的构造器创建子类对象时,我们一定会直接或者间接的调用父类构造器,进而调用父类的父类的构造器,直到调用了java.lang.Object类中的空参构造器为止,正因为加载了所有的父类构造器,所以才可以看到内存中所以的父类中的结构,子类对象才可以考虑进行调用虽然创建子类对象时,调用了父类构造器,但是自始至终就创建过一个对象,即为 new的子类对象⑥ 面向对象特征 多态性1.多态性的理解 可以理解为一个事物的多种形态,2.何为多态性:对象的多态性 (父类的引用指向了子类的对象) 子类的对象赋给父类的引用3.多态的使用当调用子父类同名同参数的方法时,实际执行的是子类重写父类的方法----虚拟方法调用有了对象的多态性以后,我们在编译器,只能调用父类中声明的方法,但在运行期我们实际执行的是子类重写的父类方法总结 编译看左边 执行看右边4.多态性的使用前提; 1.类的继承关系,2.要有方法的重写5.对象的多态性 只适用于方法,不适于属性(编译和运行都看左边)* */
public class ExendsTest {public static void main(String[] args) {System.out.println("------父类--------");Person p1= new Person();p1.age=1;p1.eat();System.out.println("------子类--------");Student s1=new Student();
// s1.eat();
// s1.sleep();s1.setAge(10);s1.sleep();System.out.println(s1.getAge());System.out.println("------老师子类--------");Teacher t1=new Teacher("tom",18);t1.show();System.out.println("------多态性--------");Person d1=new Teacher();d1.eat();System.out.println("------多态性举例--------");ExendsTest test= new ExendsTest();test.func(new Dog());test.func(new Cat());}public void func(Animal anilm){//Animal anilm=new Dog()anilm.eat();anilm.shout();}
}
class Person{String name;int age;int id=100;//身份证public Person(){System.out.println("我无处不在");}public Person(String name, int age){this.name=name;this.age=age;}public void eat(){System.out.println("吃饭");}public void sleep(){System.out.println("睡觉");eat();show();}public void setAge(int age) {this.age = age;}public int getAge() {return age;}private void show(){System.out.println("我是一个人");}public Object info(){return null;}
}class Student extends Person{
// String name;
// int age;String major;public Student(){}public Student(String name, int age,String major){this.name=name;this.age=age;this.major=major;}
// private void sleep(){
// System.out.println("睡觉");
// }//重写public void eat(){System.out.println("吃饭应该多吃营养的食物");}public void study(){System.out.println("学习");}public void show(){System.out.println("我是一个学生");}public String info(){return null;}
}
class Teacher extends Person{String job;int id=1; //工牌public Teacher(){}public Teacher(String name, int age){super(name, age);}public void show(){System.out.println("name="+name+",age"+age);System.out.println("id="+this.id);System.out.println("id="+super.id);this.eat();super.eat();}public void eat(){System.out.println("老师吃饭");}
}//多态性的举例
class Animal{public void eat(){System.out.println("动物,进食");}public void shout(){System.out.println("动物,叫");}
}
class Dog extends Animal{public void eat(){System.out.println("狗吃骨头");}public void shout(){System.out.println("汪汪汪");}
}
class Cat extends Animal{public void eat(){System.out.println("猫吃鱼");}public void shout(){System.out.println("喵喵喵");}
}