张季跃 201771010139《面向对象程序设计(java)》第七周学习总结
张季跃 201771010139《面向对象程序设计(java)》第七周学习总结
第一部分:理论知识学习部分
第五章 继承:
继承:用已有类来构建新类的一种机制。当定 义了一个新类继承了一个类时,这个新类就继 承了这个类的方法和域,同时在新类中添加新 的方法和域以适应新的情况。 l
继承是Java程序设计中的一项核心技术,也是 面向对象特征之一。
5.2 Object:所有类的超类
l1.Object类是Java中所有类的祖先——每一个类都由它扩 展而来。在不给出超类的情况下,Java会自动把Object 作为要定义类的超类。 l
- 可以使用类型为Object的变量指向任意类型的对象。但 要对它们进行专门的操作都要进行类型转换。
- equals方法
(1)Object类中的equals方法用于测试某个对象是否同另一 个对象相等。它在Object类中的实现是判断两个对象是 否具有相同的引用。如果两个对象具有相同的引用,它 们一定是相等的。 l
(2)如果需要检测两个对象状态的相等性,就需要在新类的 定义中需要覆盖equals方法。
(3)定义子类的equals方法时,可调用超类的equals方法。
- hashCode方法
(1)Object类中的hashCode方法导出某个对象的散列 码。散列码是任意整数,表示对象的存储地址。 l
(2)两个相等对象的散列码相等。
- toString方法
(1)Object类的toString方法返回一个代表该对象域值的字符串。 l
- toString方法返回字符串的格式:类名,然后在方括号中 列举域值。 l
- 通过getClass().getName()获得类名的字符串。
- toString的调用方式: l
(a)一个字符串与对象名通过操作符“+”连接起来,就会 自动调用toString方法。
(b)如果x是任意一个对象,调用System.out.println(x), 就会直接地调用x.toString(),并打印输出字符串。 l
(2)定义子类的toString方法时,可先调用超类的toString方法。
(3)toString方法是非常重要的调试工具。标准类库中,多数类 定义了toString方法,以便用户获得对象状态的必要信息。
5.3泛型数组列表
l 1.Java中,利用ArrayList类,可允许程序在运行 时确定数组的大小。 l
- ArryList是一个采用类型参数的泛型类。为指定 数组列表保存元素的对象类型,需要用一对尖括 号将数组元素对象类名括起来加在后面。
- 没有<>的ArrayList将被认为是一个删去了类型 参数的“原始”类型
- 数组列表的操作
(1)a.ArrayList定义 l
- ArrayList<T> 对象=new ArrayList<T>();
- API:ArrayList的构造器 –ArrayList<T>()构造一个空数组列表 –ArrayList<T>(intinitialCapacity)构造一个具有 指定容量的空数组列表。
(2)API:booleanadd(T obj) 把元素obj追加到数组列表的结尾
(3)API:intsize() 返回数组列表中当前元素个数
(4)API:void trimToSize() 把数组列表的存贮空间调整到当前大小
(5)API:void set(intindex, T obj) 将obj放入数组列表index位置,将覆盖这 个位置的原有内容。
API:T get(intindex) 获得指定位置index的元素值
(6)API:booleanadd(intindex, T obj) 向后移动元素,在第n 个位置插入obj
API:T remove(intindex); 将第n个位置存放的对象删除,并将后面的 元素向前移
5.4 对象包装器与自动打包
l 1.所有基本数据类型都有着与之对应的预定义类,它们被称 为对象包装器(wrapper)。 l
- 以下前6个对象包装器类都是从公共包装器类Number继承 而来。 Integer Long Float Double Short Byte Character VoidBoolean l
- 对象包装器类是不可变的,即一旦构造了包装器,就不允 许更改包装在其中的值。且对象包装器类还是final,因 此不能定义它们的子类。 l
- 使用对象包装器的好处: –基本类型转化为对象 –定义一些有用的基本方法(static方法)
在JavaSE5.0中,可以自动的将基本数据类型转换 为包装器类的对象,将这种变换称为自动打包 (autoboxing)。
- 相反地,当对一个包装器类的对象进行赋值或算法 运算时,将会自动地拆包。
- 打包和拆包是编译器认可的
5.5 参数数量可变的方法
l 1.在Java SE 5.0以前的版本中,每个Java方法都 有固定数量的参数。然而,现在的版本提供了可 以用可变的参数数量调用的方法(称为“可变参 ”方法)。
2.用户自己可以定义可变参数的方法,并将参数指 定为任意类型,甚至是基本类型
5.6 枚举类
1.声明枚举类:它包括一个关键字enum,一个新枚举类型的名字 Grade以及为Grade定义的一组值,这里的值既 非整型,亦非字符型。
5.7 继承设计的技巧
①将公共操作和域放在超类。
②不要使用受保护的域。
③使用继承实现“is-a”关系。
④除非所有继承的方法都有意义,否则就不要 使用继承。
⑤在覆盖方法时,不要改变预期的行为。
⑥使用多态,而非类型信息。 ⑦不要过多地使用反射。
第二部分:实验部分
1、实验目的与要求
(1)进一步理解4个成员访问权限修饰符的用途;
(2)掌握Object类的常用API用法;
(3)掌握ArrayList类用法与常用API;
(4)掌握枚举类使用方法;
(5)结合本章知识,理解继承与多态性两个面向对象程序设计特征,并体会其优点;
(6)熟练掌握Java语言中基于类、继承技术构造程序的语法知识(ch1-ch5);
(7)利用已掌握Java语言程序设计知识,学习设计开发含有1个主类、2个以上用户自定义类的应用程序。
2、实验内容和步骤
实验1 补充以下程序中主类内main方法体,以验证四种权限修饰符的用法。
public class TEST1 {
private String t1 = "这是TEST1的私有属性";
public String t2 = "这是TEST1的公有属性";
protected String t3 = "这是TEST1受保护的属性";
String t4 = "这是TEST1的默认属性";
private void tese1() {
System.out.println("我是TEST1用private修饰符修饰的方法");
}
public void tese2() {
System.out.println("我是TEST1用public修饰符修饰的方法");
}
protected void tese3() {
System.out.println("我是TEST1用protected修饰符修饰的方法");
}
void tese4() {
System.out.println("我是TEST1无修饰符修饰的方法");
}
}
public class TEST2 extends TEST1{
private String e1 = "这是TEST2的私有属性";
public String e2 = "这是TEST2的公有属性";
protected String e3 = "这是TEST2受保护的属性";
String e4 = "这是TEST2的默认属性";
public void demo1() {
System.out.println("我是TEST2用public修饰符修饰的方法");
}
private void demo2() {
System.out.println("我是TEST2用private修饰符修饰的方法");
}
protected void demo3() {
System.out.println("我是TEST2用protected修饰符修饰的方法");
}
void demo4() {
System.out.println("我是TEST2无修饰符修饰的方法");
}
}
public class Main {
public static void main(String[] args) {
TEST2 test2 = new TEST2();
/*以下设计代码分别调用 demo1 demo2 demo3 demo4 test1 test2 test3 test4方法和t1 t2 t3 t3 e1 e2 e3 e4属性,结合程序运行结果理解继承和权限修饰符的用法与区别*/
}
}
补充的main方法:
import java.util.Scanner;
/*
* 主类
*/
public class Main {
public static void main(String[] args) {
Scanner in = new Scanner(System.in);
System.out.println("请输入矩形的长和宽");
int length = in.nextInt();
int width = in.nextInt();
System.out.println("请输入圆的半径");
int radius = in.nextInt();
Rectangle rectangle = new Rectangle(length, width);
Circle circle = new Circle(radius);
// 求周长和面积
int perimeter = rectangle.getPerimeter();
int area = rectangle.getArea();
double perimeter2 = circle.getPerimeter();
double area2 = circle.getArea();
System.out.println(rectangle.toString() + " 周长为:" + perimeter + " 面积为:" + area);
System.out.println(circle.toString() + " 周长为:" + perimeter2 + " 面积为:" + area2);
}
}
Rectangle:
/*
* 自定义矩形类
*/
public class Rectangle {
// 类的成员变量(属性)
private int length;
private int width;
// 构造函数
public Rectangle(int length, int width) {
this.length = length;
this.width = width;
}
// 求周长的方法
public int getPerimeter() {
int perimeter = (width + length) * 2;
return perimeter;
}
// 求面积的方法
public int getArea() {
int area = width * length;
return area;
}
// 覆盖toSting方法
@Override
public String toString() {
// TODO 自动生成的方法存根
return "该矩形的长为:" + length + "宽为" + width;
}
}
Circle:
/*
* 自定义圆形类
*/
public class Circle {
// 类的成员变量(属性)
private int radius;
// 构造函数
public Circle(int radius) {
this.radius = radius;
}
// 求周长的方法
public double getPerimeter() {
double perimeter = 2 * Math.PI * radius;
return perimeter;
}
// 求面积的方法
public double getArea() {
double area = Math.PI * radius * radius;
// 可以使用求平方的方法
return area;
}
// 覆盖toSting方法
@Override
public String toString() {
// TODO 自动生成的方法存根
return "该圆形的半径为:" + radius ;
}
}
实验结果:
实验2 第五章测试程序反思,继承知识总结。
测试程序1:
Ÿ 编辑、编译、调试运行教材程序5-8、5-9、5-10(教材174页-177页);
Ÿ 结合程序运行结果,理解程序代码,掌握Object类的定义及用法;
实验程序:
Employee:
package equals;
import java.time.*;
import java.util.Objects;
public class Employee
{
private String name;
private double salary;
private LocalDate hireDay;
public Employee(String name, double salary, int year, int month, int day)
{
this.name = name;
this.salary = salary;
hireDay = LocalDate.of(year, month, day);
}
public String getName()
{
return name;
}
public double getSalary()
{
return salary;
}
public LocalDate getHireDay()
{
return hireDay;
}
public void raiseSalary(double byPercent)
{
double raise = salary * byPercent / 100;
salary += raise;
}
public boolean equals(Object otherObject)
{
// a quick test to see if the objects are identical
if (this == otherObject) return true;
// must return false if the explicit parameter is null
if (otherObject == null) return false;
// if the classes don't match, they can't be equal
if (getClass() != otherObject.getClass()) return false;
// now we know otherObject is a non-null Employee
Employee other = (Employee) otherObject;
// test whether the fields have identical values
return Objects.equals(name, other.name) && salary == other.salary && Objects.equals(hireDay, other.hireDay);
}
public int hashCode()
{
return Objects.hash(name, salary, hireDay);
}
public String toString()
{
return getClass().getName() + "[name=" + name + ",salary=" + salary + ",hireDay=" + hireDay
+ "]";
}
}
Equaltest:
package equals;
/**
* This program demonstrates the equals method.
* @version 1.12 2012-01-26
* @author Cay Horstmann
*/
public class EqualsTest
{
public static void main(String[] args)
{
Employee alice1 = new Employee("Alice Adams", 75000, 1987, 12, 15);
Employee alice2 = alice1;
Employee alice3 = new Employee("Alice Adams", 75000, 1987, 12, 15);
Employee bob = new Employee("Bob Brandson", 50000, 1989, 10, 1);
System.out.println("alice1 == alice2: " + (alice1 == alice2));
System.out.println("alice1 == alice3: " + (alice1 == alice3));
System.out.println("alice1.equals(alice3): " + alice1.equals(alice3));
System.out.println("alice1.equals(bob): " + alice1.equals(bob));
System.out.println("bob.toString(): " + bob);
Manager carl = new Manager("Carl Cracker", 80000, 1987, 12, 15);
Manager boss = new Manager("Carl Cracker", 80000, 1987, 12, 15);
boss.setBonus(5000);
System.out.println("boss.toString(): " + boss);
System.out.println("carl.equals(boss): " + carl.equals(boss));
System.out.println("alice1.hashCode(): " + alice1.hashCode());
System.out.println("alice3.hashCode(): " + alice3.hashCode());
System.out.println("bob.hashCode(): " + bob.hashCode());
System.out.println("carl.hashCode(): " + carl.hashCode());
}
}
Manager
package equals;
public class Manager extends Employee
{
private double bonus;
public Manager(String name, double salary, int year, int month, int day)
{
super(name, salary, year, month, day);
bonus = 0;
}
public double getSalary()
{
double baseSalary = super.getSalary();
return baseSalary + bonus;
}
public void setBonus(double bonus)
{
this.bonus = bonus;
}
public boolean equals(Object otherObject)
{
if (!super.equals(otherObject)) return false;
Manager other = (Manager) otherObject;
// super.equals checked that this and other belong to the same class
return bonus == other.bonus;
}
public int hashCode()
{
return java.util.Objects.hash(super.hashCode(), bonus);
}
public String toString()
{
return super.toString() + "[bonus=" + bonus + "]";
}
}
实验结果:
测试程序2:
Ÿ 编辑、编译、调试运行教材程序5-11(教材182页);
Ÿ 结合程序运行结果,理解程序代码,掌握ArrayList类的定义及用法;
测试程序:
测试程序3:
ArrayListTest:
package arrayList;
import java.util.*;
/**
* This program demonstrates the ArrayList class.
* @version 1.11 2012-01-26
* @author Cay Horstmann
*/
public class ArrayListTest
{
public static void main(String[] args)
{
// fill the staff array list with three Employee objects
ArrayList<Employee> staff = new ArrayList<>();
staff.add(new Employee("Carl Cracker", 75000, 1987, 12, 15));
staff.add(new Employee("Harry Hacker", 50000, 1989, 10, 1));
staff.add(new Employee("Tony Tester", 40000, 1990, 3, 15));
// raise everyone's salary by 5%
for (Employee e : staff)
e.raiseSalary(5);
// print out information about all Employee objects
for (Employee e : staff)
System.out.println("name=" + e.getName() + ",salary=" + e.getSalary() + ",hireDay="
+ e.getHireDay());
}
}
Employee:
package arrayList;
import java.time.*;
public class Employee
{
private String name;
private double salary;
private LocalDate hireDay;
public Employee(String name, double salary, int year, int month, int day)
{
this.name = name;
this.salary = salary;
hireDay = LocalDate.of(year, month, day);
}
public String getName()
{
return name;
}
public double getSalary()
{
return salary;
}
public LocalDate getHireDay()
{
return hireDay;
}
public void raiseSalary(double byPercent)
{
double raise = salary * byPercent / 100;
salary += raise;
}
}
实验结果:
测试程序3:
Ÿ 编辑、编译、调试运行程序5-12(教材189页);
Ÿ 结合运行结果,理解程序代码,掌握枚举类的定义及用法;
EnumTest:
package enums;
import java.util.*;
/**
* This program demonstrates enumerated types.
* @version 1.0 2004-05-24
* @author Cay Horstmann
*/
public class EnumTest
{
public static void main(String[] args)
{
Scanner in = new Scanner(System.in);
System.out.print("Enter a size: (SMALL, MEDIUM, LARGE, EXTRA_LARGE) ");
String input = in.next().toUpperCase();
Size size = Enum.valueOf(Size.class, input);
System.out.println("size=" + size);
System.out.println("abbreviation=" + size.getAbbreviation());
if (size == Size.EXTRA_LARGE)
System.out.println("Good job--you paid attention to the _.");
}
}
enum Size
{
SMALL("S"), MEDIUM("M"), LARGE("L"), EXTRA_LARGE("XL");
private Size(String abbreviation) { this.abbreviation = abbreviation; }
public String getAbbreviation() { return abbreviation; }
private String abbreviation;
}
实验结果:
实验3:采用个人账号登录https://pintia.cn/,完成《2018秋季西北师范大学面向对象程序设计(Java)(ch1-ch5)测试题2》,测试时间60分钟;
实验4: 课后完成实验3未完成的测试内容。
第三部分:实验总结:
本周的学习主要是对上一周学习知识的巩固和加深,对Java四个权限符的深入了解和对继承的进一步掌握。但在这一周的测试中,我发现我对这些知识了解掌握的并不是多么熟练,不仅是前面的理论知识不行,更重要的是后面的编程题,花费了大量时间我还是只答出了一道,还是最简单的一道,我还需要在这方面多下一些功夫。