1. 继承
1.1 继承的概述
- 继承的概念
- 让类与类之间产生关系(子父类关系),子类可以直接使用父类中非私有的成员
- 继承是面向对象三大特征之一,可以使得子类具有父类的属性和方法,还可以在子类中重新定义,以及追加属性和方法
- 实现继承的格式
- 继承通过extends实现
- 格式:public class 子类名 extends 父类名 { }
- 范例:public class Zi extends Fu { }
- Fu:是父类,也被称为基类、超类
- Zi:是子类,也被称为派生类
- 继承带来的好处
- 继承可以让类与类之间产生关系,子父类关系,产生子父类后,子类则可以使用父类中非私有的成员。
public class Fu {
public void show() {
System.out.println("show方法被调用");
}
}
public class Zi extends Fu {
public void method() {
System.out.println("method方法被调用");
}
}
public class Demo {
public static void main(String[] args) {
Fu f = new Fu();
f.show();
Zi z = new Zi();
z.method();
z.show();
}
}
1.2 继承的好处和弊端
- 继承的好处
- 提高了代码的复用性
- 提高了代码的维护性
- 让类与类之间产生了关系,是多态的前提
- 继承的弊端
- 继承是侵入性的
- 降低了代码的灵活性
- 继承关系,导致子类必须拥有父类非私有属性和方法,让子类自由的世界中多了些约束
- 增强了代码的耦合性
- 继承的应用场景:
- 当类与类之间,存在相同 (共性) 的内容,并且产生了 is a 的关系,就可以考虑使用继承来优化代码
- is…a的关系:谁是谁的一种,例如:老师和学生是人的一种,那人就是父类,学生和老师就是子类
1.3. Java中继承的特点
- Java中类只支持单继承,不支持多继承
- 错误范例:class A extends B, C { }
- Java中类支持多层继承
- 范例:子类 A 继承父类 B ,父类B 可以 继承父类 C
public class Granddad {
public void drink() {
System.out.println("爷爷爱喝酒");
}
}
public class Father extends Granddad {
public void smoke() {
System.out.println("爸爸爱抽烟");
}
}
public class Mother extends Father {
public void dance() {
System.out.println("妈妈爱跳舞");
}
}
public class Demo {
public static void main(String[] args) {
Mother m = new Mother ();
m.drink();
m.smoke();
m.dance();
}
}
1.4 继承中变量的访问特点
- 在子类方法中访问一个变量,采用的是就近原则。
- 子类局部范围找
- 子类成员范围找
- 父类成员范围找
- 注意:
- 如果子父类中,出现了重名的成员变量,通过就近原则,会优先使用子类的
如果一定要使用父类的,可以通过super关键字,进行区分。
class Fu {
int num = 10;
}
class Zi {
int num = 20;
public void show(){
int num = 30;
System.out.println(num);
}
}
public class Demo1 {
public static void main(String[] args) {
Zi z = new Zi();
z.show();
}
}
1.5 super
- this&super关键字:
- this:代表本类对象的引用
- super:代表父类存储空间的标识(可以理解为父类对象引用)
- this和super的使用分别
- 成员变量:
- this.成员变量 - 访问本类成员变量
- super.成员变量 - 访问父类成员变量
- 成员方法:
- this.成员方法 - 访问本类成员方法
- super.成员方法 - 访问父类成员方法
- 构造方法:
- this(…) - 访问本类构造方法
- super(…) - 访问父类构造方法
1.6 继承中构造方法的访问特点
- 子类中所有的构造方法默认都会访问父类中无参的构造方法
- 子类在初始化的时候有可能会使用到父类中的数据,如果父类没有完成初始化,子类将无法使用父类的数据。
- 怎么初始化?
- 构造方法的第一条语句默认都是:super()
- 注意:
- 如果我们编写的类,没有手动指定父类,系统也会自动继承Object (Java继承体系中的最顶层父类)
- 如果父类中没有无参构造方法,只有带参构造方法,该怎么办呢?
- 子类通过 super,手动调用父类的带参的构造方法
- 子类通过 this 去调用本类的其他构造方法,本类其他构造方法再通过 super 去手动调用父类的带参的构造方法
- 注意:
- this(…) super(…) 必须放在构造方法的第一行有效语句,并且二者不能共存。
- 总结
- 子类中所有的构造方法,默认都会通过 super() 访问父类中无参的构造方法
- 每一个子类构造方法的第一条语句默认都是:super()
- this(…) super(…) 必须放在构造方法的第一行有效语句,并且二者不能共存
1.7 继承中成员方法的访问特点
- 通过子类对象访问一个方法
- 子类成员范围找
- 父类成员范围找
- 如果都没有就报错(不考虑父亲的父亲…)
1.8 super内存图
- 对象在堆内存中,会单独存在一块super区域,用来存放父类的数据
![在这里插入图片描述](https://img-blog.csdnimg.cn/0183fdcd35254ed7919163f42a22b1c9.png)
1.9 方法重写
- 方法重写概念
- 在继承的体系中,子类出现了和父类中一模一样的方法声明(方法名一样,参数列表也必须一样)
- 方法重写的应用场景
- 当子类需要父类的功能,而功能主体子类有自己特有内容时,可以重写父类中的方法,这样,即沿袭了父类的功能,又定义了子类特有的内容
- Override注解
- 用来检测当前的方法,是否是重写的方法,起到【校验】的作用
1.10 方法重写的注意事项
- 父类中私有方法不能被重写
- 父类静态方法,子类必须通过静态方法进行重写,父类非静态方法,子类也必须通过非静态方法进行重写
- 静态方法不能被重写!如果子类中,也存在一个方法声明一模一样的方法,可以理解为,子类将父类中同名的方法,隐藏了起来,并非是方法重写!
- 子类重写父类方法时,访问权限必须大于等于父类
public class Fu {
private void show() {
System.out.println("Fu中show()方法被调用");
}
void method() {
System.out.println("Fu中method()方法被调用");
}
}
public class Zi extends Fu {
@Override
private void show() {
System.out.println("Zi中show()方法被调用");
}
@Override
private void method() {
System.out.println("Zi中method()方法被调用");
}
@Override
public void method() {
System.out.println("Zi中method()方法被调用");
}
}
1.11 重写重载的区别
- 方法重写:
- 在继承体系中, 子类出现了和父类一模一样的方法声明 (方法名, 参数列表, 返回值类型)
- 方法重载:
- 在同一个类中, 方法名相同, 参数列表不同, 与返回值无关.
2 权限修饰符
![在这里插入图片描述](https://img-blog.csdnimg.cn/e8bf59fcb6b04d2280797c1eb6440ba3.png)
3 黑马信息管理系统使用继承改进
- 需求
- 把学生类和老师类共性的内容向上抽取,抽取到出一个 Person 父类,让学生类和老师类继承 Person 类
- 实现步骤
- 抽取Person类。
- 优化StudentController类中,inputStudentInfo方法,将setXxx赋值方式,改进为构造方法初始化。
- 注意:直接修改这种操作方式,不符合我们开发中的一个原则
- 开闭原则 ( 对扩展开放对修改关闭 )
- 尽量在不更改原有代码的前提下以完成需求
- 解决:重新创建一个OtherStudentController类
- 编写新的inputStudentInfo方法
- 根据StudentController类、OtherStudentController类,向上抽取出BaseStudentController类
- 再让StudentController类、OtherStudentController类,继承BaseStudentController类
3.1 代码修改-Person类及学生类和老师类
public class Person {
private String id;
private String name;
private String age;
private String birthday;
public Person() {
}
public Person(String id, String name, String age, String birthday) {
this.id = id;
this.name = name;
this.age = age;
this.birthday = birthday;
}
public String getId() {
return id;
}
public void setId(String id) {
this.id = id;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public String getAge() {
return age;
}
public void setAge(String age) {
this.age = age;
}
public String getBirthday() {
return birthday;
}
public void setBirthday(String birthday) {
this.birthday = birthday;
}
}
public class Student extends Person {
public Student() {
}
public Student(String id, String name, String age, String birthday) {
super(id, name, age, birthday);
}
}
public class Teacher extends Person {
public Teacher() {
}
public Teacher(String id, String name, String age, String birthday) {
super(id, name, age, birthday);
}
}
3.2 代码修改-BaseStudentController类
import com.itheima.edu.info.manager.domain.Student;
import com.itheima.edu.info.manager.service.StudentService;
import java.util.Scanner;
public class BaseStudentController {
private StudentService studentService = new StudentService();
private Scanner sc = new Scanner(System.in);
public void start() {
studentLoop:
while (true) {
System.out.println("--------欢迎来到 <学生> 管理系统--------");
System.out.println("请输入您的选择: 1.添加学生 2.删除学生 3.修改学生 4.查看学生 5.退出");
String choice = sc.next();
switch (choice) {
case "1":
addStudent();
break;
case "2":
deleteStudentById();
break;
case "3":
updateStudent();
break;
case "4":
findAllStudent();
break;
case "5":
System.out.println("感谢您使用学生管理系统, 再见!");
break studentLoop;
default:
System.out.println("您的输入有误, 请重新输入");
break;
}
}
}
public void updateStudent() {
String updateId = inputStudentId();
Student newStu = inputStudentInfo(updateId);
studentService.updateStudent(updateId, newStu);
System.out.println("修改成功!");
}
public void deleteStudentById() {
Student[] stus = studentService.findAllStudent();
if (stus == null) {
System.out.println("查无信息, 请添加后重试");
return;
}
String delId = inputStudentId();
studentService.deleteStudentById(delId);
System.out.println("删除成功!");
}
public void findAllStudent() {
Student[] stus = studentService.findAllStudent();
if (stus == null) {
System.out.println("查无信息, 请添加后重试");
return;
}
System.out.println("学号\t\t\t姓名\t\t年龄\t\t生日");
for (int i = 0; i < stus.length; i++) {
Student stu = stus[i];
if (stu != null) {
System.out.println(stu.getId() + "\t" + stu.getName() + "\t" + stu.getAge() + "\t\t" + stu.getBirthday());
}
}
}
public void addStudent() {
String id;
while (true) {
System.out.println("请输入学生id:");
id = sc.next();
boolean flag = studentService.isExists(id);
if (flag) {
System.out.println("学号已被占用, 请重新输入");
} else {
break;
}
}
Student stu = inputStudentInfo(id);
boolean result = studentService.addStudent(stu);
if (result) {
System.out.println("添加成功");
} else {
System.out.println("添加失败");
}
}
public String inputStudentId() {
String id;
while (true) {
System.out.println("请输入学生id:");
id = sc.next();
boolean exists = studentService.isExists(id);
if (!exists) {
System.out.println("您输入的id不存在, 请重新输入:");
} else {
break;
}
}
return id;
}
public Student inputStudentInfo(String id) {
return null;
}
}
3.3 代码修改-StudentController类
import com.itheima.edu.info.manager.domain.Student;
import java.util.Scanner;
public class StudentController extends BaseStudentController {
private Scanner sc = new Scanner(System.in);
@Override
public Student inputStudentInfo(String id) {
System.out.println("请输入学生姓名:");
String name = sc.next();
System.out.println("请输入学生年龄:");
String age = sc.next();
System.out.println("请输入学生生日:");
String birthday = sc.next();
Student stu = new Student();
stu.setId(id);
stu.setName(name);
stu.setAge(age);
stu.setBirthday(birthday);
return stu;
}
}
3.4 代码修改-OtherStudentController类
import com.itheima.edu.info.manager.domain.Student;
import java.util.Scanner;
public class OtherStudentController extends BaseStudentController {
private Scanner sc = new Scanner(System.in);
@Override
public Student inputStudentInfo(String id) {
System.out.println("请输入学生姓名:");
String name = sc.next();
System.out.println("请输入学生年龄:");
String age = sc.next();
System.out.println("请输入学生生日:");
String birthday = sc.next();
Student stu = new Student(id, name, age, birthday);
return stu;
}
}
4 抽象类
4.1 抽象类概述
- 抽象方法: 将共性的行为(方法)抽取到父类之后,发现该方法的实现逻辑无法在父类中给出具体明确,该方法就可以定义为抽象方法。
- 定义格式:
- public abstract 返回值类型 方法名(参数列表);
- 抽象类: 如果一个类中存在抽象方法,那么该类就必须声明为抽象类
- 定义格式:
- public abstract class 类名{}
4.2 抽象类注意事项
- 抽象类不能实例化
- 抽象类中不一定有抽象方法,有抽象方法的类一定是抽象类
- 可以有构造方法
- 抽象类的子类
public abstract class Animal {
public void drink() {
System.out.println("喝水");
}
public Animal() {
}
public abstract void eat();
}
public class Dog extends Animal {
@Override
public void eat() {
System.out.println("狗吃肉");
}
}
public class Cat extends Animal {
@Override
public void eat() {
System.out.println("猫吃鱼");
}
}
public class Test1Animal {
public static void main(String[] args) {
Dog d = new Dog();
d.eat();
d.drink();
Cat c = new Cat();
c.drink();
c.eat();
}
}
5 模板设计模式
- 设计模式
- 设计模式(Design pattern)是一套被反复使用、多数人知晓的、经过分类编目的、代码设计经验的总结。
- 使用设计模式是为了可重用代码、让代码更容易被他人理解、保证代码可靠性、程序的重用性。
- 模板设计模式
- 把抽象类整体就可以看做成一个模板,模板中不能决定的东西定义成抽象方法
- 让使用模板的类(继承抽象类的类)去重写抽象方法实现需求
- 模板设计模式的优势
- 模板已经定义了通用结构,使用者只需要关心自己需要实现的功能即可
public abstract class CompositionTemplate {
public void write() {
System.out.println("<<我的爸爸>>");
body();
System.out.println("啊~ 这就是我的爸爸");
}
public abstract void body();
}
public class Tom extends CompositionTemplate {
@Override
public void body() {
System.out.println("那是一个秋天, 风儿那么缠绵,记忆中,那天爸爸骑车接我放学回家,我的脚卡在了自行车链当中, 爸爸蹬不动,他就站起来蹬...");
}
}
public class Test {
public static void main(String[] args) {
Tom t = new Tom();
t.write();
}
}
6 final(应用)
- fianl关键字的作用
- final代表最终的意思,可以修饰成员方法,成员变量,类
- final修饰类、方法、变量的效果
- fianl修饰类:该类不能被继承(不能有子类,但是可以有父类)
- final修饰方法:该方法不能被重写
- final修饰变量:表明该变量是一个常量,不能再次赋值
- 变量是基本类型,不能改变的是值
- 变量是引用类型,不能改变的是地址值,但地址里面的内容是可以改变的
public class TestFinal {
public static void main(String[] args) {
final int A = 10;
final int MAX = 10;
final int MAX_VALUE = 20;
final Student stu = new Student();
stu.setName("张三");
stu.setName("李四");
}
}
class Student {
final int a = 10;
private String name;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
}
7 黑马信息管理系统使用抽象类、final 改进
- 需求
- 使用抽象类的思想,将BaseStudentController 中的 inputStudentInfo 方法,定义为抽象方法
- 将不希望子类重写的方法,使用 final 进行修饰
- BaseStudentController类
import com.itheima.edu.info.manager.domain.Student;
import com.itheima.edu.info.manager.service.StudentService;
import java.util.Scanner;
public abstract class BaseStudentController {
private StudentService studentService = new StudentService();
private Scanner sc = new Scanner(System.in);
public final void start() {
studentLoop:
while (true) {
System.out.println("--------欢迎来到 <学生> 管理系统--------");
System.out.println("请输入您的选择: 1.添加学生 2.删除学生 3.修改学生 4.查看学生 5.退出");
String choice = sc.next();
switch (choice) {
case "1":
addStudent();
break;
case "2":
deleteStudentById();
break;
case "3":
updateStudent();
break;
case "4":
findAllStudent();
break;
case "5":
System.out.println("感谢您使用学生管理系统, 再见!");
break studentLoop;
default:
System.out.println("您的输入有误, 请重新输入");
break;
}
}
}
public final void updateStudent() {
String updateId = inputStudentId();
Student newStu = inputStudentInfo(updateId);
studentService.updateStudent(updateId, newStu);
System.out.println("修改成功!");
}
public final void deleteStudentById() {
Student[] stus = studentService.findAllStudent();
if (stus == null) {
System.out.println("查无信息, 请添加后重试");
return;
}
String delId = inputStudentId();
studentService.deleteStudentById(delId);
System.out.println("删除成功!");
}
public final void findAllStudent() {
Student[] stus = studentService.findAllStudent();
if (stus == null) {
System.out.println("查无信息, 请添加后重试");
return;
}
System.out.println("学号\t\t\t姓名\t\t年龄\t\t生日");
for (int i = 0; i < stus.length; i++) {
Student stu = stus[i];
if (stu != null) {
System.out.println(stu.getId() + "\t" + stu.getName() + "\t" + stu.getAge() + "\t\t" + stu.getBirthday());
}
}
}
public final void addStudent() {
String id;
while (true) {
System.out.println("请输入学生id:");
id = sc.next();
boolean flag = studentService.isExists(id);
if (flag) {
System.out.println("学号已被占用, 请重新输入");
} else {
break;
}
}
Student stu = inputStudentInfo(id);
boolean result = studentService.addStudent(stu);
if (result) {
System.out.println("添加成功");
} else {
System.out.println("添加失败");
}
}
public String inputStudentId() {
String id;
while (true) {
System.out.println("请输入学生id:");
id = sc.next();
boolean exists = studentService.isExists(id);
if (!exists) {
System.out.println("您输入的id不存在, 请重新输入:");
} else {
break;
}
}
return id;
}
public abstract Student inputStudentInfo(String id);
}
8 代码块
8.1 代码块概述
- 在Java中,使用 { } 括起来的代码被称为代码块
8.2 代码块分类
- 局部代码块
- 位置: 方法中定义
- 作用: 限定变量的生命周期,及早释放,提高内存利用率
public class Test {
public static void main(String[] args) {
{
int a = 10;
System.out.println(a);
}
}
}
- 构造代码块
- 位置: 类中方法外定义
- 特点: 每次构造方法执行的时,都会执行该代码块中的代码,并且在构造方法执行前执行
- 作用: 将多个构造方法中相同的代码,抽取到构造代码块中,提高代码的复用性
public class Test {
public static void main(String[] args) {
Student stu1 = new Student();
Student stu2 = new Student(10);
}
}
class Student {
{
System.out.println("我是构造代码块");
}
public Student() {
System.out.println("空参数构造方法");
}
public Student(int a) {
System.out.println("带参数构造方法");
}
}
- 静态代码块
- 位置: 类中方法外定义
- 特点: 需要通过static关键字修饰,随着类的加载而加载,并且只执行一次
- 作用: 在类加载的时候做一些数据初始化的操作
public class Test {
public static void main(String[] args) {
Person p1 = new Person();
Person p2 = new Person(10);
}
}
class Person {
static {
System.out.println("我是静态代码块, 我执行了");
}
public Person() {
System.out.println("我是Person类的空参数构造方法");
}
public Person(int a) {
System.out.println("我是Person类的带参数构造方法");
}
}
9 黑马信息管理系统使用代码块改进
- 需求
- 实现步骤
- 在StudentDao类中定义一个静态代码块,用来初始化一些学生数据
- 将初始化好的学生数据存储到学生数组中
- StudentDao类
import com.itheima.edu.info.manager.domain.Student;
public class StudentDao {
private static Student[] stus = new Student[5];
static {
Student stu1 = new Student("heima001", "张三", "23", "1999-11-11");
Student stu2 = new Student("heima001", "李四", "24", "2000-11-11");
stus[0] = stu1;
stus[1] = stu2;
}
public boolean addStudent(Student stu) {
int index = -1;
for (int i = 0; i < stus.length; i++) {
Student student = stus[i];
if (student == null) {
index = i;
break;
}
}
if (index == -1) {
return false;
} else {
stus[index] = stu;
return true;
}
}
public Student[] findAllStudent() {
return stus;
}
public void deleteStudentById(String delId) {
int index = getIndex(delId);
stus[index] = null;
}
public int getIndex(String id) {
int index = -1;
for (int i = 0; i < stus.length; i++) {
Student stu = stus[i];
if (stu != null && stu.getId().equals(id)) {
index = i;
break;
}
}
return index;
}
public void updateStudent(String updateId, Student newStu) {
int index = getIndex(updateId);
stus[index] = newStu;
}
}