教材学习内容总结
学习目标
- 理解封装、继承、多态的关系
- 理解抽象类与接口的区别
- 掌握S.O.L.I.D原则
- 了解模式和设计模式
- 能正确覆盖方法
- 了解垃圾回收机制
- 掌握Object类
- 掌握enum
教材第六章内容总结——继承与多态
继承
- 继承:继承基本上就是避免多个类间重复定义共同行为。
我理解的就是:在编写程序的过程中可能会出现部分代码重复的现象,把重复的部分单独定义为一类(父类),在其他代码中(子类)直接继承,这样子类可直接使用父类的方法,子类的对象也可以使用父类的方法,同时避免了大段代码的重复出现的问题。
关键字 extend
public class SwordsMan extends Role{
//SwordsMan会继承Role的行为,并对Role的行为进行扩充,仅在SwardsMan中使用新定义的行为
public void fight(){
System.out.println("挥剑攻击");
}
}
上述代码表示,类SwordsMan继承了类Role,类SwordsMan扩充了类Role的行为,也就是类Role中有定义的程序代码,类SwordsMan因为继承而都拥有了。
需要注意的是:
在Java中,子类只能继承一个父类,也就是说只能有一个爸爸。
- IS-A:因为子类继承了父类,所以子类是一种父类
- 运用“IS-A”关系判断语法的正确性:方式:判断等号右边是否为等号左边的子类,以下两例均无法通过编译:
SwordsMan swordsMan = new Role();
Magician magician = new Rile();
//Role 为 SwordsMan 和 Magician 的父类,右边类不是左边类的子类,所以编译不通过
Role role1 = new SwordsMan();
//SwordsMan 是一种 Role ,这条语句可以通过编译
SwordsMan swordsMan = role1;
//role1 为 Role 声明的名称,编译程序认为 Role 不一定是一种 SwordsMan,编译失败
- 住嘴语法:在等号右边加括号声明即可通过编译,但执行不一定能通过,例如:
Role role2 = new Magician();
SwordsMan swordsMan = (SwordsMan) role2;
//在 role1 前加括号声明关系,编译能通过,但执行时不一定能通过,要根据参考的对象实际类型判断,本例让魔法师扮演剑士,执行时会出现错误
- 多态:使用单一接口操作多种类型的对象。我的理解是:就子类和父类来讲,一个父类可以有很多子类,在程序中就可以通过一个父类操作很多它的子类,这样比较方便,同时使程序具有更高的可维护性
- 封装是继承的基础,继承是多态的基础
- 重新定义:在继承父类之后,定义与父类中相同的方法部署,但各子类中执行的内容不同。父类定义了方法,但是没有内容,子类可以重新定义父类中的实际行为。因父类定义了方
法,所以可以运用多态,传入子类中定义的方法,实现一些功能操作。 - 注意:在重新定义父类中某个方法时,子类必须撰写与父类方法中相同的签署,就是定义的方法的名字要完全一样
- 注意:重新定义方法时,对于父类中的方法权限,只能扩大但不能缩小,若原来成员
public
子类中重新定义时不可为private
或protected
- 标注:@Override 如果在子类中某个方法前标注@Override,表示要求编译程序检查,该方法是不是真的重新定义了父类中的某个方法,如果不是的话,就会引发编译错误
抽象方法、抽象类:
•抽象方法:如果某方法区块中真的没有任何程序代码操作,可以使用 abstract
标示该方法为抽象方法,该方法不用撰写{}区块,直接“;”结束即可,例如:
public abstract void fight();
- 抽象类:Java中规定内含抽象方法的类,一定要在 class 前标示 abstract ,表示一个定义不完整的抽象类
- 类中若有方法没有操作,且标示为 abstract,表示这个类定义不完整,定义不完整的类不能用来生成实例,就好像设计图不完整就不能拿来生产成品
- 如果尝试用抽象类创建实例,会引发编译错误
- 子类如果想继承抽象类,对于抽象方法有两种做法:(如果两种做法都没有实施,就会引发编译错误)
1. 继续标示该方法为 abstract(该子类因此也是个抽象类,必须在class前标示 abstract)
2. 操作抽象方法
继承语法细节
protected:被声明为 protected 的成员,相同包中的类可以直接存取,不同包中的类可以在继承后的子类中直接存取,存取时可用“this”,具体的权限关键字与范围如下:
关键字 | 类内部 | 相同包类 | 不同包类 |
---|---|---|---|
public | 可存取 | 可存取 | 可存取 |
protected | 可存取 | 可存取 | 子类可存取 |
无 | 可存取 | 可存取 | 不可存取 |
private | 可存取 | 不可存取 | 不可存取 |
super
在Java中,如果想取得父类中的方法定义,可以在调用方法前加上“super”关键字,例如:
public String toString(){
return "剑士" + super.toString();
- 注意:可以使用 super 关键字调用的父类方法,不能定义为 private(因为这就限定只能在类内使用)
- 构造函数
- 创建子类实例后,会先执行父类构造函数定义的流程,再执行子类构造函数定义的流程
- 构造函数可重载
- 如果子类构造函数中没有指定执行父类中哪个构造函数,默认会调用父类中无参数的构造函数,例如:
class Some {
Some() {
System.out.println("调用Some()");
}
}
class Other extends Some{
Other(){
//子类构造函数中没有指定执行父类中哪个构造函数,则调用父类中无参数的构造函数
System.out.println("调用Other()");
}
}
//先执行Some中的流程,再执行Other中的流程,最后的结果是:先显示“调用Some()”,再显示“调用Other()”
- 若父类中自行定义了构造函数,在父类中就不会加入任何构造函数了,此时若子类没有指定调用父类哪个函数,就会编译失败
- final
- 类加 final:如果class前加了final,则表示这个类是最后一个了,不会再有子类 → 这个类不能被继承(例:String 在定义时就已经限定为 final 了)
- 方法加 final:定义方法时也可以限定该方法为 final,表示最后一次定义方法 → 子类不可以重新定义 final 方法
- java.lang.Object
- java.lang.Object 是所有类的顶层父类,这代表了 Object 上定义的方法,所有对象都继承下来了,只要不是被定义成 final 方法,都可以重新定义
- java中所有对象,一定“是一种”Object
- 如果定义类时没有使用 extends 关键字指定继承任何类,那一定是继承了 java.lang.Object,以下两段代码是等价的
public class Some{}
public class Some extends Object{}
- toString()
- toString()的作用:传入对象
- toString()是 Object 上定义的方法
- toString()调用方法:以下两段代码是等价的,调用的时候选取第二种方式调用即可
Syetem.out.println(swordsMan.toString());
System.out.println(swordsMan);
equals()
- equlas()是 Object 类有定义的方法,程序代码如下:
public boolean equals(Object obj){
return(this == obj);
}
- 若果没有重新定义 equals(),使用 equals() 方法时,作用等同于 ==
- 要比较实质相等性,必须自行重新定义equals()
instanceof
- 用途:判断对象是否由某个类创建,左操作数是对象,右操作数是类,也就是判断 instanceof 左边的对象是否由右边的类创建
- 并非只有左操作数对象为右操作数类直接的实例才能返回 true,只要左操作数类型是右操作数类型的子类型,instanceof 也返回 true
- 垃圾收集
- 垃圾:如果程序执行流程中已无法再使用某个对象,该对象就只是徒豪内存的垃圾
- 垃圾收集机制:GC
- 执行流程中,无法通过变量参考的对象,就是GC认定的垃圾对象
- GC在进行回收对象前,会调用对象的 finalize() 方法(这是 Object 上定义的方法),如果在对象被回收前有些事情想做,可以重新定义 finalize() 方法
教材第七章内容总结——接口与多态
何谓接口
接口:书上没有明确地给出接口的定义,我理解的接口就是一段程序中可能有很多类会有共同的行为,但这些类本身并不具有任何关系,如果使用继承的话程序架构会不合理,所以使用统一的接口表示这些类具有共同的行为
interface
可定义行为,例如:
public interface Swimmer{
public abstract void swim();
}
implements
类要操作接口,必须使用 implements 关键字,例如:
public abstract class Fish implements Swimmer{
}
操作某接口时,对接口中定义的方法有两种处理方式:
1. 操作接口中定义的方法
2. 再度将该方法标示为 abstract继承与操作接口的区别:继承会有“是一种”关系,操作接口表示“拥有行为”,但不会有“是一种”的关系
多态语法合法性判断:判断等号右边是不是拥有等号左边的行为,即等号右边的对象是不是操作了等号左边的接口,以下这个例子就可以通过编译:
Swimer swimmer = new Shark();//因为 Fish 操作了 Swimmer 接口,即 Fish 拥有了 Swimmer 行为。Shark 继承 Fish ,所以Shark 拥有 Swimmer 行为
- 扮演语法:会操作 Swimmer 接口的不一定继承Fish,加上扮演语法即可通过编译,例如:
Swimmer swimmer = new Shark();
Fish fish = (Fish)swimmer;
以下的例子将会抛出 ClassCastException 错误:
Swimmer swimmer = new Human();//将 swimmer 参考到 Human实例
Shark shark = (Shark)swimmer;//让 swimmer 扮演鲨鱼
已经将 swimmer 参考到 Human 实例了,再让他扮演鲨鱼就会出现错误
- 解决需求变更问题:我们可以定义一些拥有不同行为的接口,定义一些类作为父类,当增加新的需求的时候,可以操作接口,代表拥有接口的行为,也可以继承父类,代表这个新的类“是一种”父类,原有的程序无需修改,只针对新的需求撰写程序即可
- 在java中,类可以操作两个以上的接口,也就是拥有两种以上的行为
- 在java中,接口可以继承自另一个接口,也就是继承父接口行为,再在子接口中额外定义行为,例如:
public interface Diver extends Swimmer{ //接口 Diver 继承了接口 Swimmer
public abstract void dive();
}
接口语法细节
- 在java中,可使用 interface 定义抽象的 行为与外观,如接口中的方法可声明为 public abstract ,例如:
public interface Swimmer{
public abstract void swim();
}
- 接口中的方法没有操作的时候,一定要是公开抽象,可以省略 public abstract ,例如:
public interface Swimmer{
void swim(); //此处默认是 public abstract
}
- 由于默认一定是 public ,在类操作接口的时候也要撰写 public
interface Action{
void execute();
}
class Some implements Action{
void execute(){
//Some 类在操作 execute() 方法时,没有撰写 public ,因此默认为是包权限,这等于是将 Action 中的 public 方法缩小为包权限,所以编译失败
//将 Some 类的 execut() 设为public 才可通过编译
System.out.println("做一些服务");
}
}
在 interface 中,只能定义 public static final
的枚举常数 ,例如:
public interface Action{
public static final int STOP = 0;
}
如下撰写程序时,编译程序会自动展开为public static final
public interface Action{
int STOP = 0;
}
- 在接口中枚举常数,一定要使用 = 指定值,否则编译错误
- 接口可以继承别的接口,可以同时继承两个以上的接口,使用 extends 关键字 ,代表了继承父接口的行为,例:
interface Action{
void executes();
}
// 定义 Acton 为父接口
interface Some extends Action{
void doSome();
}
interface Other extends Action{
void doOther();
}
// Some 和 Other 接口继承 Action 接口
public class Service implements Some,Other{
// Service 继承 Some 和 Other 接口
@Override
public void execute(){
System.out.println("execute()");
}
// Service 重新定义 execute() 方法
@Override
public void doSome{
System.out.println("doSome()");
}
@Override
public void doOther{
System.out.println("doOther()");
}
// Service 重新定义 doSome 和 doOther 方法
}
匿名内部类
- 某些子类或接口操作类只使用一次,不需要为这些类定义名称,这时可使用匿名内部类
- 语法:
new 父类()|接口(){
//类本体操作
};
- JDK8 之前,若要在匿名内部类中存取局部变量,则该局部变量必须是 final ,否则会发生编译错误
final int[] numbers = {10,20};
enum
enum 可定义枚举常数,但实际上 enum 定义了特殊的类,继承自 java。lang.Enum ,编译过后会产生 Action.class 文件,可用这个 Action 声明类型,例如:
public class Game {
public static void main(String[] args){
play(Action.RIGHT);
play(Action.UP);
//只能传入 Action 实例
}
public static void play(Action action){
// action 参数声明为 Action 类型,所以只接受 Action 的实例,这样就不需要必须使用 default 检查,编译程序在编译时期会进行类型检查
switch(action){
case STOP:
out.println("播放停止动画");
break;
case RIGHT:
out.println("播放向右动画");
break;
case LEFT:
out.println("播放向左动画");
break;
case UP:
out.println("播放向上动画");
break;
case DOWN:
out.println("播放向下动画");
break;
}
}
}
- enum中列举的常熟,实际上是 public static final ,且为枚举类型实例,因为构造函数权限设定为 private ,只有类中才可以实例化
教材学习中的问题和解决过程
问题一
Java中的继承和C语言中调用函数有何区别?
Answer:
Java语言与C语言本身就是两种不同思维的语言。C语言是面向过程的语言,就是我们要干一件事情,那么一步一步逐次来做,直到完成,其中可能就要调用一些函数,被调用的函数也是过程执>行的。但是Java程序的开发思路是面向对象的,它是有一个类,这个类有属性,有方法。我们要干一件事的时候,就通过该类所定义的或者说是具体化的对象(Object)来调用自己的方法来完成,这里所说的方法,也可以理解为是一个函数(类似于C语言中的),但是只不过这个方法现在是属于这个类的,属于这个对象的,是有“主人”的,只有通过它的“主人”才能调用它。而面向对象的Java有一个特点就是继承,当某一个类A继承了类B那么类A就可以使用类B的方法了,因为A继承了B,同样A的对象也可以使用B的方法了。
问题二
继承类和接口有何区别?
Answer:
- 不同的修饰符修饰(interface),(extends)
- 在面向对象编程中可以有多继承!但是只支持接口的多继承,不支持“继承”的多继承。而继承在java中具有单根性,子类只能继承一个父类
- 在接口中只能定义全局常量,和抽象方法,而在继承中可以定义属性方法,变量,常量等。
- 某个接口被类实现时,在类中一定要实现接口中的抽象方法,而继承想调用那个方法就调用那个方法,毫无压力。
举个有趣的例子:
当使用继承的时候,主要是为了不必重新开发,并且在不必了解实现细节的情况下拥有了父类我所需要的特征。
但是很多时候,一个子类并不需要父类的所有特征,它可能只是需要其中的某些特征,但是由于通过继承,父类所有的特征都有了,需要的和不需要的特征同时具备了。而那些子类实际上不需要用到的,有时候甚至是极力避免使用的特征也可以随便使用,这就是继承的副作用。特别是允许多重继承的Java语言中,很容易引起不容易发现的错误。所以在java的语言中,会创造出各种规定来限制子类使用父类中的某些方法。
如果狗的主人只是希望狗能爬比较低的树,但是不希望它继承尾巴可以倒挂在树上,像猴子那样可以飞檐走壁,以免主人管不住它。那么狗的主人肯定不会要一只猴子继承的狗。
设计模式更多的强调面向接口。猴子有两个接口,一个是爬树,一个是尾巴倒挂。我现在只需要我的狗爬树,但是不要它尾巴倒挂,那么我只要我的狗实现爬树的接口就行了。同时不会带来像继承猴子来带来的尾巴倒挂的副作用。这就是接口的好处。
Java技术发展也有好多年了,一个很明显的趋势就是继承的使用越来越少,而接口的使用越来越广泛了。其实只要稍微比较一下JDK里面那些最早就有的类库和最近才加进去的类库,就可以很明显的感觉到java技术领域的编程风格的变迁,由大量的继承到几乎无处不用的面向接口编程。
接口不是替代继承。比如说我现在就是要我的动物去爬树,我根本就不需要知道到底是狗去爬树还是猴子。我提取出“能爬树”的动物去爬。这个能爬树的动物既可以是猴子,也可以是狗。这样不是很灵活吗?
狗(爬树,咬人)
猴子(爬树,尾巴倒挂)
如果我既要爬树也要咬人,那么我当然可以选狗,也可以创建一个接口(爬树咬人),然后让狗实现(爬树咬人)接口。
代码调试中的问题和解决过程
学完了这两张,OOP的基本内容就完成了,可以自己动手写一些小程序了~
不同于书上有原始代码,自己从零开始一行一行敲还是很容易出现问题~
为了将这几章的知识融入代码增强理解,我写了如下程序:
/**
* Created by XiaYihua on 2017/1/26.
*/
abstract class Person{
private String name;
private int age;
private String sex;
public void say(){System.out.println("Hello!");};
public void setName(String name){
this.name = name;
}
public String getName(){return name;}
public void setAge(int age){
this.age = age;
}
public int getAge(){return age;}
public void setSex(String sex){
this.sex = sex;
}
public String getSex(){return sex;}
public abstract void Address();
}
class Student extends Person{
private int score;
private String address;
public void setScore(int score){
this.score = score;
}
public int getScore(){return score;}
public void say(){
super.say();
System.out.println("My score:");
}
public void setAddress(String address){
this.address = address;
}
public String getAddress(){
return address;
}
public void Address(){
System.out.println("Address: " + getAddress());
}
}
interface brain{
public abstract void learn();
}
interface mouth{
public abstract void teach();
}
class Teacher implements brain, mouth{
String grade;
int age;
@Override
public void learn() {
System.out.println("I'm learning!");
}
@Override
public void teach() {
System.out.println("I'm teaching!");
}
public void setAge(int age){
this.age = age;
}
public int getAge(){return age;}
public void setGrade(String grade){
this.grade = grade;
}
public String getGrade(){return grade;}
}
class People{
public static void main(String[] args){
Student P1 = new Student();
P1.say();
P1.setName("Sherlock");
P1.setAge(20);
P1.setSex("Male");
P1.setScore(100);
P1.setAddress("221B");
System.out.println("Name:" + P1.getName() + "\nAge:" + P1.getAge() + "\nSex:" + P1.getSex()+"Score:" + P1.getScore());
P1.Address();
Teacher T1 = new Teacher();
T1.learn();
T1.teach();
T1.setAge(35);
T1.setGrade("Senior 1");
System.out.println("Age:" + T1.getAge() + "\nGrade:" + T1.getGrade());
}
}
这个例子融合了抽象类、继承、接口等知识点,虽然没有用到全部细节,但是却让我对这几章的核心知识有了深层次的掌握。
在写的过程中有过小错误:
上面这个错误就是println、print、printf的区别问题,idea联想的第一个就是println,所以在写的时候没有发现格式化输出和println的冲突。
三者的详细区别已经在第三周的博客中总结完成了。
代码托管截图
https://git.oschina.net/xyh20145226/java-besti-is-2015-2016-2-20145226-2/tree/master/src?dir=1&filepath=src&oid=521ad5dd7b419812d6b342bb970ba6c33cd6c7b4&sha=45fff2cdd5eba8ebe469473c18d6c37d17c87529
寒假已全部完成。
学习进度条
代码行数(新增/累积) | 博客量(新增/累积) | 学习时间(新增/累积) | 重要成长 | |
---|---|---|---|---|
目标 | 5000行 | 30篇 | 400小时 | |
第二周 | 200/400 | 1/3 | 12/52 | |
预备作业 | 0/400 | 1/4 | 15/67 | |
第三周 | 500/900 | 1/5 | 10/77 | |
第四周 | 500/1500 | 1/6 | 15/92 |
参考资料
- Java学习笔记(第8版)
- 《Java学习笔记(第8版)》学习指导