超类和子类
- 使用extends关键字表明正在构造的新类派生于一个已存在的类。
- super关键字可以在子类中调用超类的方法,也可以调用超类的构造器。
- 子类的每个对象也是超类的对象。程序中出现超类对象的任何地方都可以用子类对象置换。
Employee e;
e = new Employee(...);
e = new Manger(...);
- 在java中,对象变量是多态的。一个Employee变量既可以引用一个employee类对象,也可以引用一个employee类的任何一个子类的对象。
- 然而,不能将一个超类的引用赋给子类变量。
- 子类数组的引用可以转换为超类数组的引用,而不需要采用强制类型转换。
Manager[] managers = new Manager[10];
Employee[] staff = managers;
- 在覆盖一个方法的时候,子类方法不能低于超类方法的可见性。
- 特别的,如果超类方法是public,子类方法一定要声明为public。
- 使用final修饰符来修饰类的时候,可以阻止定义子类。
- 使用final修饰方法的时候,可以阻止子类覆盖此方法。
- 使用final修饰域,构造对象之后就不允许改变值了。
- final类中所有方法自动成为final方法,但不包括域。
- 将方法或类声明为final的主要目的是:确保它们不会在子类中改变语义。
- 将一个类型强制转换成另外一个类型的过程被称为类型转换。
- 有时候需要将某个类的对象引用转换成另一个类的对象引用。
- 对象引用的转换语法与数值表达式的类型转换类型,用圆括号括起来。
- 进行类型转换的唯一原因是:在暂时忽视对象的实际类型之后,使用对象的全部功能。
- 关键字abstract来声明抽象方法或抽象类。
- 抽象方法充当着占位的角色,它们的具体实现在子类中。
- 扩展抽象类有两种选择:
1)在抽象类中定义部分抽象类方法或不定义抽象类方法,这边就必须将子类也标记为抽象类。
2)定义全部的抽象类,这样一来,子类就不是抽象的了。 - 类即使不含抽象方法,也可以将类声明为抽象类。
- 抽象类不能被实例化,不能创建这个类的对象,但可以创建一个具体子类的对象。
- 可以定义一个抽象类的对象变量,但它只能引用非抽象子类的对象。
Person p = new 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方法可用于返回表示对象值的字符串。
泛型数组列表
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");
枚举类
- 枚举类型都是Enum类的子类。
- 可以直接使用“==”来比较两个枚举类型的值。
- Enum类的toString方法可以返回枚举常量名。
- toString的逆方法是静态方法valueOf,将返回一个包含全部枚举值的数组。
- ordinal方法返回enum声明中枚举常量的位置。
反射
- 能够分析类能力的程序称为反射(reflective)。
- 反射机制可以用来:
1)在运行时分析类的能力。
2)在运行时查看对象。
3)实现通用的数组操作代码。
4)利用Method对象,这个对象很想C++中的函数指针。 - 反射机制主要被工具构造者使用,而非应用程序员。
继承的设计技巧
- 将共有操作和域放在超类。
- 不要使用受保护的域。
- 使用继承实现“is-a”关系。
- 除非所有继承的方法都有意义,否则不要使用继承。
- 在覆盖方法时,不要改变预期的行为。
- 使用多态,而非类型信息。
- 不要过多使用反射。