Java学习笔记六(继承性及相关)

面向对象的特征之二:继承性

继承性的好处:
①减少了代码的冗余,提高了代码的复用性
②便于功能的扩展
③为后面的多态提供了前提条件

继承性的格式:
class A extends B{}
其中A是子类、派生类、subclass
B是父类、超类、基类、superclass
继承性的体现:一旦子类A继承了父类B,就认为子类A获得了父类B中声明的所有属性和方法;特别的,父类中声明为private的属性和方法,子类在继承父类之后,仍然认为子类中继承了父类中的私有结构,只是因为封装性的影响,不能够直接在子类中调用。
子类在继承父类之后还可以声明自己的特有的属性和方法,实现功能的扩展。

关于继承性的规定
1、java中是单继承的,一个子类只能有一个直接父类;
2、子类直接继承的父类叫直接父类,间接继承的父类叫间接父类;子类在继承父类之后,就获得了直接父类和间接父类中声明的所有属性和方法;
3、虽然一个子类只能继承一个父类,但是一个父类可以被多个子类继承;

4、如果没有显式的声明一个类的父类的话,那么这个类就继承于java.lang.Object类,所有的类都直接或间接的继承于java.lang.Object类,也就是说所有的java类都具有java.lang.Object类中声明的属性和方法。

package com.atguigu.java;

public class Creature {
	
	public void breath(){
		System.out.println("呼吸");
	}
	
}
package com.atguigu.java;

public class Person extends Creature{//Person中虽然没有定义breath方法
									//但从Creature中继承得到了该方法
	
	String name;
	private int age;
	
	public Person(){
		
	}
	
	public Person(String name,int age){
		this.name = name;
		this.age = age;
	}
	
	public void eat(){
		System.out.println("吃饭");
		sleep();
	}
	
	private void sleep(){
		System.out.println("睡觉");
	}

	public int getAge() {
		return age;
	}

	public void setAge(int age) {
		this.age = age;
	}
	
	
}

方法的重写:(override/overwrite)

重写:子类继承父类之后,可以对父类中同名同参数的方法进行覆盖。
应用:重写以后,如果创建子类的对象,通过子类的对象调用子父类中同名同参数的方法,实际执行的是子类中重写的方法。
重写的规定
方法的声明:权限修饰符 返回值类型 方法名(形参列表) throws 异常类型{}
几点注意
①子类中重写的方法的方法名和形参列表要与父类中被重写的方法的方法名和形参列表相同;
②子类中重写方法的权限修饰符不小于父类中被重写的方法;特殊情况,子类不能重写父类中声明为private权限的方法。
③父类中被重写方法的返回值类型void,则子类中重写方法的返回值类型也只能是void。
④父类中被重写方法的返回值类型是A类型,则子类中重写方法的返回值类型可以是A类型及A类型的子类。
⑤父类中被重写的方法的返回值类型是基本数据类型,则子类中重写的方法也只能是同类型的基本数据类型。
⑥子类重写的方法抛出的异常类型不大于父类被重写的方法抛出的异常类型(具体放到异常处理时候讲)。
⑦子类和父类中的同名同参数的方法要么都声明为非static的(考虑重写),要么都声明为static的(不是重写)。

public class Student extends Person{

	String major;
	
	public Student(){
		
	}
	public Student(String major){
		this.major = major;
	}
	
	public void study(){
		System.out.println("学习。专业是:" + major);
	}
	
	//对父类中的eat()进行了重写
	public void eat(){
		System.out.println("学生应该多吃有营养的食物");
	}
	
	public void show(){
		System.out.println("我是一个学生");
	}
	
	public String info(){
		return null;
	}
	
//	public int info1(){
//		return 1;
//	}
	
//	public void walk(int distance){
//		System.out.println("重写的方法");
//	}
	
	
	public void walk(int distance) {
		System.out.println("重写的方法");
	}

}

子类对象实例化的全过程

1、从结果上来看:(继承性)
子类继承父类以后,就获取了父类中声明的属性或方法。
创建子类的对象,在堆空间中,就会加载所有父类中声明的属性。

2、 从过程上来看
当我们通过子类的构造器创建子类对象时,我们一定会直接或间接的调用其父类的构造器,进而调用父类的父类的构造器,…直到调用了java.lang.Object类中空参的构造器为止。正因为加载过所有的父类的结构,所以才可以看到内存中有父类中的结构,子类对象才可以考虑进行调用。
明确:虽然创建子类对象时,调用了父类的构造器,但是自始至终就创建过一个对象,即为new的子类对象。

super关键字的使用

1、super理解为:父类的
2、super可以用来调用:属性、方法、构造器
3、super的使用:调用属性和方法

  • 3.1 我们可以在子类的方法或构造器中。通过使用"super.属性"或"super.方法"的方式,显式的调用父类中声明的属性或方法。但是,通常情况下,我们习惯省略"super."
  • 3.2 特殊情况:当子类和父类中定义了同名的属性时,我们要想在子类中调用父类中声明的属性,则必须显式的使用"super.属性"的方式,表明调用的是父类中声明的属性。
  • 3.3 特殊情况:当子类重写了父类中的方法以后,我们想在子类的方法中调用父类中被重写的方法时,则必须显式的使用"super.方法"的方式,表明调用的是父类中被重写的方法。

4.super调用构造器

  • 4.1 我们可以在子类的构造器中显式的使用"super(形参列表)"的方式,调用父类中声明的指定的构造器
  • 4.2 "super(形参列表)"的使用,必须声明在子类构造器的首行!
  • 4.3 我们在类的构造器中,针对于"this(形参列表)"或"super(形参列表)"只能二选一,不能同时出现
  • 4.4 在构造器的首行,没有显式的声明"this(形参列表)“或"super(形参列表)”,则默认调用的是父类中空参的构造器:super()
  • 4.5 在类的多个构造器中,至少有一个类的构造器中使用了"super(形参列表)",调用父类中的构造器
public class SuperTest {
	public static void main(String[] args) {
		
		Student s = new Student();
		s.show();
		
		System.out.println();
		
		s.study();
		
		Student s1 = new Student("Tom", 21, "IT");
		s1.show();
		
		System.out.println("************");
		Student s2 = new Student();
		
	}
}

发布了11 篇原创文章 · 获赞 14 · 访问量 5454

猜你喜欢

转载自blog.csdn.net/ssnszds/article/details/104492124