先分享一张弔图
继承
继承是什么?
继承就是子类继承父类的-8特征和行为,使得子类对象(实例)具有父类的实例域和方法,或子类从父类继承方法,使得子类具有父类相同的行为。
继承是对现实生活中的“分类”概念的一种模拟。
比如说写了一个叫Animal的类,里面的成员是动物的共性。再写一个叫Dog的类来继承Animal类。狗拥有动物的一切基本特性,但同时又拥有自己的独特的特性,这就是“继承”关系的重要特性:通常简称为“IS_A”关系。
Java的继承语法:
1. public:外界可自由访问
2. private:外界不可访问
3. protected:同一包中的子类都可以访问,另一包中的子类(派生于同一个父类)也可以访问
4. default:如果不指明任何权限,则默认同一包中的类可以访问
class 父类 {
……
}
class 子类 extends 父类 {……
}
子类自动拥有父类声明为public和protected的成员。
继承类型
-
Java 的继承是单继承,但是可以多重继承,单继承就是一个子类只能继承一个父类,多重继承就是,例如 B 类继承 A 类,C 类继承 B 类,所以按照关系就是 B 类是 C 类的父类,A 类是 B 类的父类,这是 Java 继承区别于 C++ 继承的一个特性。
class Grandparent {
public Grandparent() {
System.out.println("GrandParent Created.");
}
public Grandparent(String string) {
System.out.println("GrandParent Created.String:" + string);
}
}
class Parent extends Grandparent {
public Parent() {
//请试着先后取消这两句的注释,看看会有什么事情发生
super("Hello.Grandparent.");
System.out.println("Parent Created");
//super("Hello.Grandparent.");
}
}
class Child extends Parent {
public Child() {
System.out.println("Child Created");
}
}
public class TestInherits {
public static void main(String args[]) {
Child c = new Child();
}
}
创建子类对象时,基类的构造方法会被自动地调用。而且是从顶向下调用。
如果要使用super关键字调用基类的特定构造方法,则必须是子类构造方法中的第一条语句。
因为,依据继承的特性,子类代码是可以访问父类成员的,在某些情况下,子类字段的初始化可能需要访问基类的字段,因此,为保证子类字段在任何情况下都能顺利地被初始化,先初始化基类的字段,让其有确定的值,是合理的。
关键字
super 关键字:
我们可以通过 super 关键字来实现对父类成员的访问,用来引用当前对象的父类。
java文件被编译成class文件时,在子类的所有构造函数中的第一行(第一个语句)会默认自动添加 super() 语句,在执行子类的构造函数前,总是会先执行父类中的构造函数。
在编写代码要注意:
- 1.如果父类中不含 默认构造函数(就是 类名() ),那么子类中的super()语句就会执行失败,系统就会报错。一般 默认构造函数 编译时会自动添加,但如果类中已经有一个构造函数时,就不会添加。
- 2.执行父类构造函数的语句只能放在函数内语句的首句,不然会报错。
this 关键字:
指向自己的引用,引用当前对象,即它所在的方法或构造函数所属的对象实例。
class Animal {
void eat() {
System.out.println("animal : eat");
}
}
class Dog extends Animal {
void eat() {
System.out.println("dog : eat");
}
void eatTest() {
this.eat(); // this 调用自己的方法
super.eat(); // super 调用父类方法
}
}
public class Test {
public static void main(String[] args) {
Animal a = new Animal();
a.eat();
Dog d = new Dog();
d.eatTest();
}
}
final 关键字
前面的文章中提到过,final 可以用来修饰变量(包括类属性、对象属性、局部变量和形参)、方法(包括类方法和对象方法)和类。
使用 final 关键字声明类,就是把类定义定义为最终类,不能被继承,或者用于修饰方法,该方法不能被子类重写:
利用final,我们可以设计出一种特殊的“只读” 的“不可变类”
“不可变类”可以方便和安全地用于多线程环境中,访问它们可以不用加锁,因而能提供较高的性能。
final 的作用随着所修饰的类型而不同
1、final 修饰类中的属性或者变量
无论属性是基本类型还是引用类型,final 所起的作用都是变量里面存放的"值"不能变。
这个值,对于基本类型来说,变量里面放的就是实实在在的值,如 1,"abc" 等。
而引用类型变量里面放的是个地址,所以用 final 修饰引用类型变量指的是它里面的地址不能变,并不是说这个地址所指向的对象或数组的内容不可以变,这个一定要注意。
例如:类中有一个属性是 final Person p=new Person("name"); 那么你不能对 p 进行重新赋值,但是可以改变 p 里面属性的值 p.setName('newName');
final 修饰属性,声明变量时可以不赋值,而且一旦赋值就不能被修改了。对 final 属性可以在三个地方赋值:声明时、初始化块中、构造方法中,总之一定要赋值。
2、final修饰类中的方法
作用:可以被继承,但继承后不能被重写。
3、final修饰类
作用:类不可以被继承
构造器
子类是不继承父类的构造器(构造方法或者构造函数)的,它只是调用(隐式或显式)。如果父类的构造器带有参数,则必须在子类的构造器中显式地通过 super 关键字调用父类的构造器并配以适当的参数列表。
如果父类构造器没有参数,则在子类的构造器中不需要使用 super 关键字调用父类构造器,系统会自动调用父类的无参构造器。
class SuperClass {
private int n;
// 无参数构造器
public SuperClass() {
System.out.println("SuperClass()");
}
// 带参数构造器
public SuperClass(int n) {
System.out.println("SuperClass(int n)");
this.n = n;
}
}
// SubClass 类继承
class SubClass extends SuperClass {
private int n;
// 无参数构造器,自动调用父类的无参数构造器
public SubClass() {
System.out.println("SubClass()");
}
// 带参数构造器,调用父类中带有参数的构造器
public SubClass(int n) {
super(300);
System.out.println("SubClass(int n): " + n);
this.n = n;
}
}
// SubClass2 类继承
class SubClass2 extends SuperClass {
private int n;
// 无参数构造器,调用父类中带有参数的构造器
public SubClass2() {
super(300);
System.out.println("SubClass2()");
}
// 带参数构造器,自动调用父类的无参数构造器
public SubClass2(int n) {
System.out.println("SubClass2(int n): " + n);
this.n = n;
}
}
public class TestSuperSub {
public static void main(String[] args) {
System.out.println("------SubClass 类继承------");
SubClass sc1 = new SubClass();
SubClass sc2 = new SubClass(100);
System.out.println("------SubClass2 类继承------");
SubClass2 sc3 = new SubClass2();
SubClass2 sc4 = new SubClass2(200);
}
}
子类与父类方法
在继承关系中,在调用函数(方法)或者类中的成员变量时,JVM(JAVA虚拟机)会先检测当前的类(也就是子类)是否含有该函数或者成员变量,如果有,就执行子类中的,如果没有才会执行父类中的。在子类中,若要调用父类中被重写(或覆盖)的方法,可以使用super关键字。
由于Java并未对子类方法的命名做过多的限制,因此,子类与父类各自定义的方法之间,可以出现以下三种情况:
1. 扩充(Extend):子类定义的方法,父类没有同名方法
2. 覆盖/重写(Override):子类父类定义了完全一样的方法
3. 重载(Overload):子类有父类的同名方法,但两者的参数类型或参数数目不一样
Java“方法覆盖/重写”的语法规则
1. 覆盖/重写方法的允许访问范围不能小于原方法。
2. 覆盖/重写方法所抛出的异常不能比原方法更多。
3. 声明为final方法不允许覆盖/重写。例如,Object的getClass()方法不能覆盖/重写。
4. 不能覆盖/重写静态方法。
public class ShapeAndRectangle {
}
class Shape{
private final int top;
private final int left;
//基类的构造方法
public Shape(int top, int left) {
this.top = top;
this.left = left;
}
}
class Rectangle extends Shape{
private final int width;
private final int height;
public Rectangle(int width,int height){
//调用本类的构造方法
this(0,0,width,height);
}
public Rectangle(int top, int left,
int width, int height) {
super(top, left); //调用父类的构造方法
this.width = width;
this.height = height;
}
}
顶层基类Object
在Java中,所有的类都直接或间接地派生自最顶层的基类Object,此类定义了一些方法,如图所示。
所有的类都是继承于 java.lang.Object,当一个类没有继承的关键字,则默认继承 Object(这个类在 java.lang 包中,所以不需要 import)祖先类。也就是说是所有类的根父类,这个可以运用于参数的传递。
public class Start
{
public static void main(String[] args)
{
A a=new A();
B b=new B();
C c=new C();
D d=new D();
speak(a);
speak(b);
speak(c);
speak(d);
}
// instanceof 关键字是用于比较类与类是否相同,相同返回true,不同返回false
//当你不清楚你需要的参数是什么类型的,可以用Object来代替,Object可以代替任何类
static void speak(Object obj)
{
if(obj instanceof A)//意思是:如果参数是 A 类,那么就执行一下语句
{
A aobj=(A)obj;//这里是向下转换,需要强制转换
aobj.axx();
}
else if(obj instanceof B)
{
B bobj=(B)obj;
bobj.bxx();
}
else if(obj instanceof C)
{
C cobj=(C)obj;
cobj.cxx();
}
}
}
//这里举了四个类,他们的函数都不同,但都是 Object 类的子类
class A
{
void axx()
{
System.out.println("Good morning!");
System.out.println("This is A");
}
}
class B
{
void bxx()
{
System.out.println("Holle!");
System.out.println("This is B");
}
}
class C
{
void cxx()
{
System.out.println("Look!");
System.out.println("This is C");
}
}
class D
{
void dxx()
{
System.out.println("Oh!Bad!");
System.out.println("This is D");
}
}
关于“继承”需要注意的点:
1.当子类重写基类的方法时,注意不要彻底改变方法的职责。子类应该是对基类同名方法行为特性的“调整”或“增强”。
2.尽量少写依据变量类型(是父类还是子类)进行判断的条件语句,而应该采用“多态”来简化代码。
3.避免继承的层次的过深,也就是说,从子类到最顶层的基类的“中间”类的个数不要太多,自定义类的继承层次最好限制在三层以下。
4.当类所对应的客观事物之间存在着“是一种(IS_A)”关系时,为这些类之间建立继承关系才是合理的。
5.基类中可以保存一些所有子类都要用的公用代码,但不要仅为“代码重用”而在类型之间建立继承关系,考虑“组合”取代“继承”。
抽象类
在面向对象的概念中,所有的对象都是通过类来描绘的,但是反过来,并不是所有的类都是用来描绘对象的,如果一个类中没有包含足够的信息来描绘一个具体的对象,这样的类就是抽象类。
抽象类除了不能实例化对象之外,类的其它功能依然存在,成员变量、成员方法和构造方法的访问方式和普通类一样。
由于抽象类不能实例化对象,所以抽象类必须被继承,才能被使用。也是因为这个原因,通常在设计阶段决定要不要设计抽象类。
父类包含了子类集合的常见的方法,但是由于父类本身是抽象的,所以不能使用这些方法。
在 Java 中抽象类表示的是一种继承关系,一个类只能继承一个抽象类,而一个类却可以实现多个接口。
抽象类的特点:
▪ 有abstract修饰的类称为“抽象类”,它只定义了什么方法应该存在,不能直接创建对象。你必须从它派生出一个子类,并在子类中实现其未实现的方法之后,才能使用new关键字创建它的实例。
▪ 在方法前加上abstract就形成抽象方法,这种类型的方法,只有方法声明,没有实现代码。
当然,抽象类中可以包含非抽象方法,也可以象普通类一样,定义字段。
包含抽象方法的类一定是抽象类,但抽象类中的方法不一定是抽象方法。
抽象方法
如果你想设计这样一个类,该类包含一个特别的成员方法,该方法的具体实现由它的子类确定,那么你可以在父类中声明该方法为抽象方法。
Abstract 关键字同样可以用来声明抽象方法,抽象方法只包含一个方法名,而没有方法体。
抽象方法没有定义,方法名后面直接跟一个分号,而不是花括号。
声明抽象方法会造成以下两个结果:
- 如果一个类包含抽象方法,那么该类必须是抽象类。
- 任何子类必须重写父类的抽象方法,或者声明自身为抽象类。
继承抽象方法的子类必须重写该方法。否则,该子类也必须声明为抽象类。最终,必须有子类实现该抽象方法,否则,从最初的父类到最终的子类都不能用来实例化对象。
使用抽象类
▪ 抽象类不能创建对象,一般用它来引用子类对象。
就像下面例子中的:Pet d = new Dog();
抽象类 抽象类变量 = new 派生自抽象类的具体子类();
在现实生活中,属于同一种类的事物,在特定的方面,表现出不同的特点,比如狗和猫都会叫,但两者的叫声是不同的。
• 由于狗和猫都是一种(IS_A)宠物,所以设计一个Pet类,让Dog和Cat成为它的子类。
• 由于狗和猫都会叫,所以给Pet类添加一个speak()方法,又由于并不存在一种“统一”的宠物叫声,而狗和猫的叫声又不一样,所以,应该让Pet.speak()方法成为抽象的。Pet作为抽象类,而Cat和Dog重写基类的speak()方法。
public class UseAbstractClass {
public static void main(String[] args) {
Pet d = new Dog();
//汪汪汪~~~
d.speak();
//喵喵喵~~~
Pet c = new Cat();
c.speak();
}
}
abstract class Pet {
abstract void speak();
}
class Dog extends Pet {
@Override
void speak() {
System.out.println("汪汪汪~~~");
}
}
class Cat extends Pet {
@Override
void speak() {
System.out.println("喵喵喵~~~");
}
}
tips:编译器使用技巧
当IntelliJ检测到你正在编写的类派生自一个抽象类时,会打上红色波浪线,提醒你需要实现基类所定义的抽象方法,并且提供了自动生成方法框架代码的快捷功能。
接口
接口(英文:Interface),在JAVA编程语言中是一个抽象类型,是抽象方法的集合,接口通常以interface来声明。一个类通过继承接口的方式,从而来继承接口的抽象方法。
接口并不是类,编写接口的方式和类很相似,但是它们属于不同的概念。类描述对象的属性和方法。接口则包含类要实现的方法。
除非实现接口的类是抽象类,否则该类要定义接口中的所有方法。
接口无法被实例化,但是可以被实现。一个实现接口的类,必须实现接口内所描述的所有方法,否则就必须声明为抽象类。另外,在 Java 中,接口类型可用来声明一个变量,他们可以成为一个空指针,或是被绑定在一个以此接口实现的对象。
接口与类的关系
接口与类相似点:
- 一个接口可以有多个方法。
- 接口文件保存在 .java 结尾的文件中,文件名使用接口名。
- 接口的字节码文件保存在 .class 结尾的文件中。
- 接口相应的字节码文件必须在与包名称相匹配的目录结构中。
接口与类的区别:
- 接口不能用于实例化对象。
- 接口没有构造方法。
- 接口中所有的方法必须是抽象方法,Java 8 之后 接口中可以使用 default 关键字修饰的非抽象方法。
- 接口不能包含成员变量,除了 static 和 final 变量。
- 接口不是被类继承了,而是要被类实现。
- 接口支持多继承。
接口的语法特性
定义一个接口,采用关键字interface;某个类实现某个接口,采用关键字implements。
接口的成员函数自动成为public的,数据成员自动成为static和final的。
如果接口不声明为public的,则自动变为package的。
一个类可以同时实现多个接口,但只能有一个基类。基类必须写在第一位,后面跟着的是它所实现的多个接口。
▪ 可以通过继承接口来扩充已有接口,并形成一个新的接口。下面的代码块中,[ ]内的可以不加
[可见度] interface 接口名称 [extends 其他的接口名] {
// 声明变量
// 抽象方法
}
Interface关键字用来声明一个接口。下面是接口声明的一个简单例子。
/* 文件名 : NameOfInterface.java */
import java.lang.*;
//引入包
public interface NameOfInterface
{
//任何类型 final, static 字段
//抽象方法
}
接口有以下特性:
- 接口是隐式抽象的,当声明一个接口的时候,不必使用abstract关键字。
- 接口中每一个方法也是隐式抽象的,声明时同样不需要abstract关键字。
- 接口中的方法都是公有的。
类调用接口
class 调用接口的类名 implements 接口名{
//对接口内的抽象方法具体实现
}
接口的使用
接口类型 接口类型的变量 = new 实现了接口的具体类型();
例子
因为“鸭子是一种(IS_A)鸟,又是一种(IS_A)食物”,所以我们可以创建三个类:Bird代表鸟,Food代表食物,Duck代表鸭子,让Duck派生自Bird和Food。
但是由于Java不支持多继承(但是c++支持)︿( ̄︶ ̄)︿,所以使用“接口(interface)”被用来抽象对象的行为特性或外部特征。
Duck类派生自Bird类,实现IFood和ISwim接口,鸭子是一种鸟(用“继承”特性表达),会游泳,同时又是一种食物(用它所实现的“接口”表达)
代码:注意注释
public class DuckAndBird {
public static void main(String[] args) {
Duck duck = new Duck();
duck.fly();
//Duck对象实现了IFood和ISwim接口,
//所以可以赋值给相应的变量
IFood food = duck;
//分别用下面三行替换上面这行,第一行可以正常运行,第二,三行则不行。
//这是因为Duck类中包括了IFood接口,所以可以调用接口内的方法。但是如果换成Bird和ISwim则不行,
// 因为这两个类(接口)中没有包含 .cook() 方法
//Duck food = duck;
//Bird food = duck;
//ISwim food = duck;
food.cook();
ISwim swimAnimal = duck;
swimAnimal.swim();
}
}
//所有食物应该实现这个接口
interface IFood {
void cook();
}
//所有的水鸟,应该实现此接口
interface ISwim {
void swim();
}
abstract class Bird {
//鸟有翅膀,会飞!
abstract void fly();
}
//鸭子是一种水鸟,也是一种食物
class Duck extends Bird
implements ISwim, IFood {
@Override
public void cook() {
System.out.println("我能做成烤鸭!");
}
@Override
public void swim() {
System.out.println("我能游泳!");
}
@Override
void fly() {
System.out.println("我会飞!");
}
}
利用接口定义常量
public interface ArrayBound {
public static final int LOWBOUND = 1;
public static final int UPBOUND = 100;
}
以这种方式定义的常量,说明这些常量需要被实现这个接口的类所使用。请注意要为它们设计出有意义的名字,以便保证代码的可读性。
只要一个类声明实现了这个接口,就可以直接使用这些常量。
定义在接口中的常量必须被初始化
例子:就这样使用
public interface ArrayBound {
public static final int LOWBOUND = 1; // 数组的下界
public static final int UPBOUND = 100; // 数组的上界
}
public class ConstantPrinter implements ArrayBound {
public void printBounds() {
System.out.println("下界: " + LOWBOUND);
System.out.println("上界: " + UPBOUND);
}
public static void main(String[] args) {
ConstantPrinter printer = new ConstantPrinter();
printer.printBounds(); // 打印界限值
}
}
函数式接口
JDK 8中引入了一种“函数式”接口,这种接口只定义有一个公有方法,使用@FunctionalInterface注解加以标注:
@FunctionalInterface interface MyFunctionInterface { void process(); }
这种类型的接口,主要用于“函数式编程”场景,使用它定义的变量,可以接收一个Lambda表达式。
import java.time.LocalDate;
public class UseFunctionalInterface {
public static void main(String[] args) {
MyFunctionInterface obj = () -> { //函数式接口变量,可以引用一个Lambda表达式。
var info = """
工作完成于:%s
""".formatted(LocalDate.now());
System.out.println(info);
};
obj.process();
}
}
@FunctionalInterface
interface MyFunctionInterface {
void process();
}
有关Lambda表达式的内容,之后再介绍。(疯狂挖坑)
JDK9引入的新特性
- 接口中的私有方法,主要用于封装一些接口的默认方法(可以有多个)需要调用的公用代码。
public interface MyInterface {
//JDK 9: 接口中可以定义私有成员,它可以被接口的默认方法调用
private void secret() {
System.out.println("Secret Method defined in interface");
}
//接口的默认方法
default void defaultMethod(){
secret();
System.out.println("defaultMethod()");
}
void func();
}
- 实现了接口的具体类型,如果没有重写,就直接“继承”接口所定义的默认方法。
public class MyClass implements MyInterface {
@Override
public void func() {
System.out.println("MyClass.func()");
}
public static void main(String[] args) {
MyInterface obj = new MyClass();
obj.defaultMethod(); //直接使用接口所定义的默认方法
obj.func();
}
}
interface MyInterface {
//JDK 9: 接口中可以定义私有成员,它可以被接口的默认方法调用
private void secret() {
System.out.println("Secret Method defined in interface");
}
//接口的默认方法
default void defaultMethod(){
secret();
System.out.println("defaultMethod()");
}
void func();
}
MyClass
实现了 MyInterface
接口,并提供了 func()
方法的具体实现。因为 MyClass
没有重写 defaultMethod()
,所以它自动“继承”了接口中定义的默认实现。当调用obj.defaultMethod()时,该方法中的逻辑将被执行。
多态
在面向对象理论中,多态性的定义是:同一操作,作用于不同的类的实例,不同的类将进行不同的解释,最后产生不同的结果。
利用多态特性,看上去完全一样的一条语句,在不同的运行环境中,却可以产生不同的运行结果。
多态代码的最本质特征就是——父类(或接口)变量可以引用子类(或实现了接口的类)对象。换句话说:子类对象可以被当成基类对象使用。其典型代码如下所示:
//父类变量引用子类对象
Parent p = new Child();
//MyClass类实现了IMyInterface接口
IMyInterface obj = new MyClass();
这个特性其实在前面的例子代码中也写了,只是没有展开来细说。
作用:
- 总是可以让更一般的对象引用更具体化的对象。
- Java类库的最顶层类是Object。因此每个对象都可以赋值给Object变量。但是这样只能调用Object类中的方法,比如equal , getClass等
子类与基类变量间的赋值
向上转型
子类对象赋值给基类引用是一种常见的多态性特征。通过这种方式,可以使用基类类型的引用来操作子类对象。具体来说,当你将子类对象赋值给基类引用时,Java会自动进行类型转换(向上转型)。这被称为向上转型(Upcasting)。
向上转型是安全的,不会导致运行时错误,但在使用基类引用时只能访问基类的方法和属性。
向下转型
- 基类对象要赋给子类对象变量,必须执行类型强制转换,其语法是:
子类对象变量=(子类名)基类对象变量;
- 如果需要访问子类特有的方法,需要进行向下转型(Downcasting)。向下转型时需要小心操作,因为只有在基类引用实际上指向的对象是所转换的子类的实例时,向下转型才是安全的。换句话说,如果你尝试将一个基类引用转换为不兼容的子类类型,就会抛出异常。
父类引用指向不同子类的实例: 如果父类引用实际指向一个不同子类的实例,而你尝试将其转换为不相干的子类。
class Animal {}
class Dog extends Animal {}
class Cat extends Animal {}
public class Test {
public static void main(String[] args) {
Animal myAnimal = new Cat(); // myAnimal指向Cat实例
Dog myDog = (Dog) myAnimal; // 这个向下转型会抛出ClassCastException
}
}
null
值: 尝试将 null
值进行向下转型不会抛出 ClassCastException
,但如果你在访问该引用的成员时,就会遇到 NullPointerException
。
使用 instanceof
检查: 在进行向下转型之前,最好使用 instanceof
关键字检查对象的实际类型。这可以确保你不会尝试将一个不兼容的类进行转换。
为了避免 ClassCastException
,可以使用 instanceof
来检查类型:
class Animal {}
class Dog extends Animal {
void bark() {
System.out.println("Bark");
}
}
public class Test {
public static void main(String[] args) {
Animal myAnimal = new Dog(); // 向上转型
// 使用 instanceof 进行检查
if (myAnimal instanceof Dog) {
Dog myDog = (Dog) myAnimal; // 安全的向下转型
myDog.bark(); // 调用 Dog 类的方法
} else {
System.out.println("myAnimal is not an instance of Dog");
}
}
}
子类和父类出现同名成员
同名方法
先给一个例子来测试:
public class ParentChildTest {
public static void main(String[] args) {
var parent = new ParentClass();
parent.printValue();
var child = new ChildClass();
child.printValue();
parent = child;
parent.printValue();
parent.myValue++;
parent.printValue();
((ChildClass) parent).myValue++;
parent.printValue();
}
}
class ParentClass {
public int myValue = 100;
public void printValue() {
System.out.println("Parent.printValue(),myValue=" + myValue);
}
}
class ChildClass extends ParentClass {
public int myValue = 200;
public void printValue() {
System.out.println("Child.printValue(),myValue=" + myValue);
}
}
这个代码的运行结果是
当子类与父类拥有一样的方法,并且让一个父类变量引用一个子类对象时,到底调用哪个方法,由引用对象的“真实”类型所决定,这就是说:引用的对象是子类型的,它就调用子类型的方法,是父类型的,它就调用父类型的方法。
同名同类型的字段
如果子类与父类有相同的字段,则子类中的字段会代替或隐藏父类的字段,子类方法中访问的是子类中的字段(而不是父类中的字段)。如果子类方法确实想访问父类中被隐藏的同名字段,可以用之前提到过的super关键字来访问它。
然而,如果子类被当作父类使用,则通过子类访问的字段是父类的。
class Parent
{
public int value=100;
public void Introduce(){
System.out.println("I'm father");
}
}
class Son extends Parent{
public int value=101;
public void Introduce(){
System.out.println("I'm son");
}
}
class Daughter extends Parent{
public int value=102;
public void Introduce(){
System.out.println("I'm daughter");
}
}
public class TestPolymorphism {
public static void main(String args[]){
Parent p=new Parent();
p.Introduce();
System.out.println(p.value);
p=new Son();
p.Introduce();
System.out.println(p.value);
p=new Daughter();
p.Introduce();
System.out.println(p.value);
}
}
c++与Java的面向对象语法比较
由于是先学的c++的面向对象,所以在此比较一下两种语言的区别
1. Java 虚函数
虚函数的存在是为了多态。
C++ 中普通成员函数加上 virtual 关键字就成为虚函数。
Java 中其实没有虚函数的概念,它的普通函数就相当于 C++ 的虚函数,动态绑定是 Java 的默认行为。如果 Java 中不希望某个函数具有虚函数特性,可以加上 final 关键字变成非虚函数。
PS: 其实 C++ 和 Java 在虚函数的观点大同小异,异曲同工罢了。
2. Java抽象函数(纯虚函数)
抽象函数或者说是纯虚函数的存在是为了定义接口。
C++ 中纯虚函数形式为:
virtual void print() = 0;
Java 中纯虚函数形式为:
abstract void print();
PS: 在抽象函数方面 C++ 和 Java 还是换汤不换药。
3. Java 抽象类
抽象类的存在是因为父类中既包括子类共性函数的具体定义,也包括需要子类各自实现的函数接口。抽象类中可以有数据成员和非抽象方法。
C++ 中抽象类只需要包括纯虚函数,既是一个抽象类。如果仅仅包括虚函数,不能定义为抽象类,因为类中其实没有抽象的概念。
Java 抽象类是用 abstract 修饰声明的类。
PS: 抽象类其实是一个半虚半实的东西,可以全部为虚,这时候变成接口。
4. Java 接口
接口的存在是为了形成一种规约。接口中不能有普通成员变量,也不能具有非纯虚函数。
C++ 中接口其实就是全虚基类。
Java 中接口是用 interface 修饰的类。
PS: 接口就是虚到极点的抽象类。
总结来说就是这样:
C++ 虚函数 == Java 普通函数
C++ 纯虚函数 == Java 抽象函数
C++ 抽象类 == Java 抽象类
C++ 虚基类 == Java 接口