201621123031 《Java程序设计》第4周学习总结

Week04-面向对象设计与继承


  1. 本周学习总结

1.1 写出你认为本周学习中比较重要的知识点关键词

关键词:继承覆盖多态抽象

1.2 尝试使用思维导图将这些关键词组织起来。

1.3 可选:使用常规方法总结其他上课内容。

这里总结一下关于注释
在代码中添加注释可以很好的提高程序的可读性。在Java中主要有三种注释方法

  • 单行注释
    表示从符”//”开始直到换行为止的内容都将成为注释。
    语法:
//注释内容
  • 多行注释
    表示符号”/”到符号”/”之间的所有内容都会成为注释。
    语法:
/*
注释内容1
注释内容2
*/
  • 文档注释
    表示符号”/**””*/”之间的内容均为文档注释的内容。它一般用于类的声明,成员方法的声明,成员变量声明之前。
    语法:
/**
文档注释
*/
某些声明
  1. 书面作业

1. 面向对象设计(大作业1-该作业将作为以后其他作业的基础,请务必完成)

1.1 讲故事:用50字以上讲一个你在网上商城购物或者在班级博客进行学习的故事。使用Markdown的加粗标记把关键名词标注出来,使用语句块标记把里面关键的动词标注出来。

整天抱着电脑瞎晃荡的我终于把我的鼠标砸坏了,为了方便,决定去某购物网站上买一只回来。我先输入了账号密码登录了系统进入商城界面,然后搜索鼠标
因为懒得挑选,于是选择了综合排序最高的一个点进去。进去查看了一下产品的一些颜色尺寸型号的参数,觉得还行,于是决定购买。
在分类中选择了想要的型号,然后为了以防万一,决定再买一个备用的,于是数量选择2,添加到了购物车
最后切换到购物车,选中商品,进行结算

1.2 找出系统中包含的类及其属性、方法,类与类之间的关系,并绘制相应类图。注意:不一定非要体现继承关系。只要能将系统描述完整即可。一开始设计的时候不要考虑太多,不要出现太多的类,仅出现必要的类,完成最小功能即可。对商城购物系统建议只做购物车模块。

用户类[Users]:存放用户信息,并可以查看和修改。
属性:

private String Name		//用户名
private String Account	//账号
private String Password	//密码
static ArrayList<User> user = new ArrayList()	//用于存储每个用户的信息

方法:

public void change(String message)	//用于修改用户信息

注册类(需继承用户类)[Register]:用于用户的注册。
属性:

String Account		//账号
String Password	//密码
StringName		//用户名

方法:

public void setAccoAndCode()	//用于设置账号密码
public boolean ExistYoN()			//判断所设置的账号是否有重复
public void addUser(String name,String account,String password)		//用于添加新注册的用户  

登录类[Login]:用于用户的登录。
属性:

String Account		//账号
String Password	//密码
String IdentifyCode		//验证码

方法:

public boolean currentYoN()	//判断账号密码是否匹配 

商品类[Goods]:用于存放商城中的商品信息,可对其中信息进行修改和查询。
属性:

String GoodsName;	//商品名
double Price;		//单价
static ArrayList<Goods> goods = new ArrayList();//用于存储商品信息

方法:

public Goods(String GoodsName,double Price)	//导入商品信息

搜索类[Search]:继承于商品类,根据关键字查询商品类中的信息。
属性:

String GoodsName;	//商品名

方法:

public void fit()	//用于匹配和关键字有关的信息并输出

购物车类[ShoppingCar]:用于存放挑选好的商品。
属性:

String GoodsName;	//商品名
int num;	//商品数量
static double AllSum;	//总价
double Price;	//价钱
static ArrayList<ShoppingCar> info = new ArrayList();	//存储购物车内商品信息

方法:

public ShoppingCar(String GoodsName,int num)	//用于将商品加入购物车
public static void RemoveGoods(String GoodsName)	//删除购物车内的商品
public static void AddGoods(String GoodsName,int addnum)	//添加商品数量
public static void Display()	//展示购物车信息

