201621123031 《Java程序设计》第4周学习总结
Week04-面向对象设计与继承
- 本周学习总结
1.1 写出你认为本周学习中比较重要的知识点关键词
关键词:继承
、覆盖
、多态
、抽象
1.2 尝试使用思维导图将这些关键词组织起来。
1.3 可选:使用常规方法总结其他上课内容。
这里总结一下关于注释
:
在代码中添加注释可以很好的提高程序的可读性。在Java中主要有三种注释方法
- 单行注释
表示从符”//”开始直到换行为止的内容都将成为注释。
语法:
//注释内容
- 多行注释
表示符号”/”到符号”/”之间的所有内容都会成为注释。
语法:
/*
注释内容1
注释内容2
*/
- 文档注释
表示符号”/**””*/”之间的内容均为文档注释的内容。它一般用于类的声明,成员方法的声明,成员变量声明之前。
语法:
/**
文档注释
*/
某些声明
- 书面作业
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()
方法,当然,虽然上面说过了,但还是再说一遍,要记得判空!
- 码云及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 |