第一节 JAVA中的多态
对象的多种形态,继承是多态的实现基础。
1.引用多态
父类的引用可以指向本类的对象
父类的引用可以指向子类的对象
Animal obj1=new Animal();
Animal obj2=new Dog();
2.方法多态
创建本类对象,调用的方法为本类方法;创建子类对象,调用的方法为子类重写的方法或者继承的方法。如果是独有的方法(子类新建的方法,不与继承方法同名,非重写,父类的引用指向子类的对象时(多态情况下),不能调用该方法)注意不能使用子类的引用指向父类对象。
第二节 引用类型转换
这里说的对象类型转换,是指存在继承关系的对象,不是任意类型的对象。当对不存在继承关系的对象进行强制类型转换时,java 运行时将抛出 java.lang.ClassCastException 异常。
1.向上类型转换(隐式/自动类型转换),是从小类型到大类型的转换
2.向下类型转换(强制类型转换),是从大类型到小类型,这种转换有风险,可能会溢出。
3.instanceof运算符,解决引用对象的类型,避免类型转换的安全问题
在继承链中,我们将子类向父类转换称为“向上转型”,将父类向子类转换称为“向下转型”。很多时候,我们会将变量定义为父类的类型,却引用子类的对象,这个过程就是向上转型。程序运行时通过动态绑定来实现对子类方法的调用,也就是多态性。然而有些时候为了完成某些父类没有的功能,我们需要将向上转型后的子类对象再转成子类,调用子类的方法,这就是向下转型。注意:不能直接将父类的对象强制转换为子类类型,只能将向上转型后的子类对象再次转换为子类类型。也就是说,子类对象必须向上转型后,才能再向下转型。
Dog dog=new Dog();
Animal animal=dog;
Cat cat=(Cat)animal;
编译时不会出错(按Cat类型进行编译),但运行时会报错,因为它开辟的是Dog类型的空间,而(无法将引用类型进行转换)无法将dog对象转换成Cat类型,并且此方法对程序的安全性有影响。此时应该利用instanceof和if语句结合使用,进行验证,以保证程序的安全性,如:
if(animal instanceof Cat){
//判断animal类中是否包含Cat类型的元素,若包含则进行转换,instanceof返回值为布尔类型
Cat cat=(Cat)animal;
}else{
System.out.println("无法进行类型转换");
}
instanceof可以判断一个引用是否是某个类型或者是某个类型的子类型,它会返回一个布尔值。if(animal instanceof Cat) 如果你这个对象里含有cat元素,就可以转换由于animal先被转换成了Dog类型,指向了dog的内存空间,类型不匹配。
第三节 抽象类
1.语法:抽象类前使用abstract关键字修饰
2.应用场景
a.约束子类,在某些情况下,某个父类只是知道其子类应该包含怎样的方法,但无法准确知道这些子类如何实现这些方法
b.可以把有多个相同特征的类抽象出一个抽象类,以这个抽象类作为子类的模板,从而避免了子类设计的随意性
3.作用:限制规定子类必须实现某些方法,但不关注实现细节
4.使用规则
a.abstract定义抽象类
b.abstract定义抽象方法,只有声明,不需要实现
c.包含抽象方法的类是抽象类
d.抽象类中可以包含普通方法,也可以没有抽象方法
e.抽象类不能直接创建,通常是使用它定义一个引用变量,指向一个子类对象
抽象类没有方法体,分号结束。public abstract void call{};
第四节 JAVA中的接口
1.接口的概念:接口可以理解为一种特殊的类,由全局常量和公共的抽象方法所组成,类是一种具体的实现体,而接口定义了某一批类需要遵守的规范,接口不关心这批类的内部数据,也不关心这些类方法的实现细节,它只规定了这些类必须提供某些方法。
2.接口定义:和类定义不同,定义接口不再使用class关键字,而是使用interface关键字。
接口定义的基本语法:
修饰符 interface 接口名 extends 父接口1 父接口2.......{
0-多个常量定义....
0-多个抽象方法的定义....
}
接口就是用来被定义和实现的,修饰符一般使用public,注意不能使用private和protected修饰接口,类在JAVA中是单继承的,接口是多继承的,interface前面有abstract,不写系统也会默认添加。
常量:接口中的属性是常量,即使定义时不添加public static final修饰符,系统也会自动加上。
方法:接口中的方法只能是抽象方法,总是使用,即使定义的时候不添加 public abstract 修饰符,系统也会自动加上。
3.使用接口
一个类可以实现一个或多个接口,实现接口使用implements关键字,java中的一个类只能继承一个父类,是不够灵活的。通过实现多个接口可以作补充,一个类可以实现一个或多个接口。继承父类实现接口的语法:
修饰符 class 类名 extends 父类 implements 接口1,接口2 ....{
类体部分//如果继承了抽象类,需要实现继承的抽象方法,要实现接口中的方法
}
如果要继承父类,继承父类必须在实现接口前。接口一般是 I 字开头,使用接口的引用指向了一个实现接口的对象,之后调用方法。
接口的使用还经常与匿名内部类配合使用,匿名内部类就是没有名字的内部类,多用于关注实现而不关注实现类的名称。
语法格式:直接new一个接口,然后在接口里直接实现这个方法。
Interface i=new Interface(){
public void method(){
输出打印“匿名内部类实现接口的方式”
}
};
注意大括号后面还有个分号,接口的引用指向了一个new接口的代码,并且直接在代码里写接口的实现部分,实现以后要用分号结束,还有一种方法,直接,new 接口{方法}.方法名();直接引用,如下:
第五节 UML
1、UML概念:Unified Modeling Language(UML),又称统一建模语言或标准建模语言,是一个支持模型化和软件系统开发的图形化语言,为软件开发的所有阶段提供模型化和可视化支持。
2、UML图示:UML2.2中一共定义了14种图示(diagrams)
3、常用UML图
用例图:用例图能够以可视化的方式,表达系统如何满足所收集的业务规则,以及特定的用户需求等信息。
序列图:序列图用于按照交互发生的一系列顺序,显示对象之间的这些交互。
类图:UML类图、业务逻辑和所有支持结构一同被用于定义全部的代码结构。
类图:类名,属性,方法,-表示私有,+表示公有
第六节 汽车租赁系统
数据分析,模型分析,业务模型分析,显示和流程分析,模拟一个汽车租赁系统。
测试代码
package com.imooc;
import java.util.Scanner;
/**
* Created by Administrator on 2017/3/9.
*/
public class Test {
public static void main(String[] args) {
int zuCheShuLiang = 0;
int num4 = 0;
int zongRenShu = 0;
int zongZaiHuo = 0;
int zongTianShu = 0;
int zongZuJin = 0; //定义变量
System.out.println("欢迎使用答答租车系统:");
System.out.println("您是否要租车:1是 0否");
Scanner numShuRu1 = new Scanner(System.in);
int num1 = numShuRu1.nextInt();
final String[][] Xinxi = {{"序号 ", "汽车名称 ", "租金 ", "容量 "}, {
"1. ", "奥迪A4 ", "500元/天 ", "载人:4人 "}, {
"2. ", "马自达6 ", "400元/天 ", "载人:4人 "}, {
"3. ", "皮卡雪6 ", "450元/天 ", "载人:4人,载货:2吨"}, {
"4. ", "金龙 ", "800元/天 ", "载人:20人 "}, {
"5. ", "松花江 ", "400元/天 ", "载货:4吨 "}, {
"6. ", "依维柯 ", "1000元/天 ", "载货:20吨 "}};
if (num1 == 1) {
System.out.println("您可租车的类型及其价目表:");
for (int i = 0; i < Xinxi.length; i++) {
for (int j = 0; j < Xinxi[i].length; j++) {
System.out.print(Xinxi[i][j]);
}
System.out.println();
}
System.out.println("近期租车人数过多,每类车型限租1辆,请输入您要租汽车的数量:");
Scanner numShuRu2 = new Scanner(System.in);
int num2 = numShuRu2.nextInt();
zuCheShuLiang = num2;
if (zuCheShuLiang <= 0 || zuCheShuLiang > 6) {
do {
System.out.println("您输入的值有误,请再次输入您要租汽车的数量:");
Scanner numShuRu3 = new Scanner(System.in);
int num3 = numShuRu3.nextInt();
zuCheShuLiang = num3;
num2=zuCheShuLiang;
} while (zuCheShuLiang <= 0||zuCheShuLiang > 6);
}
String[] cheName = new String[zuCheShuLiang];
AoDi car1 = new AoDi();
MaZiDa car2 = new MaZiDa();
PiKaXue car3 = new PiKaXue();
JinLong car4 = new JinLong();
SongHuaJiang car5 = new SongHuaJiang();
YiWeiKe car6 = new YiWeiKe();
for (int x = 0; zuCheShuLiang > 0; zuCheShuLiang--, x++) {
System.out.println("请输入第" + (x + 1) + "辆车的序号:");
Scanner numShuRu4 = new Scanner(System.in);
num4 = numShuRu4.nextInt();
switch (num4) {
case 1:
cheName[x] = car1.carName;
zongRenShu = zongRenShu + car1.zaiRen;
zongZaiHuo = zongZaiHuo + car1.zaiHuo;
zongZuJin = zongZuJin + car1.zuJin;
continue;
case 2:
cheName[x] = car2.carName;
zongRenShu = zongRenShu + car2.zaiRen;
zongZaiHuo = zongZaiHuo + car2.zaiHuo;
zongZuJin = zongZuJin + car2.zuJin;
continue;
case 3:
cheName[x] = car3.carName;
zongRenShu = zongRenShu + car3.zaiRen;
zongZaiHuo = zongZaiHuo + car3.zaiHuo;
zongZuJin = zongZuJin + car3.zuJin;
continue;
case 4:
cheName[x] = car4.carName;
zongRenShu = zongRenShu + car4.zaiRen;
zongZaiHuo = zongZaiHuo + car4.zaiHuo;
zongZuJin = zongZuJin + car4.zuJin;
continue;
case 5:
cheName[x] = car5.carName;
zongRenShu = zongRenShu + car5.zaiRen;
zongZaiHuo = zongZaiHuo + car5.zaiHuo;
zongZuJin = zongZuJin + car5.zuJin;
continue;
case 6:
cheName[x] = car6.carName;
zongRenShu = zongRenShu + car6.zaiRen;
zongZaiHuo = zongZaiHuo + car6.zaiHuo;
zongZuJin = zongZuJin + car6.zuJin;
continue;
}
}
System.out.println("请输入您租车的天数:");
Scanner numShuRu5 = new Scanner(System.in);
zongTianShu = numShuRu5.nextInt();
System.out.println("您的账单:");
System.out.println("***可载人的车有:");
for (int y = 0; y < num2; y++) {
if ((cheName[y]==car1.carName) && (car1.zaiRen > 0))System.out.print(cheName[y]+" ");
if ((cheName[y]==car2.carName) && (car2.zaiRen > 0)) System.out.print(cheName[y]+" ");
if ((cheName[y]==car3.carName) && (car3.zaiRen > 0)) System.out.print(cheName[y]+" ");
if ((cheName[y]==car4.carName) && (car4.zaiRen > 0)) System.out.print(cheName[y]+" ");
if ((cheName[y]==car5.carName) && (car5.zaiRen > 0)) System.out.print(cheName[y]+" ");
if ((cheName[y]==car6.carName) && (car6.zaiRen > 0)) System.out.print(cheName[y]+" ");
}
System.out.println(" 共载人:" + zongRenShu + "人");
System.out.println("***可载货的车有:");
for (int y = 0; y < num2; y++) {
if ((cheName[y]==car1.carName) && (car1.zaiHuo > 0)) System.out.print(cheName[y]+" ");
if ((cheName[y]==car2.carName) && (car2.zaiHuo > 0)) System.out.print(cheName[y]+" ");
if ((cheName[y]==car3.carName) && (car3.zaiHuo > 0)) System.out.print(cheName[y]+" ");
if ((cheName[y]==car4.carName) && (car4.zaiHuo > 0)) System.out.print(cheName[y]+" ");
if ((cheName[y]==car5.carName) && (car5.zaiHuo > 0)) System.out.print(cheName[y]+" ");
if ((cheName[y]==car6.carName) && (car6.zaiHuo > 0)) System.out.print(cheName[y]+" ");
}
System.out.println(" 共载货:" + zongZaiHuo + "吨");
System.out.println("租车总价格:" + zongTianShu * zongZuJin + "元");
}
}
}
父类
public class Car { String carName; int zuJin; int zaiRen; int zaiHuo; int bianHao;}
子类
public class AoDi extends Car { String carName="奥迪A4"; int zuJin=500; int zaiRen=4; int zaiHuo=0; int bianHao=1; }
public class JinLong extends Car { String carName="金龙"; int zuJin=800; int zaiRen=20; int zaiHuo=0; int bianHao=4; }
public class MaZiDa extends Car { String carName="马自达6"; int zuJin=400; int zaiRen=4; int zaiHuo=0; int bianHao=2;}
public class SongHuaJiang extends Car { String carName="松花江"; int zuJin=400; int zaiRen=0; int zaiHuo=4; int bianHao=5;}
public class YiWeiKe extends Car { String carName="依维柯"; int zuJin=1000; int zaiRen=0; int zaiHuo=20; int bianHao=6;}