1.继承


继承是一种代码复用的方式.
Student与Person有相同部分的代码。
Student可以从Person继承,这样Student获得了Person的所有功能,只需要编写新增的功能即可。通过继承,可以实现代码的复用。

继承使用关键字extends,一个类只能有一个父类。
如果没有写明继承类,编译器会自动指定该类继承于基类Object。
Person:超类super,父类,基类
Student:子类subclass,扩展类

Person.java
```#java
//默认继承Object
public class Person /*extends Object */{
private String name;
private int age;
public void setName(String name){
this.name = name;
}
public String getName(){
return this.name;
}
public void setAge(){
this.age = age;
}
public int getAge(){
return this.age;
}
public void run(){
System.out.println(name+" is running!");
};
}
```
Student.java
```#java
public class Student extends Person{
private int score;
public void setScore(int score){
this.score = score;
}
public int getScore(){
return this.score;
}
}
```
Hello.java
```#java
public class Hello {
public static void main(String[] args){
Person ming = new Person();
Student hong = new Student();
ming.setName("小明");
hong.setName("小红");
ming.run();
hong.run();
}
}
```

2.protected

上例中Person类定义的private字段无法被子类访问,用protected修饰的字段可以被子类访问。

protected把字段和方法的访问权限控制在继承树内部

Person.java

public class Person /*extends Object */{
protected String name; //将name的修饰符更改为protected
private int age;
public void setName(String name){
this.name = name;
}
public String getName(){
return this.name;
}
public void setAge(){
this.age = age;
}
public int getAge(){
return this.age;
}
public void run(){
System.out.println(name+" is running!");
};
}

Student.java

public class Student extends Person{
private int score;
public void setScore(int score){
this.score = score;
}
public int getScore(){
return this.score;
}
public String hello(){
return "Hello, " + this.name; //引用name
}
}
public class Hello {
public static void main(String[] args){
Person ming = new Person();
Student hong = new Student();
ming.setName("小明");
hong.setName("小红");
ming.run();
System.out.println(hong.hello());//引用hello方法
hong.run();
}
}

3.继承关系中的构造方法

编写Person类时,可以编写构造方法,或由编译器自动生成构造方法。由于子类包含有父类的所有功能,必须手动调用父类的构造方法。

Java语言规定,子类的构造方法第一行语句必须调用父类的构造方法。调用方式是super();

如果我们没有写super(),编译器会自动生成super()。如果父类没有默认的构造方法,此时编译器自动生成的构造方法会报错,因为父类的构造方法必须传入参数。此时我们需要显式的写上super,传入参数。

Person.java

public class Person /*extends Object */{
protected String name;
private int age;
public Person(){ //默认的构建方法
this.name = "王重阳";
}
public Person(String name){
this.name = name;
}
public void setAge(int age){
this.age = age;
}
public int getAge(){
return this.age;
}
public void run(){
System.out.println(name+" is running!");
};
}

Student.java

public class Student extends Person{
public Student(){
//根据父类默认的构建方法,自动生成super();
}
public Student(String name){
super(name);
}
private int score;
public void setScore(int score){
this.score = score;
}
public int getScore(){
return this.score;
}
}

Hello.java

public class Hello {
public static void main(String[] args){
Student p = new Student("林朝英");
Student s = new Student();
p.run();
s.run();
}
}


# 4.向上转型
定义一个类时,实际是定义一个数据类型;定义继承时,是在数据类型之间加上了继承的关系。
定义一个实例对象时,需要把它赋值给一个引用类型的变量,让这个变量指向实例对象。
向上转型把一个子类型安全地变为更加抽象的类型(父类型)。
反过来,即向下转型是把一个父类型(抽象)变成一个自类型(具体)。如把一个Person类型变量强制转型为Student类型变量。
可以对实例变量进行向上转型(upcasting)和向下转型(downcasting)
```#java
//假如我们持有一个Student对象实例,name我们可以保证Student包含Person类型的全部功能,因此把它看成一个Person类型对象是没有问题的。
Student s = new Student("jack");
Person ps = s;
Student s2 = (Student) ps;
s2.run();
```

        /*Person实例p可能指向Person实例,也可能指向Student实例。向下转型为Student很有可能报错。如果变量p指向的实际类型并不是Student实例,JVM运行时会抛出ClassCastException的错误。*/
Person p = new Person("tom");
Student s2 = (Student) p;
s2.run();


## 4.1instanceof操作符
```#java
public class Hello {
public static void main(String[] args){
Person p = new Person("tom");
System.out.println(p instanceof Person);
System.out.println((p instanceof Student ) + "\n");

    Student s = new Student("jack");
System.out.println(s instanceof Person);
System.out.println((s instanceof Student) + "\n"); Student n = null;
System.out.println(n instanceof Object);
System.out.println(n instanceof Person);
System.out.println(n instanceof Student);
}

}

<img src="https://img2018.cnblogs.com/blog/1418970/201901/1418970-20190114213553125-21523129.png" width="500" />
使用instanceof修改向下转型代码
```#java
Person p = new Person("tom");
Student s = new Student("jack");
Person ps = s;
if(p instanceof Student) { //只在类型正确时转型,避免异常
Student s2 = (Student) p;
s2.run();
}


