1.接口
1.1黑马信息管理系统集合改进 (应用)
-
使用数组容器的弊端
-
容器长度是固定的,不能根据添加功能自动增长
-
没有提供用于赠删改查的方法
-
-
优化步骤
-
创建新的StudentDao类,OtherStudentDao
-
创建ArrayList集合容器对象
-
OtherStudentDao中的方法声明,需要跟StudentDao保持一致
注意:如果不一致,StudentService中的代码就需要进行修改
-
完善方法(添加、删除、修改、查看)
-
替换StudentService中的Dao对象
-
-
代码实现
OtherStudentDao类
public class OtherStudentDao {// 集合容器private static ArrayList<Student> stus = new ArrayList<>(); static {Student stu1 = new Student("heima001","张三","23","1999-11-11");Student stu2 = new Student("heima002","李四","24","2000-11-11"); stus.add(stu1);stus.add(stu2);} // 添加学生方法public boolean addStudent(Student stu) {stus.add(stu);return true;} // 查看学生方法public Student[] findAllStudent() { Student[] students = new Student[stus.size()]; for (int i = 0; i < students.length; i++) {students[i] = stus.get(i);} return students;} public void deleteStudentById(String delId) {// 1. 查找id在容器中所在的索引位置int index = getIndex(delId);stus.remove(index);} public int getIndex(String id){int index = -1;for (int i = 0; i < stus.size(); i++) {Student stu = stus.get(i);if(stu != null && stu.getId().equals(id)){index = i;break;}}return index;} public void updateStudent(String updateId, Student newStu) {// 1. 查找updateId, 在容器中的索引位置int index = getIndex(updateId);stus.set(index, newStu);} }
StudentService类
public class StudentService {// 创建StudentDao (库管)private OtherStudentDao studentDao = new OtherStudentDao();// 其他方法没有变化,此处省略... }
1.2黑马信息管理系统抽取Dao (应用)
-
优化步骤
-
将方法向上抽取,抽取出一个父类 ( BaseStudentDao )
-
方法的功能实现在父类中无法给出具体明确,定义为抽象方法
-
让两个类分别继承 BaseStudentDao ,重写内部抽象方法
-
-
代码实现
BaseStudentDao类
public abstract class BaseStudentDao {// 添加学生方法public abstract boolean addStudent(Student stu);// 查看学生方法public abstract Student[] findAllStudent();// 删除学生方法public abstract void deleteStudentById(String delId);// 根据id找索引方法public abstract int getIndex(String id);// 修改学生方法public abstract void updateStudent(String updateId, Student newStu); }
StudentDao类
public class StudentDao extends BaseStudentDao {// 其他内容不变,此处省略 }
OtherStudentDao类
public class OtherStudentDao extends BaseStudentDao {// 其他内容不变,此处省略 }
1.3接口的概述(理解)
-
接口就是一种公共的规范标准,只要符合规范标准,大家都可以通用。
-
Java中接口存在的两个意义
-
用来定义规范
-
用来做功能的拓展
-
1.4接口的特点(记忆)
-
接口用关键字interface修饰
public interface 接口名 {}
-
类实现接口用implements表示
public class 类名 implements 接口名 {}
-
接口不能实例化
我们可以创建接口的实现类对象使用
-
接口的子类
要么重写接口中的所有抽象方法
要么子类也是抽象类
1.5接口的成员特点(记忆)
-
成员特点
-
成员变量
只能是常量 默认修饰符:public static final
-
构造方法
没有,因为接口主要是扩展功能的,而没有具体存在
-
成员方法
只能是抽象方法
默认修饰符:public abstract
关于接口中的方法,JDK8和JDK9中有一些新特性,后面再讲解
-
-
代码演示
-
接口
public interface Inter {public static final int NUM = 10; public abstract void show(); }
-
实现类
class InterImpl implements Inter{ public void method(){// NUM = 20;System.out.println(NUM);} public void show(){ } }
-
测试类
public class TestInterface {/*成员变量: 只能是常量 系统会默认加入三个关键字public static final构造方法: 没有成员方法: 只能是抽象方法, 系统会默认加入两个关键字public abstract*/public static void main(String[] args) {System.out.println(Inter.NUM);}}
-
1.6类和接口的关系(记忆)
-
类与类的关系
继承关系,只能单继承,但是可以多层继承
-
类与接口的关系
实现关系,可以单实现,也可以多实现,还可以在继承一个类的同时实现多个接口
-
接口与接口的关系
继承关系,可以单继承,也可以多继承
1.7黑马信息管理系统使用接口改进 (应用)
-
实现步骤
-
将 BaseStudentDao 改进为一个接口
-
让 StudentDao 和 OtherStudentDao 去实现这个接口
-
-
代码实现
BaseStudentDao接口
public interface BaseStudentDao {// 添加学生方法public abstract boolean addStudent(Student stu);// 查看学生方法public abstract Student[] findAllStudent();// 删除学生方法public abstract void deleteStudentById(String delId);// 根据id找索引方法public abstract int getIndex(String id);// 修改学生方法public abstract void updateStudent(String updateId, Student newStu); }
StudentDao类
public class StudentDao implements BaseStudentDao {// 其他内容不变,此处省略 }
OtherStudentDao类
public class OtherStudentDao implements BaseStudentDao {// 其他内容不变,此处省略 }
1.8黑马信息管理系统解耦合改进 (应用)
-
实现步骤
-
创建factory包,创建 StudentDaoFactory(工厂类)
-
提供 static 修改的 getStudentDao 方法,该方法用于创建StudentDao对象并返回
-
-
代码实现
StudentDaoFactory类
public class StudentDaoFactory {public static OtherStudentDao getStudentDao(){return new OtherStudentDao();} }
StudentService类
public class StudentService {// 创建StudentDao (库管)// private OtherStudentDao studentDao = new OtherStudentDao(); // 通过学生库管工厂类, 获取库管对象private OtherStudentDao studentDao = StudentDaoFactory.getStudentDao(); }
2.接口组成更新
2.1接口组成更新概述【理解】
-
常量
public static final
-
抽象方法
public abstract
-
默认方法(Java 8)
-
静态方法(Java 8)
-
私有方法(Java 9)
2.2接口中默认方法【应用】
-
格式
public default 返回值类型 方法名(参数列表) { }
-
作用
解决接口升级的问题
-
范例
public default void show3() { }
注意事项
-
默认方法不是抽象方法,所以不强制被重写。但是可以被重写,重写的时候去掉default关键字
-
public可以省略,default不能省略
-
如果实现了多个接口,多个接口中存在相同的方法声明,子类就必须对该方法进行重写
-
2.3接口中静态方法【应用】
-
格式
public static 返回值类型 方法名(参数列表) { }
范例
-
静态方法只能通过接口名调用,不能通过实现类名或者对象名调用
-
public可以省略,static不能省略
-
-
public static void show() { }
注意事项
-
静态方法只能通过接口名调用,不能通过实现类名或者对象名调用
-
public可以省略,static不能省略
2.4接口中私有方法【应用】
-
私有方法产生原因
Java 9中新增了带方法体的私有方法,这其实在Java 8中就埋下了伏笔:Java 8允许在接口中定义带方法体的默认方法和静态方法。这样可能就会引发一个问题:当两个默认方法或者静态方法中包含一段相同的代码实现时,程序必然考虑将这段实现代码抽取成一个共性方法,而这个共性方法是不需要让别人使用的,因此用私有给隐藏起来,这就是Java 9增加私有方法的必然性
-
定义格式
-
格式1
private 返回值类型 方法名(参数列表) { }
-
范例1
private void show() { }
-
格式2
private static 返回值类型 方法名(参数列表) { }
-
范例2
private static void method() { }
-
-
注意事项
-
默认方法可以调用私有的静态方法和非静态方法
-
静态方法只能调用私有的静态方法
-
3.多态
3.1多态的概述(记忆)
-
什么是多态
同一个对象,在不同时刻表现出来的不同形态
-
多态的前提
-
要有继承或实现关系
-
要有方法的重写
-
要有父类引用指向子类对象
-
-
代码演示
class Animal {public void eat(){System.out.println("动物吃饭");} } class Cat extends Animal {@Overridepublic void eat() {System.out.println("猫吃鱼");} } public class Test1Polymorphic {/*多态的前提: 1. 要有(继承 \ 实现)关系2. 要有方法重写3. 要有父类引用, 指向子类对象*/public static void main(String[] args) {// 当前事物, 是一只猫Cat c = new Cat();// 当前事物, 是一只动物Animal a = new Cat();a.eat(); } }
3.2多态中的成员访问特点(记忆)
-
成员访问特点
-
成员变量
编译看父类,运行看父类
-
成员方法
编译看父类,运行看子类
-
-
代码演示
class Fu {int num = 10; public void method(){System.out.println("Fu.. method");} } class Zi extends Fu {int num = 20; public void method(){System.out.println("Zi.. method");} } public class Test2Polymorpic {/*多态的成员访问特点: 成员变量: 编译看左边 (父类), 运行看左边 (父类) 成员方法: 编译看左边 (父类), 运行看右边 (子类)*/public static void main(String[] args) {Fu f = new Zi();System.out.println(f.num);f.method();} }
3.3多态的好处和弊端(记忆)
-
好处
提高程序的扩展性。定义方法时候,使用父类型作为参数,在使用的时候,使用具体的子类型参与操作
-
弊端
不能使用子类的特有成员
3.4多态中的转型(应用)
-
向上转型
父类引用指向子类对象就是向上转型
-
向下转型
格式:子类型 对象名 = (子类型)父类引用;
-
代码演示
class Fu {public void show(){System.out.println("Fu..show...");} } class Zi extends Fu {@Overridepublic void show() {System.out.println("Zi..show...");} public void method(){System.out.println("我是子类特有的方法, method");} } public class Test3Polymorpic {public static void main(String[] args) {// 1. 向上转型 : 父类引用指向子类对象Fu f = new Zi();f.show();// 多态的弊端: 不能调用子类特有的成员// f.method(); // A: 直接创建子类对象// B: 向下转型 // 2. 向下转型 : 从父类类型, 转换回子类类型Zi z = (Zi) f;z.method();} }
3.5多态中转型存在的风险和解决方案 (应用)
-
风险
如果被转的引用类型变量,对应的实际类型和目标类型不是同一种类型,那么在转换的时候就会出现ClassCastException
-
解决方案
-
关键字
instanceof
-
使用格式
变量名 instanceof 类型
通俗的理解:判断关键字左边的变量,是否是右边的类型,返回boolean类型结果
-
-
代码演示
abstract class Animal {public abstract void eat(); } class Dog extends Animal {public void eat() {System.out.println("狗吃肉");} public void watchHome(){System.out.println("看家");} } class Cat extends Animal {public void eat() {System.out.println("猫吃鱼");} } public class Test4Polymorpic {public static void main(String[] args) {useAnimal(new Dog());useAnimal(new Cat());} public static void useAnimal(Animal a){ // Animal a = new Dog();// Animal a = new Cat();a.eat();//a.watchHome(); // Dog dog = (Dog) a; // dog.watchHome(); // ClassCastException 类型转换异常// 判断a变量记录的类型, 是否是Dogif(a instanceof Dog){Dog dog = (Dog) a;dog.watchHome();}} }
3.6黑马信息管理系统多态改进 (应用)
-
实现步骤
-
StudentDaoFactory类中方法的返回值定义成父类类型BaseStudentDao
-
StudentService中接收方法返回值的类型定义成父类类型BaseStudentDao
-
-
代码实现
StudentDaoFactory类
public class StudentDaoFactory {public static BaseStudentDao getStudentDao(){return new OtherStudentDao();} }
StudentService类
public class StudentService {// 创建StudentDao (库管)// private OtherStudentDao studentDao = new OtherStudentDao(); // 通过学生库管工厂类, 获取库管对象private BaseStudentDao studentDao = StudentDaoFactory.getStudentDao(); }