201521123003《Java程序设计》第5周学习总结

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());
            }
         }
原文地址:https://www.cnblogs.com/dongmf/p/6606664.html