第六周&java实验报告四

一.实验目的
(1)掌握类的继承.
1、继承 是面向对象软件技术当中的一个概念,与多态、封装共为面向对象的三个基本特征。 继承可以使得子类具有父类的属性和方法或者重新定义、追加属性和方法等。
2、子类拥有父类的特征,而父类没有,父类更通用,子类更具体,(特征包括属性和方法,自身的特性,拥有父类没有的)
  2、使用extends继承父类,语句格式:class 子类名 extends 父类名{}
  3、父类中一般只定义一般属性和方法(这个一般可以理解为是子类共有的,这就是父类更通用,而子类拥有其他的,所以子类更具体)
  4、子类中通过super关键字来调用父构造方法
  5、在子类中可以继承父类得那些东西,哪些不可以继承
  父类中public,protected修饰的属性,方法可以继承,private修饰的属性和方法不能被继承
  6、规则: 创建子类对象的时候,首先调用的是父类的无参构造方法创建一个父类对象
  7、可以在子类中显示调用父类的有参构造方法
  8、如果父类的属性均为private修饰,则可以通过共有的getter,setter方法来调用
注:来自度娘的《类的继承》。
(2)变量的继承和覆盖,方法的继承,重载和覆盖的实现;
1、方法继承:利用extends关键字一个方法继承另一个方法,而且只能直接继承一个类。
当Sub类和Base类在同一个包时Sub类继承Base类中的public/protected/默认级别的变量个方法
在不同包时继承public/protected级别的变量和方法。

2、方法重载:如果有两个方法的方法名相同,但参数不一致,哪么可以说一个方法是另一个方法的重载。
方法名相同
方法的参数类型,个数顺序至少有一项不同
方法的返回类型可以不相同
方法的修饰符可以不相同
main方法也可以被重载

3、方法覆盖:如果在子类中定义一个方法,其名称、返回类型及参数签名正好与父类中某个方法的名称、返回类型及参数签名相匹配,那么可以说,子类的方法覆盖了父类的方法。
子类的方法名称返回类型及参数签名 必须与父类的一致
子类方法不能缩小父类方法的访问权限
子类方法不能抛出比父类方法更多的异常
方法覆盖只存在于子类和父类之间,同一个类中只能重载
父类的静态方法不能被子类覆盖为非静态方法
子类可以定义于父类的静态方法同名的静态方法,以便在子类中隐藏父类的静态方法(满足覆盖约束),
而且Java虚拟机把静态方法和所属的类绑定,而把实例方法和所属的实例绑定。
父类的非静态方法不能被子类覆盖为静态方法
父类的私有方法不能被子类覆盖
父类的抽象方法可以被子类通过两种途径覆盖(即实现和覆盖)(P169)
父类的非抽象方法可以被覆盖为抽象方法

4、Super关键字:super和this关键字都可以用来覆盖Java语言的默认作用域,使被屏蔽的方法或变量变为可见(三种情况下的不可见P171)。
父类的成员变量和方法为private使用super访问编译出错
在类的构造方法种,通过super语句调用这个类的父类的构造方法
在子类种访问父类的被屏蔽的方法和属性
只能在构造方法或实例方法内使用super关键字,而在静态方法和静态代码块内不能使用super

5、多态:
对于一个引用类型的变量,Java编译器按照它的声明的类型来处理
对于一个引用类型的变量,运行时Java虚拟机按照它的实际引用的对象来处理
运行时环境中,通过引用类型变量来访问所引用对象的方法和属性时,Java虚拟机采用以下绑定规则
1)实例方法与引用变量实际引用的对象的方法绑定,属于动态绑定
2)静态方法与引用变量所声明的类型的方法绑定,属于静态绑定
3)成员变量(包括静态和实例变量)与引用变量所声明的类型的成员变量绑定,属于静态绑定

