Java面试Java

Java对象、类、多态回顾

2017-04-14  本文已影响38人  stefanJi

内存中的对象

数据类型 储存位置
基本类型(char byte short int long float double boolean)
Java对象

⚠️

一个对象可以被多个引用变量引用:

Book myBook = new Book();
Book yourBook = myBook;

第二行代码把myBook的值复制给yourBook,结果是yourBook现在和myBook引用同一个Book对象。

类访问控制修饰符

类具有两种访问修饰

package base;
public class Book {
    String name;
    int number;
}

Book 类是base包的一个成员,由于Book类是公用的,可从任何其他类实例化。

package base;

class Stu{
    String name;
    int age;
}

Stu 类的访问限制就是缺省的,只能被同一个包中的其他类使用。

一个Java源文件只能包含一个公共类,但可以包含多个非公共类

类成员访问修饰符

访问级别 从其他包中 从同一个包中的类 从子类 从同一个类
Public
Protected
缺省(包私有)
Private

final变量

final 声明的变量,一旦赋值,就不能改变
⚠️ 注意

本质是因为:final 限定了变量的栈区不可改变

package base;
public class Book{  
    String name;
    int age;
    
    public static void main(String args[]){
        final Book book = new Book();
        final int a = 10;
        a = 20; //报错
        book.name = "name";
        book.age = 20;      
        book.age = 21;
        book.name = "name2";
        book = new Book(); //报错
    }
}

static静态变量

⚠️ 静态引用变量 static Book book = new Book(); book变量将包含Book对象的地址,对象还是储存在堆中

参数传递

数据类型 传递方式 说明
基本类型 值传递 JVM将会把传递进来的变量复制给一个新的局部变量
引用变量 引用传递 局部变量就指向和引用变量一样的对象

JVM执行类的流程

加载->链接->初始化

类型 默认值
boolean false
byte 0
short 0
int 0
long 0L
float 0.0f
char \u0000
double 0.0d
对象引用 null

对象比较

⚠️ 引用变量并不包含对象,而是包含对象在内存(栈)中的地址

因此:

Object a = new Object();
Object b = new Object();

a == b 将返回 false
这样的比较实际没有多大用处,因为大多数时候我们更关心对象,而不是对象的地址。
所以,比较对象可以通过两种方式:

  1. 提供比较对象的工具类
  2. 为类实现继承自java.lang.ObjectequalshashCode方法

Java多态

多态是Java面向对象三大特性之一

要实现Java多态有3个必要条件和2种实现方式

必要条件

  1. 继承
  2. 重写
  3. 向上转型

实现方式

对于继承实现的多态:

对于引用子类类型的父类,在处理该引用时,它适用于继承该父类的所有子类,子类的
对象不同,对方法的实现也就不同,执行相同动作产生的行为也就不同。

对于接口实现的多态:

指向接口的引用必须是指定实现了该接口的一个类的实例,在运行时,根据对象引用的实际类型来执行对应的方法

在继承链中,对象方法的调用存在一个优先级

该优先级为:

this.fun() 
    super.fun() 
        this.fun((super)O) 
            super.fun((super)O)

下面是经典的多态例子:

class A {
    public String show(D obj) {
        return ("A and D");
    }
    public String show(A obj) {
        return ("A and A");
    }
}

class B extends A{
    public String show(B obj){
        return ("B and B");
    }
    
    public String show(A obj){
        return ("B and A");
    }
}

class C extends B{
}

class D extends B{
}

public class Main {
    public static void main(String[] args){ 
        A a1 = new A();
        A a2 = new B();
        B b = new B();
        C c = new C();
        D d = new D();
        
        System.out.println("1--" + a1.show(b));//输出A-A
        //执行A的show(A object)
        //this.show() no
        //super.show() no
        //(super)b = A
        //this.show((super)b) yes
        
        System.out.println("2--" + a1.show(c));//输出A-A
        //因为A中没有show(C object)方法,
        //所有到A的父类中寻找,但是A没有父类,
        //所以执行show((super)c),c的父类为B和A
        //因为A中只有show(A object)方法,所以最终执行show(A object)
        /**
         * 实质执行:a1.show((super)c);
         */
        //this.show() no
        //super.show() no
        //(super)c = B/A
        //this.show((super)c=B) no
        //this.show((super)c=A) yes
        
        System.out.println("3--" + a1.show(d));//输出A-D
        //执行A的show(D Object)
        
        System.out.println("4--" + a2.show(b));//输出B-A
        //a2.show()实质优先调用a2引用的对象的show方法
        //a2的this是指向B的对象
        //this.show no
        //super.show no
        //(super)b = A
        //this.show((super)b) yes
        
        System.out.println("5--" + a2.show(c));//输出B-A
        //a2.show()实质优先调用a2引用的对象的show方法
        //但是B中没有show(C object)方法,于是到B的父类A中寻找,也没有show(C object)方法,
        //于是回到B中尝试执行show((super)c),c的super是B和A,
        //B的父类A中只有针对A的show方法,所以执行B中重写的父类的show(A object)方法,而不会执行B的show(B object)
        //this.show no
        //super.show no
        //(super)c = B/A
        //this.show((super)c = B) yes
        
        System.out.println("6--" + a2.show(d));//输出A-D
        //a2指向B的对象中没有show(D object)方法,于是到父类A中查找,发现有符合,
        //所以执行B的父类A的show(D object)
        //this.show no
        //super.show yes
        
        System.out.println("7--" + b.show(b));//输出B-B
        //this.show yes
        
        System.out.println("8--" + b.show(c));//输出B-B
        //this.show no
        //super.show no
        //(super)c = B/A
        //this.show((super)c=B) yes
        
        System.out.println("9--" + b.show(d));//输出A-D
        //this.show no
        //super.show yes
        
    }
}

运行结果如下:
 1--A and A
 2--A and A
 3--A and D
 4--B and A
 5--B and A
 6--A and D
 7--B and B
 8--B and B
 9--A and D
上一篇下一篇

猜你喜欢

热点阅读