1.3 使用Java代码实现这个系统(不必很完善)。粘贴运行截图与关键代码。将来要在这个基础上逐渐完善、扩展成一个完整的面向对象的系统。合作完成:2-3人一个小组。

可选:团队协作可使用git与码云。在码云中新建项目。所有队员都应在码云中该项目下均应有有提交记录。截图你们的提交记录。
这次的合作对象是廖文姑和贾海涛,因为时间关系,很多类的都没有实现,只是简单地写了几个类和里面的方法,后续再慢慢补充。码云的话,试了一下,好像失败了,不知道怎么让多个人同时上传到一个项目中,等周末再试试吧。主要代码截图及测试如下:
1.商品信息类

2.购物车类

3.用于测试的商品信息

4.测试数据

2. ManagerTest.zip代码分析(继承、多态)

分析ManagerTest.zip中的代码,回答几个问题:

2.1 简述文件中表现出的继承关系。哪些是共有方法,哪些是子类特有的属性和方法?

在Java中使用了关键字extends来标识两个类之间的继承关系。

class Manager extends Employee
{
   ……………
}

由以上代码可以看出Manager类继承自Employee类。
共有方法:

public String getName()
public double getSalary()
public Date getHireDay()
public void raiseSalary(double byPercent)

子类特有属性和方法:

private double bonus;  //特有属性
public void setBonus(double b)		//特有方法

2.2 文件第26行e.getSalary(),到底是调用Manager类的还是Employee类的getSalary方法?

编译器在运行的时候会根据e的类型来区分是哪一个类的方法,其被称为动态绑定,因此在代码中,staff[0]调用的是Manager类中的方法,staff[1]和staff[2]调用的是Employee类中的方法。

2.3 Manager类的构造函数使用super调用父类的构造函数实现了代码复用,这样有什么好处?为什么不把父类构造函数中的相关代码复制粘贴到Manager的构造函数中,这样看起来不是更直观吗?

调用父类构造函数更加方便,减少了代码量。
看起来直观但是却增加了代码量,让代码看起来显得不简洁,直接用super调用来实现代码复用更加方便且更加容易阅读。

3. Object类中的toString与equals

3.1 编写Fruit类,属性String name。如果覆盖其toString()方法,其父类中的toString方法的代码就没有了吗?编写Fruit的toString()方法,在该方法中要将调用父类的toString方法得到的字符串与自己特有的属性name拼接起来,怎么编写?(使用代码展示)

若子类的方法覆盖了父类的方法,父类的方法不会没有,还是可以通过super关键字来调用。
Fruit类的toString()方法:

package week04;

class Fruit{
	private String name;
	public Fruit(String name){
		this.name = name;
	}
	public String toString() {
		return name+"-"+super.toString();
	}
}

public class FFruit {

	public static void main(String[] args) {
		Fruit fruit = new Fruit("Apple");
		System.out.println(fruit.toString());
	}
}

运行结果:

3.2 为Fruit类编写equals方法覆盖父类相应方法,功能为当两个Fruit对象name相同时(忽略大小写),返回true。(使用代码证明你自己覆盖的equals方法是正确的)

对于equals()方法的重写的要求,有以下几点

1、自反性:对于任何非空引用x,x.equals(x)应该返回true。
2、对称性:对于任何引用x和y,如果x.equals(y)返回true,那么y.equals(x)也应该返回true。
3、传递性:对于任何引用x、y和z,如果x.equals(y)返回true,y.equals(z)返回true,那么x.equals(z)也应该返回true。
4、一致性:如果x和y引用的对象没有发生变化,那么反复调用x.equals(y)应该返回同样的结果。
5、非空性:对于任意非空引用x,x.equals(null)应该返回false。

在查询equals()方法重写的过程中,还发现了一个需要和该方法一起被重写的方法hashCode()方法。

