20145218 《Java程序设计》第四周学习总结
教材学习内容总结
继承
继承共同行为
- 继承基本上就是避免多个类间重复定义共同行为。
- 继承的三个好处:减少代码冗余;维护变得简单;扩展变得容易。
- 构造方法不能被继承。
- 代码如下:
public class RPG1
{
public static void main (String[] args)
{
demoSwordsMan();
demoMagician();
}
static void demoSwordsMan()
{
SwordsMan1 swordsMan1 = new SwordsMan1();
swordsMan1.setName("Justin");
swordsMan1.setLevel(1);
swordsMan1.setBlood(200);
System.out.printf("剑士 : (%s, %d, %d)%n",swordsMan1.getName(),
swordsMan1.getLevel(),swordsMan1.getBlood());
}
static void demoMagician()
{
Magician1 magician = new Magician1();
magician.setName("Moinca");
magician.setLevel(1);
magician.setBlood(100);
System.out.printf("魔法师 :(%s ,%d ,%d)%n",magician.getName(),
magician.getLevel(),magician.getBlood());
}
}
- 运行结果截图如下:
多态与is-a
- is-a代表的是类之间的继承关系。中文称为“是一种”关系。
- 使用是一种原则,就可以判断何时编译成功,何时编译失败。
- 编译程序会检查父子类间的“是一种”关系.
- 定义继承需要注意:类与类之间要有是一种(is-a)关系,比如SwordsMan继承了Role,所以SwordsMan is a Role。
- 代码如下:
public class RPG2
{
public static void main(String[] args)
{
SwordsMan swordsMan = new SwordsMan();
swordsMan.setName("Justin");
swordsMan.setLevel(1);
swordsMan.setBlood(200);
Magician magician = new Magician();
magician.setName("Monica");
magician.setLevel(1);
magician.setBlood(100);
showBlood(swordsMan);
showBlood(magician);
}
static void showBlood(Role role)
{
System.out.printf("%s 血量 %d%n",role.getName(),role.getBlood());
}
}
- 运行结果截图如下:
重新定义行为
- 重新定义:在继承父类之后,定义与父类中相同的方法部署,但实行内容不同。
- 代码如下:
public class RPG3
{
public static void main(String[] args)
{
SwordsMan3 swordsMan3 = new SwordsMan3();
swordsMan3.setName("Justin");
swordsMan3.setLevel(1);
swordsMan3.setBlood(200);
Magician3 magician3 = new Magician3();
magician3.setName("Monica");
magician3.setLevel(1);
magician3.setBlood(100);
drawFight(swordsMan3);
drawFight(magician3);
}
static void drawFight(Role3 role)
{
System.out.print(role.getName());
role.fight3();
}
}
- 运行结果截图如下:
继承语法细节
protected成员
- protected与private基本相似,只有在继承时有较大的区别。继承的类可以访问protected成员,但是不能访问private成员。
- 被声明为protected的成员,相同包中的类可以直接存取,不同包中的类可以在继承后的子类直接存取。
- 权限关键字与范围
关键字 类内部 相同包类 不同包类
public 可存取 可存取 可存取
protected 可存取 可存取 子类可存取
private 可存取 不可存取 不可存取 - 练习代码如下:
public abstract class Role5
{
protected String name;
protected int level;
protected int blood;
public int getBlood()
{
return blood;
}
public void setBlood(int blood)
{
this.blood = blood;
}
public int getLevel()
{
return level;
}
public void setLevel(int level)
{
this.level = level;
}
public String getName()
{
return name;
}
public void setName(String name)
{
this.name = name;
}
}
public class Magician5 extends Role5
{
public void fight()
{
System.out.println("魔法攻击");
}
public void cure()
{
System.out.println("魔法治疗");
}
public String toString()
{
return String.format("魔法师 (%s, %d, %d)", this.name, this.level, this.blood);
}
}
public class SwordsMan5 extends Role5
{
public void fight()
{
System.out.println("挥剑");
}
public String toString()
{
return String.format("剑士 (%s, %d %d)", this.name, this.level, this.blood);
}
}
重新定义的细节
- 对于父类中的方法权限,只能扩大但不能缩小。若原来成员public,子类中重新定义时不可为private或protected。
- 练习代码如下:
public abstract class Role6
{
public String toString()
{
return String.format("(%s, %d, %d)", this.name, this.level, this.blood);
}
}
再看构造函数
- 如果类有继承关系,在创建子类实例后,会先进行父类定义的初始流程,再进行子类中的初始流程,也就是创建子类实例后,会先执行父类构造函数定义的流程,再执行子类构造函数定义的流程。
- 构造函数可以重载,父类中可重载多个构造函数,如果子类构造函数中没有指定执行父类中哪个构造函数,默认会调用父类中无参数构造函数。
- this()与super()只能择一调用,而且一定要在构造函数第一行执行。
再看final关键字
- 若class前使用了final关键字定义,那么这个类不会有子类,不能被继承。
- 子类不可以重新定义final方法。
- final可以修饰类,方法,变量。
- final修饰的方法不可以被覆盖。
- final修饰的变量是一个常量,只能被赋值一次。
- 内部类只能访问被final修饰的局部变量。
java.lang.Object
- 子类只能继承一个父类,如果定义类时没有使用extends关键字指定继承任何类,那就是继承java.lang.Object。相当于public class Some Object{...}
- Java中所有对象,一定“是一种”Object。
- 任何类型的对象,都可以使用Object声明的名称来参考。
- 练习代码如下:
import java.util.Arrays;
public class ArrayList {
private Object[] list;
private int next;
public ArrayList(int capacity) {
list=new Object[capacity];
}
public ArrayList() {
this(16);
}
public void add(Object o) {
if(next==list.length) {
list=Arrays.copyOf(list, list.length*2);
}
list[next++]=o;
}
public Object get(int index) {
return list[index];
}
public int size() {
return next;
}
}
import java.util.Scanner;
import static java.lang.System.out;
public class Guest {
public static void main(String[] args) {
ArrayList names=new ArrayList();
collectNameTo(names);
out.println("访客名单:");
printUpperCase(names);
}
static void collectNameTo(ArrayList names) {
Scanner console=new Scanner(System.in);
while(true) {
out.print("访客名称:");
String name=console.nextLine();
if(name.equals("quit")) {
break;
}
names.add(name);
}
}
static void printUpperCase(ArrayList names) {
for(int i=0;i<names.size();i++) {
String name=(String) names.get(i);
out.println(name.toUpperCase());
}
}
}
- 代码运行结果如下:
关于垃圾收集
- 程序执行流程中无法通过变量参考的某个对象,就是徒耗内存的垃圾。
- jvm有垃圾收集机制(GC),收集到的垃圾对象所占据的内存空间,会被垃圾收集器释放。
再看抽象类
- 练习代码如下:
public abstract class GuessGame {
public void go() {
int number=(int)(Math.random()*10);
int guess;
do {
print("输入数字:");
guess=nextInt();
}while(guess!=number);
println("猜中了");
}
public void println(String text) {
print(text+"\n");
}
public abstract void print(String text);
public abstract int nextInt();
}
import java.util.Scanner;
public class ConsoleGame extends GuessGame {
private Scanner scanner=new Scanner(System.in);
@Override
public void print(String text) {
System.out.print(text);
}
@Override
public void println(String text) {
System.out.println(text);
}
@Override
public int nextInt() {
return scanner.nextInt();
}
}
public class Guess {
public static void main(String[] args){
GuessGame game=new ConsoleGame();
game.go();
}
}
- 代码运行结果如下:
接口
接口定义行为
- 接口可以用于定义行为但不定义操作。
- 对于“定义行为”,可以使用interface关键字定义,接口中的方法不能操作,直接标示为abstract,而且一定是
- 类要操作接口,必须使用implements关键字。操作某接口时,对接口中定义的方法有两种处理方式,一是操作接口中定义的方法,二是再度将该方法标示为abstract。
- 继承会有“是一种”关系,接口操作则表示“拥有行为”,但不会有“是一种”关系。
- 练习代码如下:
public abstract class Fish implements Swimmer {
protected String name;
public Fish(String name){
this.name = name;
}
public String getName()
{
return name;
}
@Override
public abstract void swim();
}
public class Human implements Swimmer {
private String name;
public Human(String name){
this.name=name;
}
public String getName()
{
return name;
}
@Override
public void swim()
{
System.out.printf("人类 %s 游泳 %n",name);
}
}
行为的多态
- 接口多态语法的判断,方式是“右边是不是拥有左边的行为”或者“右边对象是不是操作了左边接口”。
- 练习代码如下:
public class Ocean{
public static void main(String[] args)
{
doSwim(new Anemonefish("尼莫"));
doSwim(new Shark("兰尼"));
doSwim(new Human("贾斯汀"));
doSwim(new Submarine("黄色一号"));
}
static void doSwim(Swimmer swimmer){
swimmer.swim();
}
}
- 运行结果如下:
解决需求变化
- 有弹性、具可维护性的程序:如果增加新的需求,原有的程序无需修改,只需针对新需求撰写程序。
- 练习代码如下:
public interface Flyer
{
public abstract void fly();
}
public class FlyingFish extends Fish implements Flyer{
public FlyingFish(String name){
super(name);
}
@Override
public void swim(){
System.out.println("飞鱼游泳");
}
@Override
public void fly(){
System.out.println("飞鱼会飞");
}
}
public class Airplane implements Flyer{
protected String name;
public Airplane(String name){
this.name=name;
}
@Override
public void fly(){
System.out.printf("飞机 %s 在飞%n",name);
}
}
public class Ocean2{
public static void main(String[] args)
{
doSwim(new Anemonefish("尼莫"));
doSwim(new Shark("兰尼"));
doSwim(new Human("贾斯汀"));
doSwim(new Submarine("黄色一号"));
doSwim(new Seaplane("空军零号"));
doSwim(new FlyingFish("甚平"));
}
static void doSwim(Swimmer swimmer)
{
swimmer.swim();
}
}
接口语法细节
接口的默认
- 在java中,可使用interface来定义抽象的行为与外观,如接口中的方法可声明为public abstract。
- 类可以操作两个以上的类,也就是拥有两种以上的行为。类可以同时继承某个类,并操作某些接口。
- 接口可以继承另一个接口,也就是继承父接口行为,再在子接口中额外定义行为。
- 练习代码如下:
public interface Action{
public static final int STOP=0;
public static final int RIGHT=1;
public static final int LEFT=2;
public static final int UP=3;
public static final int DOWN=4;
}
import static java.lang.System.out;
public class Game{
public static void main(String[] args){
play(Action.RIGHT);
play(Action.UP);
}
public static void play(int action){
switch(action){
case Action.STOP:
out.println("bofangtingzhidonghua");
break;
case Action.RIGHT:
out.println("bofangxiangyoudonghua");
break;
case Action.LEFT:
out.println("bofangxiangzuodonghua");
break;
case Action.UP:
out.println("bofangxiangshangdonghua");
break;
case Action.DOWN:
out.println("bofangxiangxiadonghua");
break;
default:
out.println("buzhichicidongzuo");
}
}
}
- 运行结果如下:
匿名内部类
- 临时继承某个类或操作某个接口并建立实例,可使用匿名内部类。
- 匿名内部类语法为:new 父类()|接口(){// 类本体操作 };
- 练习代码如下:
public class Client{
public final String ip;
public final String name;
public Client(String ip,String name){
this.ip=ip;
this.name=name;
}
}
public class ClientEvent{
private Client client;
public ClientEvent(Client client){
this.client=client;
}
public String getName(){
return client.name;
}
public String getIp(){
return client.ip;
}
}
public interface ClientListener{
void clientAdded(ClientEvent event);
void clientRemoved(ClientEvent event);
}
import java.util.ArrayList;
public class ClientQueue{
private ArrayList clients=new ArrayList();
private ArrayList listeners=new ArrayList();
public void addClientListener(ClientListener listener){
listeners.add(listener);
}
public void add(Client client){
clients.add(client);
ClientEvent event=new ClientEvent(client);
for(int i=0;i<listeners.size();i++){
ClientListener listener=(ClientListener) listener.get(i);
listener.client.Added(event);
}
}
public void remove(Client client){
client.remove(client);
ClientEvent event=new ClientEvent(client);
for(int i=0;i<listeners.size();i++){
ClientListener listener=(ClientListener) listener.get(i);
listener.client.Removed(event);
}
}
}
public class Multichat{
public static void main(String[] args){
Client c1=new Client("127.0.0.1","Caterpillar");
Client c1=new Client("192.168.0.2","Justin");
ClientQueue queue=new ClientQueue();
queue.addClientListener(new ClientListener(){
@Override
public void clientAdded(ClientEvent event){
System.out.printf("%s cong %s 联机%n",
event.getGame(),event.getIp());
}
@Override
public void clientRemoved(ClientEvent event){
System.out.printf("%s cong %s 脱机%n",
event.getGame(),event.getIp());
}
});
queue.add(c1);
queue.add(c2);
queue.remove(c1);
queue.remove(c1);
}
}
- 运行结果如下:
使用enum枚举常数
- 练习代码如下:
public enum Action2{
STOP,RIGHT,LEFT,UP,DOWN
}
import static java.lang.System.out;
public class Game2{
public static void main(String[] args){
play(Action.RIGHT);
play(Action.UP);
}
public static void play(Action action){
switch(action){
case STOP:
out.println("bofangtingzhidonghua");
break;
case RIGHT:
out.println("bofangxiangyoudonghua");
break;
case LEFT:
out.println("bofangxiangzuodonghua");
break;
case UP:
out.println("bofangxiangshangdonghua");
break;
case DOWN:
out.println("bofangxiangxiadonghua");
break;
}
}
}
- 运行结果如下:
代码调试中的问题和解决过程
课本上的代码只有两个是可以在cmd中运行的,其余的只可以在idea中运行。
本周代码托管截图
感悟
Java的学习已经入第四周,之前总是拖到周末去看视频,写博客,而第四周显然不可以再拖沓了,课本深度的增加,代码难度的加强,都是督促我们学习的动力。与前三周不同的是,这周的代码不是独立的,而是几组代码组合在一起才能运行。而且在cmd中可以运行的少之又少,只得在之前不是很熟悉的IDE中运行。但学习任务难度的加大也不是只提供压力,而是转变为学习的动力,在第四周我每天都会抽出时间来学习Java,也熟练掌握了之前没使用过的IDE,并且学会了用wc统计代码行数,这周的学习可谓是获益匪浅。
学习进度条
代码行数(新增/累积) | 博客量(新增/累积) | 学习时间(新增/累积) | 重要成长 | |
---|---|---|---|---|
目标 | 5000行 | 30篇 | 400小时 | |
第一周 | 200/200 | 2/2 | 20/20 | |
第二周 | 300/500 | 1/3 | 18/40 | |
第三周 | 400/1000 | 1/4 | 25/60 | |
第四周 | 1292/1300 | 1/5 | 30/90 |