1.多态
1)Java引用变量的两种类型:一个是编译时类型,一个是运行时类型。
编译时类型由声明该变量时使用的类型决定,运行时类型由实际赋给该变量的对象决定。
当编译时类型和运行时类型不一致的时候,就可能产生多态
package cn.it.lsl;
class BaseClass{
public int book=6;
public void base(){
System.out.println("父类方法");
}
public void test(){
System.out.println("被子类覆盖的方法");
}
}
public class SubClass extends BaseClass{
public String book = "JavaSe";
public void test(){
System.out.println("子类test()方法");
}
public void sub(){
System.out.println("子类方法");
}
public static void main(String[] args) {
BaseClass bc = new BaseClass();
System.out.println(bc.book);
bc.base();
bc.test();
System.out.println("-----------");
SubClass sc = new SubClass();
System.out.println(sc.book);
sc.sub();
sc.test();
System.out.println("-----------");
BaseClass ploy = new SubClass();
System.out.println(ploy.book); // 对象的Field不具备多态
ploy.base();
ploy.test();
//ploy.sub(); //编译出错
}
}
分析:
BaseClass ploy = new SubClass();编译时类型是BaseClass,运行时类型是SubClass,当调用引用变量的test方法是,实际上执行的是SubClass类中覆盖后的test方法,这就出现多态。
当运行时调用该引用变量的方法时,其方法总是变现出子类方法的行为特征,而不是父类的行为特征。
所以,相同类型变量,调用同一个方法是出现不用的特性,这就是多态。
ploy.sub();会产生错误,因为代码在编译时出错的。因为他的编译时是类型是BaseClass,编译时无法调用sub()。 (引用变量在编译阶段只能调用其编译时类型所具有的方法)
System.out.println(ploy.book);输出6,因为对象的Field不具备多态。
2)把一个子类对象直接赋值给父类引用变量,无须任何类型转换,这种称为向上转型。
把一个父类对象赋值给子类引用变量时,就需要进行强制类型转换,这时候应该使用instanceof运算符保证强制类型转换更安全。
3)instanceof
instandeof运算符的前一个操作数是一个引用类型的变量,后一个操作数是一个类(接口),它用于判断前面的对象是否是后面的类,或子类、实现类的实例。
instanceof运算符前面操作数的编译时类型要么与后面类相同,要么与后面的类具有父子继承关系。否则会引起编译错误。
package cn.it.lsl;
public class IntanceofDemo {
public static void main(String[] args) {
Object hello = "hello";
System.out.println(hello instanceof Object);
System.out.println(hello instanceof Math);
String a = "hello";
//System.out.println(a instanceof Math);
}
}
在强制类型转换之前,首先判断前一个对象是否是后一个类的实例,是否可以转换成功,从而报障代码的健壮性。
2.继承与组合
1)尽量不要在父类构造器中调用将要被子类重写的方法。
以下代码将发生空指针异常,Exception in thread "main" java.lang.NullPointerException
package cn.it.lsl;
class Base{
public Base(){
test();
}
public void test(){
System.out.println("要被子类重写的方法");
}
}
public class Sub extends Base{
private String name;
public void test(){
System.out.println(name.length());
}
public static void main(String[] args){
Sub s = new Sub();
}
}
new Sub()的时候会先执行父类的构造器,如果父类的构造器调用了被其子类重写的方法,则变成调用被子类重写后的方法。
即调用
public void test(){
System.out.println(name.length());
}
此时name是null,所以会抛出异常。
2)在代码复用的时候,运用继承会带来破封装性。组合也能实现代码的复用性,采用组合能够提供更好的封装。
3)组合是把旧类对象作为新类Field嵌入,用以实现新类的功能。一般在新类里使用private修饰被嵌入的旧类对象。
package cn.it.lsl;
class Animal{
private void beat(){
System.out.println("心脏跳动");
}
public void breath(){
beat();
System.out.println("呼吸");
}
}
class Bird{
private Animal a;
public Bird(Animal a){
this.a = a;
}
public void breath(){
a.breath();
}
public void fly(){
System.out.println("飞翔");
}
}
public class CompositeTest {
public static void main(String[] args) {
Animal a = new Animal();
Bird b = new Bird(a);
b.breath();
b.fly();
}
}
3.初始化块
1)初始化块可以对Java对象进行初始化操作。
2)若初始化块带有修饰符,则修饰符只能是static,其修饰的初始化块被称为静态初始化块。
3)当创建Java对象的时候,系统先调用该类里面的初始化块。而且在构造器之前执行。
4)当创建一个Java对象时,系统会先执行java.lang.Object类的初始化块,开始执行java.lang.Object的构造器,在执行器父类的初始化块,在开始执行其父类的构造器...然后才是执行该类的初始化块和构造器。
5)系统在类初始化阶段就执行了静态初始化块,而不是创建对象时才执行。所以静态初始化块总是比普通初始化块先执行。
package cn.it.lsl;
class Root{
static{
System.out.println("root静态初始化块");
}
{
System.out.println("root普通初始化块");
}
public Root(){
System.out.println("root无参构造");
}
}
class Mid extends Root{
static{
System.out.println("mid静态初始化块");
}
{
System.out.println("mid普通初始化块");
}
public Mid(){
System.out.println("mid无参构造");
}
public Mid(String msg){
this();
System.out.println("mid有参构造:"+msg);
}
}
class Leaf extends Mid{
static{
System.out.println("leaf静态初始化块");
}
{
System.out.println("leaf普通初始化块");
}
public Leaf(){
super("JavaSe");
System.out.println("leaf构造器");
}
}
public class Test {
public static void main(String[] args) {
new Leaf();
new Leaf();
}
}
执行结果:
root静态初始化块
mid静态初始化块
leaf静态初始化块
root普通初始化块
root无参构造
mid普通初始化块
mid无参构造
mid有参构造:JavaSe
leaf普通初始化块
leaf构造器
root普通初始化块
root无参构造
mid普通初始化块
mid无参构造
mid有参构造:JavaSe
leaf普通初始化块
leaf构造器