1. 本周学习总结
1.1 尝试使用思维导图总结有关多态与接口的知识点。
2. 书面作业
作业参考文件下载
1. 代码阅读:Child压缩包内源代码
1.1 com.parent包中Child.java文件能否编译通过?哪句会出现错误?试改正该错误。并分析输出结果。
Child.java文件不能编译通过,上图中提示第13行i
出错,"Thread [main] (Suspended (The field Parent.i is not visible))",说明i在Parent类中是不可见的;会有以下方式修改该错误:
按照以往我们的习惯会选第二种方法,"create getter and setter for 'i'",这样文件就可以编译通过了。i在Parent类中无修饰,但默认是私有"private",外部类是无法访问的,get()和ger()方法可以对数据进行设置和获取。
我们可以尝试第一种方法,"change visibility of 'i' to 'protected'",文件同样可以通过。protected对于子女、朋友,也就是同包下的所有类和不同包的子类来说,就是public的,可以自由使用,没有任何限制,而对于其他的外部class,protected就变成private。所以Parent类中i用"protected"修饰后,对Child类来说i就是可见的,因为Child类继承Parent类。
输出结果是7个数字“1 2 2 1 1 2 1”,以下代码后的注释是该行代码的输出结果
c.getParenti();//1
c.getParentj();//2,2,1,1
Other.showParentj(p);//2,1
同时补上Other类的代码:
1.2 另外一个包中的OutOfParentPackage.java,能否编译通过?提示什么错误?分析原因。
不能编译通过,
提示"The type Parent is not visible","The field Parent.j is not visible","The method geti() from the type Parent is not visible",也就是说Parent类对OutOfParentPackage.java文件是不可见的,还有j和geti()是protected修饰的,也不对OutOfParentPackage开放。protected权限问题在上题中已解答。
以下是各种修饰符的权限比较:
1.3 回答:如果为了访问到protected修饰的属性或方法应该怎么办?
最直接也是有点麻烦的方法就是把protected修饰改成public修饰,但是会破坏原属性和方法的封装性。
2. abstract进阶:阅读GuessGame抽象类的设计与使用源代码
2.1 Guess改造前代码很简单,而改造后的代码使用了抽象类、抽象方法看起来很复杂,那这样的改造到底有什么好处呢?
2.2 如果想将该游戏改造成图形界面,应该进行一些什么操作?
2.3 结合该例子,你觉得什么时候应该使用abstract?
该例创建了一个公有抽象类GuessGame,并声明了公共抽象方法print()
,println()
,nextInt()
,供非抽象子类ConsoleGame类实现,子类ConsoleGame覆盖了父类抽象类GuessGame中的print()
,println()
,nextInt()
方法。
2.4 重要:在这个例子中,变化的是什么,不变的是什么?尝试结合abstract、继承等概念进行说明。
在这个例子中,变化的是对抽象类GuessGame中声明的方法的实现方式,不变的是声明的方法。先声明方法,后继承重写,主要是为了定义一个标准、规范。
3. Comparable与Comparator
3.1 描述Comparable接口的用途。为什么某个类实现了Comparable接口就可以直接使用Arrays.sort对其进行排序?
让类实现Comparable接口,并覆盖compareTo方法就可以实现对象的自动排序。Arrays有一方法public static void sort()
根据元素的自然顺序对指定对象数组按升序进行排序,数组中的所有元素都必须实现 Comparable 接口.所以某个类实现了Comparable接口就可以直接使用Arrays.sort对其进行排序。
3.2 有了Comparable接口为什么还需要Comparator接口呢?
public interface Comparator<T> {
int compare(T o1, T o2);
boolean equals(Object obj);
}
4. 面向接口案例分析
阅读Case-StudentDao.zip案例
4.1 画出类关系图,描述每个类与接口的作用。
StudentDaoArrayImpl类和StudenDaoListImpl类实现了StudentDao接口,覆盖了StudentDao接口中的writeStudent();readStudent();diplayAllStudent();
方法,让其在本类中有各自的实现。
4.2 StudenDaoListImpl与StudentDaoArrayImpl有何区别?
StudenDaoListImpl建立的是List<Student>
列表,而StudentDaoArrayImpl建立的是Student[]
数组,都是实现StudenDao接口,但各自的实现方法不同。
5. 什么是面向接口编程?面向接口编程的好处是什么?
结合题目3与4中的Test.java的代码讨论分析。
概念很抽象,现在我们针对Test.java代码进行分析:
01 package chpt06;
02
03 public class Test {
04
05 /**
06 * @param args
07 */
08 public static void main(String[] args) {
09 Student[] students = new Student[3];
10 students[0] = new Student("Tom");
11 students[1]= new Student("Jerry");
12 students[2] = new Student("Sophia");
13
14
15 StudentDao sdm = new StudentDaoArrayImpl(50);//使用数组实现
16 //StudentDao sdm = new StudenDaoListImpl();//使用列表实现
17
18 //往后台写数据,无需考虑后台是什么(到底是数据库、文件、数组、List)
19 //因为这里是面向StudentDao接口
20 System.out.println("===========写入学生========");
21 for(Student e:students){
22 if (!sdm.writeStudent(e)){
23 System.out.println("添加学生失败");
24 }else{
25 System.out.println("插入成功!!");
26 }
27 }
28 System.out.println("===========显示所有学生========");
29 sdm.diplayAllStudent();
30 System.out.println("===========查询学生========");
31 Student temp = sdm.readStudent("Tom") ;
32 if(temp == null){
33 System.out.println("查无此人");
34 }else{
35 System.out.println(temp);
36 }
37
38 }
39
40 }
第15行代码StudentDao sdm = new StudentDaoArrayImpl(50);
中,指向数组的变量sdm实现了StudentDao接口,可以调用接口中所有方法:
第22行代码if (!sdm.writeStudent(e)){}
,调用了StudentDao接口的writeStudent()
方法,
第29行代码sdm.diplayAllStudent();
,调用了StudentDao接口的diplayAllStudent()
方法,
第31行代码Student temp = sdm.readStudent("Tom") ;
,调用了StudentDao接口的readStudent()
方法。
若是把15行代码注释掉,而编译第16行代码StudentDao sdm = new StudenDaoListImpl();
,则是指向列表的变量sdm实现了StudentDao接口,同样可以调用接口中所有方法。
6. 结对编程:面向对象设计(大作业2-非常重要)
内容:使用Java代码完成上周做的面向对象设计大作业,需要有初步界面。实现的功能尽量简单,少而精,只包含必要的功能,不要追求高大全。
写出:类图(尽量精简,不用太多子类,两个即可)、系统常用功能描述、关键代码与界面
形式: 两人依托码云合作完成。请在这里贴出你们的学号、姓名与任务分工。
注意: 再过几次课要讲Java图形界面编程,到时候要将该系统升级为图形界面。系统的业务逻辑部分应该变化不大,变化大的是输入与输出部分。所以编码的时候,输入(Scanner)与输出(System.out)的代码,请不要将其与某个业务处理函数绑死。
选做加分: 给出两人在码云上同一项目的提交记录截图,额外加分。注:两个人在码云上新建一个项目。
参考资料:
结对编程参考资料
可以使用Processon画图
- 6.1
谭燕 | null | null |
- 6.2常用功能描述框架图
- 6.3
package Shopping;
public class Login {
private String name="customer";
private String secret="123456";
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public String getSecret() {
return secret;
}
public void setSecret(String secret) {
this.secret = secret;
}
public Login(String name, String secret) {
super();
this.name = name;
this.secret = secret;
}
@Override
public boolean equals(Object obj) {
if (this == obj)
return true;
if (obj == null)
return false;
if (getClass() != obj.getClass())
return false;
Login other = (Login) obj;
if (name == null) {
if (other.name != null)
return false;
} else if (!name.equals(other.name))
return false;
if (secret == null) {
if (other.secret != null)
return false;
} else if (!secret.equals(other.secret))
return false;
return true;
}
}
package Shopping;
public class Goods {
private String name;
private double price;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public double getPrice() {
return price;
}
public void setPrice(double price) {
this.price = price;
}
public Goods(String name, double price) {
super();
this.name = name;
this.price = price;
}
@Override
public boolean equals(Object obj) {
if (this == obj)
return true;
if (obj == null)
return false;
if (getClass() != obj.getClass())
return false;
Goods other = (Goods) obj;
if (name == null) {
if (other.name != null)
return false;
} else if (!name.equals(other.name))
return false;
if (Double.doubleToLongBits(price) != Double.doubleToLongBits(other.price))
return false;
return true;
}
@Override
public String toString() {
return name + "-" + price;
}
}
package Shopping;
public class Shopcart {
private Goods goods;
private static int count=0;
private double sum;
public int getCount() {
return count;
}
public void setCount(int count) {
Shopcart.count = count;
}
public double getSum() {
return sum;
}
public void setSum(double sum) {
this.sum = sum;
}
public Goods getGoods() {
return goods;
}
public void setGoods(Goods goods) {
this.goods = goods;
}
public Shopcart Add(Goods goods){
Shopcart goods_=new Shopcart(goods);
count++;
goods_.sum=goods.getPrice()*count;
return goods_;
}
public Shopcart(Goods goods) {
super();
this.goods = goods;
}
@Override
public String toString() {
return goods.toString() + "-" + count + "-" + sum;
}
}
package Shopping;
import java.util.ArrayList;
import java.util.Scanner;
import Shopping.Shopcart;
public class Main {
public static void main(String[] args) {
Scanner in = new Scanner(System.in);
Goods[] goods = new Goods[5];
goods[0] = new Goods("Java", 65.3);
goods[1] = new Goods("Math", 37.2);
goods[2] = new Goods("skirt", 80.0);
goods[3] = new Goods("jacket", 46.8);
goods[4] = new Goods("Ajax", 65.3);
System.out.println("请输入帐号和密码:");
Login account = new Login(in.next(), in.next());
Login obj = new Login("customer", "123456");
if (!account.equals(obj)) {
System.out.println("帐号或密码错误!");
return;
}
System.out.println("登入成功!");
ArrayList<Shopcart> list = new ArrayList<Shopcart>();
while (true) {
System.out.println("请输入要搜索的商品名:");
String name = in.next();
int i;
for (i = 0; i < goods.length; i++) {
if (name.equals(goods[i].getName())) {
System.out.println(goods[i]);
break;
}
}
if (i == goods.length) {
System.out.println("找不到该商品!");
return;
} else {
System.out.println("1.结算 2.添加至购物车");
int choice = in.nextInt();
if (choice == 1) {
System.out.printf("需支付 %.2f", goods[i].getPrice());
System.out.println();
} else if (choice == 2) {
Shopcart g = new Shopcart(goods[i]);
// if (!list.contains(g))
list.add(g.Add(goods[i]));
// else
// 找出list中的同名商品,数量count增加
System.out.println("查看购物车:");
for (Shopcart s : list) {
System.out.println(s);
}
System.out.println("1.结算 2.添加商品 3.退出");
int choice1 = in.nextInt();
switch (choice1) {
case '1': {
int sum = 0;
for (Shopcart s : list) {
sum += s.getSum();
}
System.out.printf("总共需支付 %.2f", sum);
System.out.println();
break;
}
case '2':
break;
case '3':
return;
}
}
}
}
}
}
- 6.4输出结果如下,很多细节还没做,有些地方会出差错:
3. 码云上代码提交记录及PTA实验总结
题目集:jmu-Java-04-面向对象2-进阶-多态接口内部类
3.1. 码云代码提交记录
在码云的项目中,依次选择“统计-Commits历史-设置时间段”, 然后搜索并截图
3.2. PTA实验
继续完成前面未完成的实验面向对象1-基础-封装继承
函数(选做:4-1)、编程(5-1, 5-2)
- 类实现Comparable接口时,格式为
implements Comparable<T>
,T为对象的引用。 - Comparable接口中唯一方法
compareTo()
,实现接口时要覆盖该方法,自定义比较方式。 - 列表用
Collections.sort(List);
实现排序,数组用Arrays.sort(array);
实现,前提是对象都实现了Comparable接口。 - 数值格式在
java.text.DecimalFormat
中,有方法format()
,返回的都是string型字符串。 - 比较器Comparator也是一个接口,与Comparable类似也有不同,有方法
compare(obj o1,obj o2);
对不支持Comparable的类控制其次序.