# 5.继承和组合
组合和子类使用不会,需要翻书
Student可以持有一个Book实例
继承是is关系,组合是has关系
```#java
public class Student extends Person{
private Book book;
public Student(String name){
super(name);
}
private int score;
public void setScore(int score){
this.score = score;
}
public int getScore(){
return this.score;
}
...
}
```

6.总结:

继承是面向对象编程的一种代码复用方式。

Java只允许单继承

protected允许子类访问父类的字段和方法

子类的构造方法可以通过super()调用父类的构造方法

可以安全地向上转型为更为抽象的类型

可以强制向下转型,最好借助instanceof判断

子类和父类的关系是is,has关系不能用继承

廖雪峰Java2面向对象编程-3继承和多态-1继承的更多相关文章

  1. 廖雪峰Java2面向对象编程-1面向对象-1面向对象基础

    1.对象的概念 面向对象编程:Object-Oriented Programming 对现实世界建立计算机模型的一种编程方法. 现实世界 计算机模型 Java代码 人 类/class class Pe ...

  2. 廖雪峰Java2面向对象编程-3继承和多态-2多态

    1.重载 子类覆写父类的方法称为重载Override. 父类和子类拥有一摸一样的方法(方法的名字.返回值.参数是相同的,但是方法的语句是不一样的) 方法签名如果不同就不是重载,而是创建了一个新的方法. ...

  3. 廖雪峰Java2面向对象编程-4抽象类和接口-2接口

    1.接口的定义 抽象方法本质上是定义接口规范. 在抽象类中定义了一个抽象方法,子类中必须实现这个抽象方法. public abstract class Person{ public abstract ...

  4. 廖雪峰Java2面向对象编程-4抽象类和接口-1抽象类

    每个子类都可以覆写父类的方法 如果父类的方法没有实际意义,能否去掉方法的执行语句?子类会报编译错误 如果去掉父类的方法,就失去了多态的特性 可以把父类的方法声明为抽象方法. 如果一个class定义了方 ...

  5. 廖雪峰Java2面向对象编程-6Java核心类-6常用工具类

    1.Math Math提供了数学计算的静态方法 序号 方法 描述 1 abs() 返回参数的绝对值.Math.abs(-9)//9 2 ceil() 返回大于等于( >= )给定参数的的最小整数 ...

  6. 廖雪峰Java2面向对象编程-6Java核心类-5枚举类

    直接定义常量 public class Weekday { //定义int常量 public static final int SUN = 0; public static final int MON ...

  7. 廖雪峰Java2面向对象编程-6Java核心类-3包装类型

    Java的数据类型: 基本类型:int boolean float 引用类型:所有class类型 为一个基本类型int赋值为null,会提示"incompatible types" ...

  8. 廖雪峰Java2面向对象编程-5包和classpath-3作用域

    1.访问权限 Java的类.接口.字段和方法都可以设置访问权限 访问权限是指在一个类的内部,能否引用另一个类以及访问它的字段和方法 访问权限有public.protected.private和pack ...

  9. 廖雪峰Java2面向对象编程-6Java核心类-4JavaBean

    1.JavaBean定义 符合以下命名规范的class被成为JavaBean private 类型的field 针对这个field的get和set方法 public class Person { pr ...

随机推荐

  1. 2018.4.23 深入理解java虚拟机(转)

    深入理解java虚拟机 精华总结(面试) 一.运行时数据区域 Java虚拟机管理的内存包括几个运行时数据内存:方法区.虚拟机栈.本地方法栈.堆.程序计数器,其中方法区和堆是由线程共享的数据区,其他几个 ...

  2. 为移动端而设计的bootstrap的使用

    一.下载 --bs ---css ----bootstrap.css ----bootstrap.min.css ---fonts ---js ----bootstrap.js ----jquery. ...

  3. HTML表格元素

      学习要点:     1.表格元素总汇     2.构建表格解析 一.表格元素总汇     表格的基本构成最少需要三个元素:<table>.<tr>.<td>,其 ...

  4. 求约束------------------------ do while循环 算法思想

    前段代码: <%@ Page Language="C#" AutoEventWireup="true" CodeFile="Default2.a ...

  5. CH4908 Race

    题意 4908 Race 0x49「数据结构进阶」练习 描述 给定一棵 N 个节点的树,每条边带有一个权值. 求一条简单路径,路径上各条边的权值和等于K,且路径包含的边的数量最少. 输入格式 第一行两 ...

  6. UTF-8 Invalid Byte Sequences

    Chances are, some of you have run into the issue with the invalid byte sequence in UTF-8 error when ...

  7. nginx配置基于域名的虚拟主机

    其实基于域名和基于ip的虚拟主机配置是差不多的,在配置基于ip的虚拟主机上我们只需要修改几个地方就能变成基于域名的虚拟主机,一个是要修改域名,一个是host文件直接看代码 [root@localhos ...

  8. mysql再探

    select子句及其顺序 select from where group by having order by limit 创建表 create table student(id int not nu ...

  9. mysqldump命令之常用选项

    ===============================================mysqldump常用选项-h, --host=name:服务器IP-u, --user=name:登录名 ...

  10. Gravitee.io 架构

    Gravitee.io 官方文档提供了几张架构图,通过图我们可以有一个整体的认识. 全局架构 平台架构 参考资料 https://docs.gravitee.io/apim_overview_arch ...