当equals此方法被重写时,通常有必要重写 hashCode 方法,以维护 hashCode 方法的常规协定,该协定声明相等对象必须具有相等的哈希码。如下:
(1)当obj1.equals(obj2)为true时,obj1.hashCode() == obj2.hashCode()必须为true 
(2)当obj1.hashCode() == obj2.hashCode()为false时,obj1.equals(obj2)必须为false
hashcode是用于散列数据的快速存取,如利用HashSet/HashMap/Hashtable类来存储数据时,都是根据存储对象的hashcode值来进行判断是否相同的。
这样如果我们对一个对象重写了euqals,意思是只要对象的成员变量值都相等那么euqals就等于true,但不重写hashcode,那么我们再new一个新的对象,当原对象.equals(新对象)等于true时,两者的hashcode却是不一样的,由此将产生了理解的不一致。

参考链接,写的很棒呀!
代码如下:

package week04;

class Fruit{
	private String name;
	public Fruit(String name){
		this.name = name;
	}
	public String toString() {
		return name+"-"+super.toString();
	}
	@Override
	public int hashCode() {
		final int prime = 31;
		int result = 1;
		result = prime * result + ((name == null) ? 0 : name.hashCode());
		return result;
	}
	@Override
	public boolean equals(Object obj) {
		if (this == obj)
			return true;
		if (obj == null)
			return false;
		if (getClass() != obj.getClass())
			return false;
		Fruit other = (Fruit) obj;
		if (name == null) {
			if (other.name != null)
				return false;
		} else if (!name.equalsIgnoreCase(other.name))
			return false;
		return true;
	}
	
}

public class FFruit {

	public static void main(String[] args) {
		Fruit fruit1 = new Fruit("Apple");
		Fruit fruit2 = new Fruit("orange");
		Fruit fruit3 = new Fruit("Orange");
		System.out.println(fruit1.equals(fruit2));
		System.out.println(fruit2.equals(fruit3));
	}
}

运行结果:

3.3 完成3.2后,使用ArrayList fruitList存储多个fruit,添加时要求如果要添加的fruit对象在fruitList中已存在就不添加,不存在就添加。编写相关测试代码。并分析ArrayList的contains代码与equals方法有何关系?

提示:直接使用ArrayList的contains方法实现判断对象是否存在。

public class FFruit {

	public static void main(String[] args) {
		ArrayList<Fruit> fruitList = new ArrayList<Fruit>();
		Fruit[] fruit = new Fruit[5];
		fruit[0] = new Fruit("banana");
		fruit[1] = new Fruit("Banana");
		fruit[2] = new Fruit("apple");
		fruit[3] = new Fruit("Apple");
		fruit[4] = new Fruit("orange");
		
		for(int i=0;i<fruit.length;i++){
			if(fruitList.contains(fruit[i])==false)
				fruitList.add(fruit[i]);
		}
		for(int i=0;i<fruitList.size();i++)
			System.out.println(fruitList.get(i));
	}
}

运行结果:

由此可见在我们改写了equals()方法后,ArrayList的contains也开始忽略了大小写。所以contains方法也用到了equals()的方法来进行比较。

4.3 程序填空、函数题(1-3)。

  • 程序填空:这题就是按照题目要求来就OK了,注意要使用this来调用有参构造函数。
  • 函数题1:这题要求覆盖toString()方法,使用super关键字来调用父类的对象,但是一直产生内部错误orz……
  • 函数题2:这题只要按题目步骤一步一步来就可以了,要注意的是最后数组的输出要是倒序的。
  • 函数题3:这题要重写equals()方法,当然,虽然上面说过了,但还是再说一遍,要记得判空!
  1. 码云及PTA

题目集:[jmu-Java-03-面向对象1-基础-封装继承]

3.1. 码云代码提交记录

在码云的项目中,依次选择“统计-Commits历史-设置时间段”, 然后搜索并截图

3.2 截图PTA题集完成情况图

本周完成(形状-继承,覆盖)
需要有两张图(1. 排名。2.PTA提交列表)

3.3 统计本周完成的代码量

这周代码量偏少了,国庆没好好打代码……

周次 总代码量 新增代码量 总文件数 新增文件数
2 607 607 15 15
3 1642 1035 33 18
5 2044 402 42 9
posted @ 2017-10-13 21:42  一只水饺  阅读(516)  评论(16编辑  收藏  举报