1. 本章学习总结
1.1 尝试使用思维导图总结有关继承的知识点。
1.2. 使用常规方法总结其他上课内容。
①instenceof运算符:可通过它判断父类引用对象实例的实际类型,且在父类转化成子类时应用instenceof进行检查。
②protectd:同包的其他类可以访问父类的protectd的属性和方法,且当希望子类可以直接访问父类时,用protect进行声明。 ③子类具有父类的所有属性和方法。2. 书面作业
Q1.注释的应用
使用类的注释与方法的注释为前面编写的类与方法进行注释,并在Eclipse中查看。(截图)
Q2. 面向对象设计(大作业1,非常重要)
Q2.1 将在网上商城购物或者在班级博客进行学习这一过程,描述成一个故事。(不得少于50字,参考QQ群中PPT的范例)
某日想买水杯,于是进入X宝首页,输入账号密码后登陆完毕。接着在搜索栏输入水杯,进入某店铺(实力爆款!)后,选好水杯的样式后确认加入购物车。在购物车中确认要够买的商品及数量后,选择或填写收货地址,通过支付平台完成支付。
Q2.2 通过这个故事我们能发现谁在用这个系统,系统中包含的类及其属性方法,类与类之间的关系。尝试找到这些类与属性,并使用思维导图描述类、属性、方法及类与类之间的关系。
Q3.ManagerTest.zip代码分析
Q3.1 在本例中哪里体现了使用继承实现代码复用?回答时要具体到哪个方法、哪个属性。
public Manager(String n, double s, int year, int month, int day) { super(n, s, year, month, day); bonus = 0; } public double getSalary() { double baseSalary = super.getSalary(); return baseSalary + bonus; }
使用了Manager类中的getSalary方法。
属性体现为name,salary,hireday。Q3.2 Employee类及其子类Manager都有getSalary方法,那怎么区分这两个方法呢?
父类程序:
public double getSalary() { return salary; }
子类程序:
public double getSalary() { double baseSalary = super.getSalary(); return baseSalary + bonus; }
对比可以看出,子类会用super关键字来表示。故用super关键字来区分这两种方法。
Q3.3 文件第26行e.getSalary(),到底是调用Manager类的getSalary方法还是Employee类的getSalary方法。
程序如下:
public class ManagerTest { public static void main(String[] args) { // construct a Manager object Manager boss = new Manager("Carl Cracker", 80000, 1987, 12, 15); boss.setBonus(5000); Employee[] staff = new Employee[3]; // fill the staff array with Manager and Employee objects staff[0] = boss; staff[1] = new Employee("Harry Hacker", 50000, 1989, 10, 1); staff[2] = new Employee("Tommy Tester", 40000, 1990, 3, 15) // print out information about all Employee objects for (Employee e : staff) System.out.println("name=" + e.getName() + ",salary=" + e.getSalary()); }}
其中有Employee[] staff = new Employee[3];这样一行。意味着staff被创建为Employee类。所以26行中的e.getSalary()调用的自然是Employee类的getSalary()。若e引用的是Manager对象时,情况就不同了。
Q3.4 Manager类的构造函数使用super调用父类的构造函数实现了代码复用,你觉得这样的有什么好处?为什么不把父类构造函数中的相关代码复制粘贴到Manager的构造函数中,这样看起来不是更直观吗?
①直接来说,就是大大减少代码量。一个优秀的程序必然在其精简性上有所体现。且避免了将时间浪费在敲重复代码上的弊端。
②不进行复制粘贴,其原因是在程序员眼中,代码复用已然是自然而然的方法,并不存在所谓“读起来不直观”这种概念。且为了使程序更直观就让程序变得冗长重复是不值得的。(其实就想说复制粘贴显得很Low)Q4. Object类
Q4.1 编写一个Fruit类及属性String name,如没有extends自任何类。使用System.out.println(new Fruit());是调用Fruit的什么方法呢?该方法的代码是从哪来的?尝试分析这些代码实现了什么功能?
代码如下:
class Fruit { private String name;}public class jojo { public static void main(String[] args) { System.out.println(new Fruit()); }}
系统调用了toString()方法来显示,该方法来自Object类。因为此时系统默认Fruit继承自Object类。代码功能是将输入对象的值转换成了字符串,输出且换行。
Q4.2 如果为Fruit类添加了toString()方法,那么使用System.out.println(new Fruit());调用了新增的toString方法。那么其父类中的toString方法的代码就没有了吗?如果同时想要复用其父类的toString方法,要怎么操作?(使用代码演示)
添加toString()方法后,程序代码如下:
class Fruit { private String name; public String toString() { return "The World"; } }public class jojo { public static void main(String[] args) { System.out.println(new Fruit()); }}
此时运行结果为The World
添加并调用了toString方法后,父类中的toString方法代码依然存在。 若同时想复用父类的toString方法,用super关键字即可。修改后有如下代码:class Fruit { private String name; public String toString() { System.out.println(super.toString()); return "The World"; } }public class jojo { public static void main(String[] args) { System.out.println(new Fruit()); }}
此时运行结果为
java2.Fruit@15db9742 The WorldQ4.3 Fruit类还继承了Object类的equals方法。尝试分析其功能?自己编写一个equals方法覆盖父类的相应方法,功能为当两个Fruit对象name相同时(忽略大小写),那么返回true。(使用代码证明你自己覆盖的equals方法是正确的)
Object类的equals方法的功能是比较两个对象引用是否指向同一个对象。
这里引用Eclipse自动生成的eqauls代码。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.equals(other.name)) return false; return true; }
Q4.4 在4.3的基础上使用ArrayList fruitList存储多个fruit,要求如果fruitList中已有的fruit就不再添加,没有的就添加进去。请编写相关测试代码。并分析ArrayList的contatins方法是如何实现其功能的?
contains代码如下:
public boolean contains(Object o){ return indexOf(o) >= 0;}public int indexOf(Object o) { if (o == null) { for (int i = 0; i < size; i++) if (elementData[i]==null) { return i; } else { for (int i = 0; i < size; i++) if (o.equals(elementData[i])) return i; } return -1; }
该题不太会,大体觉得是indexOf()中使用了equals比较两个字符串是否相等。
Q5.代码阅读:PersonTest.java(abstract、多态)
Q5.1 画出类的继承关系
Q5.2 读懂main函数,将自己推测的出代码运行结果与真正运行结果进行比较。尝试分析原因
运行结果如下:
Manager [bonus=12000.3, toString()=Employee [salary=90000.1, toString()=Person [name=Clark, adress=GE, phonenumber=111, email=111@mail.com, age=10, gender=mail]]]Student [status=1, toString()=Person [name=wang, adress=110, phonenumber=15959, email=15959@163.com, age=18, gender=male]]Employee [salary=1000.0, toString()=Person [name=zhang, adress=136, phonenumber=1360, email=1360@mail.com, age=21, gender=female]]Programmer [allowance=50000.0, toString()=Employee [salary=100000.0, toString()=Person [name=Gates, adress=usa, phonenumber=911, email=911@com, age=59, gender=male]]]
student类和employee类继承person类。
manager类和programmer类继承employee类。 以各自类中的toString方法中的输出形式输出结果。输出顺序按照年龄升序。Q5.3子类中里面使用了super构造函数,作用是什么?如果将子类中的super构造函数去掉,行不行?
作用是调用父类的构造函数。
不行,如果去掉子类中的super构造函数,则编译器会默认调用父类Employee的无参构造函数,但父类中并没有这个无参构造函数,结果就会出现编译出错的情况。Q5.4 PersonTest.java中的代码哪里体现了多态?你觉得多态有什么好处?多态和继承有什么关系吗?
for (Person person : peoples) { System.out.println(person); }
以上程序段体现了多态。此处输出person,但person是一个person类(抽象类)。
多态的好处(引用自http://www.cnblogs.com/wl0000-03/p/5964881.html): (1)可替换性(substitutability)。多态对已存在代码具有可替换性。例如,多态对圆Circle类工作,对其他任何圆形几何体,如圆环,也同样工作。 (2)可扩充性(extensibility)。多态对代码具有可扩充性。增加新的子类不影响已存在类的多态性、继承性,以及其他特性的运行和操作。实际上新加子类更容易获得多态功能。例如,在实现了圆锥、半圆锥以及半球体的多态基础上,很容易增添球体类的多态性。 (3)接口性(interface-ability)。多态是超类通过方法签名,向子类提供了一个共同接口,由子类来完善或者覆盖它而实现的。图中超类Shape规定了两个实现多态的接口方,computeArea()以及computeVolume()。子类,如Circle和Sphere为了实现多态,完善或者覆盖这两个接口方法。 (4)灵活性(flexibility)。它在应用中体现了灵活多样的操作,提高了使用效率。 (5)简化性(simplicity)。多态简化对应用软件的代码编写和修改过程,尤其在处理大量对象的运算和操作时,这个特点尤为突出和重要。 多态与继承的关系(引用自http://wenda.so.com/q/1364416529060978?999): 继承,子类继承父类中所以的属性和方法,但是对于private的属相和方法,由于这个是父类的隐私,所以子类虽然是继承了,但是没有可以访问这些属性和方法的引用,所以相当于没有继承到。很多时候,可以理解为,没有继承。 多态:就是父类引用可以持有子类对象。这时候只能调用父类中的方法,而子类中特有方法是无法访问的,因为这个时候(编译时)你把他看作父类对象的原因,但是到了运行的时候,编译器就会发现这个父类引用中原来是一个子类的对像,所以如果父类和子类中有相同的方法时,调用的会是子类中的方法,而不是父类的。 可以这么说:编译时看父类,运行时看子类。3.使用码云管理Java代码
4.PTA实验总结
①运用到继承的知识,在子类中使用父类的属性和方法。学会通过super调用父类。
②掌握关于抽象类的相关知识,如运用到abstrat,一旦类包含了abstract方法则这个类必须声明为abstract,且抽象类不能被实例化。 ③toSting方法还需熟练运用。 ④总的来说,就如本周三的课堂测试一样,老师教的知识点确实覆盖在大多题目上,但是自己运用在编程上时就显得十分乏力。应该是练习不够多的缘故。