6、继承的利弊和使用原则:
集成数的层次不可太多
集成数的上层为抽象层
(1)定义了下层子类都用友的相同属性和方法,并且尽可能默认实现,从而提高重用性
(2)代表系统的接口,描述系统所能提供的服务
继承关系最大的弱点:打破封装
精心设计专门用于被继承的类
(1)对这些类必须提供良好的文档说明
(2)尽可能的封装父类的实现细节,把代表时间细节的属性和方法定义为private类型
(3)如果某些实现细节必须被子类访问,定义为protected类型
(4)把不允许子类覆盖的方法定义为final类型
(5)父类的构造方法不允许调用可被子类覆盖的方法
(6)如果某些类不是专门为了继承而设计,那么随意继承它是不安全的
注:转自:http://blog.csdn.net/cdsnmdl/article/details/3968688
二.实验的内容
(1)根据下面的要求实现圆类Circle。
1.圆类Circle的成员变量:radius表示圆的半径。
2.圆类Circle的方法成员:

       Circle():构造方法,将半径置0
       Circle(double r):构造方法,创建Circle对象时将半径初始化为r
       double getRadius():获得圆的半径值
       double getPerimeter():获得圆的周长
       void disp():将圆的半径和圆的周长,圆的面积输出到屏幕
package 上课时间;

public class Circle {
	private double radius;
    Circle() {   //构造方法,将半径置0
        radius = 0;
    }
    
    Circle(double r) {  //构造方法,创建Circle对象时将半径初始化为r
        radius = r;
    }
    
    double getRadius() {  //获得圆的半径值
        return radius;
    }
    
    double getPerimeter() {  //获得圆的周长
        return 2*3.14*getRadius();
    }
    
    double getArea() {  //获得圆的面积
        return 3.14*Math.pow(radius, 2);
    }
    public static void main(String agrs[]) {   //将圆的半径和圆的周长,圆的面积输出到屏幕
    	new Circle1().disp();
    }
        
}
class Circle1 extends Circle{ //构造void disp()
	public void disp() {
		Circle c = new Circle(6371);
        
		System.out.println("Radius:"+c.getRadius()+"km");
        System.out.println("Perimeter:"+c.getPerimeter()+"km");
        System.out.println("Area:"+c.getArea()+"km*2");
	}
}

(2)继承第一题中的圆Circle类,派生圆柱体类Cylinder。要求如下:
1.圆柱体类Cylinder的成员变量:height表示圆柱体的高。
2.圆柱体类Cylinder的成员方法:

        Cylinder(double r,double h)构造方法,创建Cylinder对象时将圆半径初始化为r,圆柱体高初始化为h。
        double getHeight():获得圆柱体的高
        double getVol():获得圆柱体的体积
        void dispVol():将圆柱体的体积输出到屏幕
3.按照第二题的Cylinder类,接上键盘的输入值作为圆的半径和圆柱体的高,计算圆柱体的体积
package 上课时间;

public class Cylinder {
	private double radius;
	private double height;
    
	Cylinder(double r,double h) {  //构造方法,创建Cylinder对象时将圆半径初始化为r,圆柱体高初始化为double h;
        radius = r;
        height = h;
    }
    
	 double getRadius() {  //获得圆柱体的半径
        return radius;
    }
	
	double getHeight() {  //获得圆柱体的高
        return height;
    }
    
    double getVol() {  //获得圆柱体的体积
        return 3.14*Math.pow(radius, 2)*getHeight();
    }
    
    public static void main(String agrs[]){ 
    	double r = 0;    //前面没有将半径置0
    	double h = 0;    //前面没有将高置0
		((Cylinder1) new Cylinder1(r,h)).dispVol();  //使用dispVol()
    }

}
class Cylinder1 extends Cylinder{  //继承Cylinder
    Cylinder1(double r, double h) {
		super(r, h);
		// TODO 自动生成的构造函数存根
	}

	public void dispVol() {
    	Cylinder c = new Cylinder(40000,50);  //输入古罗马斗兽场半径和高
    	
    	System.out.println("Vol:"+c.getVol());    //输出
    }
}

原文地址:https://www.cnblogs.com/he932206959/p/11631163.html