这世上有三样东西是别人抢不走的:一是吃进胃里的食物,二是藏在心中的梦想,三是读进大脑的书。

《Java技术》第四次作业计科1501赵健宇

(一)学习总结

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方法,就可以喂一类所有动物

(三)代码托管

点击此处进入码云

posted @ 2017-04-14 21:18  进击的小白V  阅读(437)  评论(2编辑  收藏  举报
感谢观看。转载复制请注明源处~!