(一)学习总结
1.学习使用思维导图对Java面向对象编程的知识点(封装、继承和多态)进行总结。
2.阅读下面程序,分析是否能编译通过?如果不能,说明原因。应该如何修改?程序的运行结果是什么?
class Grandparent {
public Grandparent() {
System.out.println("GrandParent Created.");
}
public Grandparent(String string) {
System.out.println("GrandParent Created.String:" + string);
}
}
class Parent extends Grandparent {
public Parent() {
System.out.println("Parent Created");
super("Hello.Grandparent.");
}
}
class Child extends Parent {
public Child() {
System.out.println("Child Created");
}
}
public class Test{
public static void main(String args[]) {
Child c = new Child();
}
}
不能编译通过。
错误信息:
修改方法:
与this()类似,super()只能放在构造首行要放在应该将Parent类中super("Hello.Grandparent.");放在该构造方法的第一句。
修改后运行结果:
GrandParent Created.String:Hello.Grandparent.//父类的父类有参构造
Parent Created//父类的构造
Child Created//子类构造
在创建子类的对象时,jvm会首先执行父类的构造方法,然后再执行子类的构造方法,如果是多级继承,会先执行最顶级父类的构造方法,然后依次执行各级个子类的构造方法。
为什么子类的构造方法在运行之前,必须调用父 类的构造方法?能不能反过来?
子类对象实例化默认调用父类中的构造方法,就好像没有父亲就没有孩子,实例化子类时(new)必须要初始化父类中的属性,但是不能反过来,父类实例化自己的对象时,其并不知道谁是自己的子类,并不知道子类有什么属性。
3 . 阅读下面程序,分析程序中存在哪些错误,说明原因,应如何改正?正确程序的运行结果是什么?
class Animal{
void shout(){
System.out.println("动物叫!");
}
}
class Dog extends Animal{
public void shout(){
System.out.println("汪汪......!");
}
public void sleep() {
System.out.println("狗狗睡觉......");
}
}
public class Test{
public static void main(String args[]) {
Animal animal = new Dog();
animal.shout();
animal.sleep();//此处有错1
Dog dog = animal;//此处有错2
dog.sleep();
Animal animal2 = new Animal();
dog = (Dog)animal2;//错3
dog.shout();
}
}
- 错1:对象发生了向上转型关系,所调用的方法一定是被子类复写的方法,所以应该在父类中加上sleep()方法.
- 错2:需要强制转化Dog dog = (Dog)animal;
- 错3:ClassCastException类转化异常.此处无语法错误,但是父类不知道谁是自己的子类
应改为:
Animal animal2 = new Dog();
dog = (Dog) animal2;
dog.shout();
4.运行下列程序
class Person {
private String name ;
private int age ;
public Person(String name,int age){
this.name = name ;
this.age = age ;
}
}
public class Test{
public static void main(String args[]){
Person per = new Person("张三",20) ;
System.out.println(per);
System.out.println(per.toString()) ;
}
}
(1)程序的运行结果如下,说明什么问题?
Person@166afb3
Person@166afb3
- 加不加toString()方法都随机输出了一些地址信息,对象的输出都调用了object中toString()方法
- System.out的println方法只能在控制台输出字符串,而Person实例是一个内存中的对象
其中object中方法内容为:
public String toString() {
return getClass().getName() + "@" + Integer.toHexString(hashCode());
}
(2)那么,程序的运行结果到底是什么呢?利用eclipse打开println(per)方法的源码,查看该方法中又调用了哪些方法,能否解释本例的运行结果?参考教材P229
经查看println()方法源码如下
public void println(Object x) {
String s = String.valueOf(x);//如果x为空返回空,不为空返回调用toString的内容
synchronized (this) {
print(s);
newLine();
}
}
实际上String是Object的子类
valueOf(Object x)如果x为空返回空,不为空返回调用toString的内容
(3)在Person类中增加如下方法
public String toString(){
return "姓名:" + this.name + ",年龄:" + this.age ;
}
重新运行程序,程序的执行结果是什么?说明什么问题?
姓名:张三,年龄:20
姓名:张三,年龄:20
子类Person类覆写了Object类中toString()方法
5.其他需要总结的内容。
java保留两位小数4种方法
1 import java.math.BigDecimal;
2 import java.text.DecimalFormat;
3 import java.text.NumberFormat;
4 public class format {
5 double f = 111231.5585;
6 public void m1() {
7 BigDecimal bg = new BigDecimal(f);
8 double f1 = bg.setScale(2, BigDecimal.ROUND_HALF_UP).doubleValue();
9 System.out.println(f1);
10 }
11 /**
12 * DecimalFormat转换最简便
13 */
14 public void m2() {
15 DecimalFormat df = new DecimalFormat("#.00");
16 System.out.println(df.format(f));
17 }
18 /**
19 * String.format打印最简便
20 */
21 public void m3() {
22 System.out.println(String.format("%.2f", f));
23 }
24 public void m4() {
25 NumberFormat nf = NumberFormat.getNumberInstance();
26 nf.setMaximumFractionDigits(2);
27 System.out.println(nf.format(f));
28 }
29 public static void main(String[] args) {
30 format f = new format();
31 f.m1();
32 f.m2();
33 f.m3();
34 f.m4();
35 }
36 }
(二)实验总结
实验内容:
1.定义员工类,具有姓名、年龄、性别属性,并具有构造方法和显示数据方法。定义管理层类,继承员工类,有自己的属性职务和年薪。定义职员类,继承员工类,并有自己的属性所属部门和月薪。定义一个测试类,进行测试。画出类图。
工具:ProcessOn
参考资料:UML 类图
2.按照下面要求完成类的设计
(1)设计一个平面图形抽象类(提供求该类对象周长和面积的方法)和一个立体图形抽象类(提供求该类对象表面积和体积的方法)
(2)设计球类、圆柱类,圆锥类、矩形类、三角形类、圆类,分别继承平面图形抽象类和立体图形抽象类。
(3)建立测试类,进行测试。画出类图。
3.. 参考类图,重构下面的实例,分析和理解多态的含义和用途
(1)某动物园有一饲养员小李,每天需要给他所负责饲养的一只狮子、五只猴子和十只鸽子喂食。 请用一个程序来模拟他喂食的过程。
问题总结
第一次重构之前,Animal类从未创建过对象,Animal类只需要充当模板;每种动物都要写一个feed方法,造成代码结构臃肿
第二次重构之后,Animal类为抽象类,各种动物依据它的格式创建对象;Feeder类中各种动物的feed方法合并为一个feedAniaml方法,通过向上转型,将各种动物类转换为Aniaml类,每种动物都使用feedAniaml方法
第三次重构之后,feedAniaml方法接受一个Aniaml类的对象数组,当一种动物有多个对象时,feedAniaml方法就可以接受该对象数组批量执行,只需调用一次feedAniaml方法,就可以喂一类所有动物