1. 本章学习总结
1.1 尝试使用思维导图总结有关多态与接口的知识点。
参考资料:
百度脑图
XMind
2. 书面作业
Q1.代码阅读:Child压缩包内源代码
1.1 com.parent包中Child.java文件能否编译通过?哪句会出现错误?试改正该错误。并分析输出结果。
不能编译通过,Child类中的
getParenti()
方法中System.out.println(i);
会出现错误,如下图所示:
父类Parent类中i的属性是private,只对所在的类可见,对子类是不可见的,应把父类中的i属性改为protected。
输出结果如下图所示:
package com.parent;
class Parent{
protected int i=1;
protected int j=2;
protected int geti(){
return i;
}
public void getj(){
System.out.println(j);
}
}
public class Child extends Parent{
public static void main(String[] args){
Parent p = new Parent();
Child c = new Child();
c.getParenti();
c.getParentj();
Other.showParentj(p);
}
public void getParenti(){
System.out.println(i);//继承父类,输出父类的i,运行结果为1
}
public void getParentj(){
System.out.println(super.j);//用关键字super(如果子类与父类的某个方法的 名称相同、参数相同,可用super调用父类的方法),继承父类的j,运行结果为2
System.out.println(j);//继承父类,输出父类的j,运行结果为2
System.out.println(geti());//继承父类geti()方法,运行结果为1
System.out.println(super.geti());//用super关键字,调用父类的方法,运行结果为1
}
}
class Other{
public static void showParentj(Parent p)//showParentj()方法是定义为static,直接用类名调用即可
{
System.out.println(p.j);//调用Parent类的j,运行结果为2
System.out.println(p.geti());//调用Parent类的i,运行结果为1
}
}
1.2 另外一个包中的OutOfParentPackage.java,能否编译通过?提示什么错误?分析原因。如何更改才能使之正常编译?
不能编译通过,Parent类前面没有标明任何修饰符,默认对本包可见,所以
showParentj()
只能在本包内被调用,而OutOfParentPackage类不在Parent包内,不能调用showParentj()
,所以在Parent类前加上public
;同时protected关键字对本包和所有子类可见,所以Parent类中的protected int j=2;
和protected int geti()
对其他包是不可见的,所以把j属性改为public,geti()方法的修饰符改为public即可。
更改后代码如下:
package com.parent;
public class Parent{
protected int i=1;
public int j=2;
public int geti(){
return i;
}
public void getj(){
System.out.println(j);
}
}
1.3 回答:如果为了访问到protected修饰的属性或方法应该怎么办?
为了访问protected修饰的属性或方法有两种方法,(1)该类为父类,其所有子类都可以访问(即使不在同一个包);(2)同包的其他类可以访问该类的protected属性和方法
Q2.abstract进阶:阅读GuessGame抽象类的设计与使用源代码
2.1 Guess改造前代码很简单,而改造后的代码使用了抽象类、抽象方法看起来很复杂,那这样的改造到底有什么好处呢?
改造前,未使用抽象类,只能在控制台输出;改造后,使用了抽象类,可以在控制台,也可以使用对话框图形界面等输入,不仅局限于控制台。
2.2 如果想将该游戏改造成图形界面,应该进行一些什么操作?
设计一个图形界面的类去继承Guess类, 可以用
java.awt.*
包或javax.swing.*;
来设计图形界面的类。
2.3 结合该例子,你觉得什么时候应该使用abstract
?
抽象类:用abstract关键字来修饰;抽象类必须被继承;抽象类不能被直接实例化。当该类中有方法需要具体问题具体分析,无法确定下来的时候,可以暂且用abstract将一些已有的方法声明确定下来。在本例子当中,
public abstract void print(String text);
public abstract void println(String text);
public abstract int nextInt();
,就是因为不知道会以何种具体的方式进行输入输出,所以暂且用abstract
来达到可以运用多种方式输入输出的目的。
2.4 重要:在这个例子中,变化的是什么,不变的是什么?尝试结合abstract、继承等概念进行说明。
变化的是利用abstract抽象类声明,使其可以以多种方式进行输入输出,并且用ConsoleGame类继承GuessGame类,使其与控制台绑定,可用控制台进行输入输出。
不变的是游戏本身的功能玩法没有进行改变,同样是随机生成数字0~9进行猜测。
Q3.Comparable与Comparator
3.1 描述Comparable接口的用途。为什么某个类实现了Comparable接口就可以直接使用Arrays.sort对其进行排序?
Comparable接口的用途:Comparable让两个对象可以进行比较。
Arrays有一方法public static void sort(Object[] a)
其功能是对指定对象数组按升序进行排序, 数组中的所有元素都必须实现 Comparable 接口。
3.2 有了Comparable接口为什么还需要Comparator接口呢?
首先,我们需要明确两者的区别。Comparable接口是定义在类内部的,是一个排序接口,表明了该类支持排序,它的比较功能和String完全一样,可以随时随地的拿来比较大小,可以把它看成是一个“内部比较器”;Comparator接口是定义在类外部的,是一个比较器接口,可以看成“外部比较器”。如果我们若需要对某个类的进行排序,而该类本身不支持排序(即没有实现Comparable接口);那么,我们可以建立一个“该类的比较器”来进行排序。这时,Comparator接口就派上用场了,这个“比较器”只需要实现Comparator接口即可实现排序。
Q4.面向接口案例分析
阅读Case-StudentDao.zip案例
4.1 画出类关系图,描述每个类与接口的作用。
Student类:定义学生姓名属性和toString()方法返回学生信息。
StudentDao接口:定义三个抽象方法:writeStudent(Student student)写入学生数据、public Student readStudent(String name);读取学生数据和public void diplayAllStudent()显示所有学生信息。
StudentDaoArrayImpl类:用数组students[]来存放学生信息,具体实现接口的三个抽象方法:writeStudent(Student student)写入学生数据、public Student readStudent(String name);读取学生数据和public void diplayAllStudent()显示所有学生信息。
StudenDaoListImpl类:用动态数组students来存放学生信息,具体实现接口的三个抽象方法:public Student readStudent(String name);读取学生数据和public void diplayAllStudent()显示所有学生信息。来存放学生信息,具体实现接口的三个抽象方法。
4.2 StudenDaoListImpl与StudentDaoArrayImpl有何区别?
StudenDaoListImpl是用数组存放学生信息;
StudentDaoArrayImpl是用ArrayList动态数组来存放学生信息。
Q5.什么是面向接口编程?面向接口编程的好处是什么?
结合题目3与4中的Test.java的代码讨论分析。不要百度原封不动照搬!
题目3中运用到了Comparable接口和Comparator接口,可以非常方便地利用
Arrays.sort()
对我们所定义的类进行比较排序,提高了程序的灵活性。题目4运用了StudentDao接口,在其接口中定义了三个抽象类方法,在几个类中都有这三个方法,体现了has-a的关系,但是各自都有不相同的地方,对于某些具体方法需要改变的时候,只要接口及接口的功能不变,则不需要对其进行更改,使代码的维护性大大得到了提高。同时,面向接口编程,使得整个代码更加具有结构性。
Q6.结对编程:面向对象设计(大作业2-非常重要)
内容:使用Java代码完成上周做的面向对象设计大作业,需要有初步界面。实现的功能尽量简单,少而精,只包含必要的功能,不要追求高大全。
写出:类图(尽量精简,不用太多子类,两个即可)、系统常用功能描述、关键代码与界面
形式: 两人依托码云合作完成。请在这里贴出你们的学号、姓名与任务分工。
注意: 再过几次课要讲Java图形界面编程,到时候要将该系统升级为图形界面。系统的业务逻辑部分应该变化不大,变化大的是输入与输出部分。所以编码的时候,输入(Scanner)与输出(System.out)的代码,请不要将其与某个业务处理函数绑死。
选做加分: 给出两人在码云上同一项目的提交记录截图,额外加分。注:两个人在码云上新建一个项目。
参考资料:
结对编程参考资料
可以使用Processon画图
6.1
董美凤 | 杨雪莹 | 项目地址 |
---|---|---|
http://www.cnblogs.com/dongmf/ | http://www.cnblogs.com/yangxy/ | http://git.oschina.net/yangxueying/shopping |
6.2 常用功能描述框架图
6.3 关键代码
class Clothes extends Product{
private String color;
private String size;
private String style;
public Clothes(String name, double price, int num, String color, String size, String style) {
super(name, price, num);
this.color = color;
this.size = size;
this.style = style;
}
public String getColor() {
return color;
}
public void setColor(String color) {
this.color = color;
}
public String getSize() {
return size;
}
public void setSize(String size) {
this.size = size;
}
public String getStyle() {
return style;
}
public void setStyle(String style) {
this.style = style;
}
@Override
public String toString() {
return "名称:" + name + ",价格:" + price + ",颜色:" + color + ",尺码:" + size
+ ",材质:" + style +",库存:" + num;
}
}
import java.util.ArrayList;
class Cart extends Product{
private int nums;
public Cart(String name, double price, int num, int nums) {
super(name, price, num);
this.nums = nums;
}
@Override
public String toString() {
return "[购物车 商品名称:" + name + ",价格:" + price + ", 购买数量" + nums+"] ";
}
ArrayList <Cart> carts = new ArrayList<Cart>();
public boolean writeCart(Cart cart) {
carts.add(cart);
return true;
}
public int getNums() {
return nums;
}
public void setNums(int nums) {
this.nums = nums;
}
public double sumprice()
{
double sum=0.0;
for(Cart e:carts)
{
sum+=e.getPrice()*e.getNums();
}
return sum;
}
}
6.4 运行界面
3. PTA实验总结及码云上代码提交记录
3.1本周Commit历史截图
在码云的项目中,依次选择“统计-Commits历史-设置时间段”,然后搜索并截图,如下图所示
3.2 实验总结
实验碰到的问题、思考、收获与解决方案
实验5-1:刚开始接触接口,敲代码时有一点懵。
在Comparable接口的PersonSortable类中加入compareTo(PersonSortable o)
方法时,最好用@Override
覆盖Object类中原来的compareTo(Object o)
方法,以达到我们想要对某些属性进行比较的需要。
以下是自己编写compareTo(PersonSortable o)
方法的具体代码:
public int compareTo(PersonSortable o){
int x=this.name.compareTo(o.name);
if(x!=0)
return x;
else
return this.age-o.age;
}
实验5-2:这道题是对名字和年龄进行排序,这时Comparable接口已经不能满足需求了,这时就运用到了Comparator接口。以下是NameComparator类实现Comparator接口的代码,AgeComparator类与其相似。
class NameComparator implements Comparator<PersonSortable2>{
@Override
public int compare(PersonSortable2 o1, PersonSortable2 o2) {
// TODO Auto-generated method stub
return new String (o1.getName()).compareTo(o2.getName());
}
}