Java核心技术(四)——继承

超类和子类
  • 使用extends关键字表明正在构造的新类派生于一个已存在的类。
  • super关键字可以在子类中调用超类的方法,也可以调用超类的构造器。
  • 子类的每个对象也是超类的对象。程序中出现超类对象的任何地方都可以用子类对象置换。
	// Manager extends Employee;
	Employee e;
	e = new Employee(...);
	e = new Manger(...);
  • 在java中,对象变量是多态的。一个Employee变量既可以引用一个employee类对象,也可以引用一个employee类的任何一个子类的对象。
  • 然而,不能将一个超类的引用赋给子类变量。
  • 子类数组的引用可以转换为超类数组的引用,而不需要采用强制类型转换。
 	// Manager extends Employee;
	Manager[] managers = new Manager[10];
	Employee[] staff = managers;
  • 在覆盖一个方法的时候,子类方法不能低于超类方法的可见性。
  • 特别的,如果超类方法是public,子类方法一定要声明为public。
  • 使用final修饰符来修饰类的时候,可以阻止定义子类。
  • 使用final修饰方法的时候,可以阻止子类覆盖此方法。
  • 使用final修饰域,构造对象之后就不允许改变值了。
  • final类中所有方法自动成为final方法,但不包括域。
  • 将方法或类声明为final的主要目的是:确保它们不会在子类中改变语义。
  • 将一个类型强制转换成另外一个类型的过程被称为类型转换。
  • 有时候需要将某个类的对象引用转换成另一个类的对象引用。
  • 对象引用的转换语法与数值表达式的类型转换类型,用圆括号括起来。
  • 进行类型转换的唯一原因是:在暂时忽视对象的实际类型之后,使用对象的全部功能。
  • 关键字abstract来声明抽象方法或抽象类。
  • 抽象方法充当着占位的角色,它们的具体实现在子类中。
  • 扩展抽象类有两种选择:
    1)在抽象类中定义部分抽象类方法或不定义抽象类方法,这边就必须将子类也标记为抽象类。
    2)定义全部的抽象类,这样一来,子类就不是抽象的了。
  • 类即使不含抽象方法,也可以将类声明为抽象类。
  • 抽象类不能被实例化,不能创建这个类的对象,但可以创建一个具体子类的对象。
  • 可以定义一个抽象类的对象变量,但它只能引用非抽象子类的对象。
	Person p = new Student();
	//p 是抽象类Person的变量,Person引用了一个非抽象子类Student的实例。
  • 关键字proteced可以允许超类中的某些方法或域被子类访问。
  • Java中受保护部分对所有子类以及同一个包中的所有其他类都可见。
  • 控制可见性的4个访问修饰符:
    1)仅对本类可见——private。
    2)对所有类可见——public。
    3)对本包和所以子类可见——protected。
    4)对本包可见——默认,不需要修饰符。
Object类
  • Object类是Java中所有类的始祖。
  • 可以使用Object类型的变量引用任何类型的对象。
  • 在Java中,只有基本类型不是对象,如,数值、字符和布尔类型的值都不是类型。
  • 所有数组类型,不管是对象数组还是基本类型的数组都扩展了Object类。
  • 在C++中没有所有类的根类,不过,每个指针都可以转换成void*指针。
  • Object类中的equals方法用于检测一个对象是否等于另外一个对象。这个方法将判断两个对象是否具有相同的引用。
  • equals方法特性:
    1)自反性:对于任何非空引用x,x.equials(x)返回true;
    2)对称性:对于任何引用x和y,当且仅当y.equals(x)返回true,x.equals(y)也应该返回true。
    3)传递性:对于任何引用x、y和z,如果x.equals(y)返回true,y.equals(z)返回true,x.equals(z)也应该返回true。
    4)一致性:如果x和y引用的对象没有发生变化,反复调用x.equals(y)应该返回同样的结果。
    5)对于任意非空引用x,x.equals(null)应该返回false。
  • 散列码(hash code)是由对象导出的一个整型值。
  • 散列码是没有规律的。
  • hashCode方法定义在Object类中,每个对象都有个默认的散列码,其值为对象的存储地址。
  • 字符串的散列码是由内容导出的。
  • Object的toString方法可用于返回表示对象值的字符串。
泛型数组列表
  • Java允许在运行时确定数组的大小。
	int actualSize = ...;
	Employee[] staff = new Employee[actualSize];
  • ArrayList是一个采用类型参数的泛型类。
  • 声明和构造一个保存employee对象的数组列表:
	ArrayList<Employee> staff = new ArrayList<Employee>();
	ArrayList<Employee> staff = new ArrayList<>();
	ArrayList<Employee> staff = new ArrayList<Employee>(100); //设置一个初始容量
  • 使用add方法可以将元素添加到数组列表中。如果调用add且内部数组已经满了,数组列表将自动创建一个更大的数组,并将所有的对象从较小的数组中拷贝到较大的数组中。
  • 如果能估计所需数组列表的大小 可以在填充数组前调用ensurtCapacity方法。
  • 确定数组列表大小不发生变化之后,可以调用trimToSize方法,将存储空间大小调整为当前元素数量的空间大小。
  • ArrayList类并非Java语言的一部分,而是在标准库中的一个实用类。
  • ArrayList类通过使用get和set方法实现访问或改变数组元素的操作。
对象包装器与自动装箱
  • 所有的基本类型都有一个与之相对的类,这些类称为包装器(wrapper)
  • 对象包装器类是不可变的,一旦构造了包装器,就不允许更改包装在其中的值。
  • 对象包装器类还是final,因此不能定义他们的子类。
参数数量可变的方法
	public class PrintStream
	{
		public PrintStream Printf(String fmt, Object... args){return format(fmt, args);}
	}
	System.out.printf("%d",n);
	System.out.printf("%d %s",n,"widgets");
	//省略号...是Java代码的易付宝,表明这个方法可以接受任意数量的对象。
	//printf方法接受两个参数,一个是格式字符串,另一个是Object[]数组,其中保存着所有参数。
	//将扫描fmt字符串,并将第i个格式说明符与args[i]的值匹配起来。
枚举类
  • 枚举类型都是Enum类的子类。
  • 可以直接使用“==”来比较两个枚举类型的值。
  • Enum类的toString方法可以返回枚举常量名。
  • toString的逆方法是静态方法valueOf,将返回一个包含全部枚举值的数组。
  • ordinal方法返回enum声明中枚举常量的位置。
反射
  • 能够分析类能力的程序称为反射(reflective)。
  • 反射机制可以用来:
    1)在运行时分析类的能力。
    2)在运行时查看对象。
    3)实现通用的数组操作代码。
    4)利用Method对象,这个对象很想C++中的函数指针。
  • 反射机制主要被工具构造者使用,而非应用程序员。
继承的设计技巧
  • 将共有操作和域放在超类。
  • 不要使用受保护的域。
  • 使用继承实现“is-a”关系。
  • 除非所有继承的方法都有意义,否则不要使用继承。
  • 在覆盖方法时,不要改变预期的行为。
  • 使用多态,而非类型信息。
  • 不要过多使用反射。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值