补充:面试常见的问题之 Java 基础

一、Object 常用的方法

  • public native int hashCode():返回散列值。
  • public boolean equals(Object obj):比较两对象是否相等。
  • protected native Object clone() throws CloneNotSupportedException:创建并返回此对象的副本。
  • public String toString():一般要重写这个方法,重写后返回的是对象的字符串表示形式。
  • public final native Class<?> getClass():返回此 Object 的运行时类。
  • protected void finalized() throws Throwable{}:当垃圾收集确定没有对该对象的更多引用时,由对象上的垃圾收集器调用。
  • public final native void notify():唤醒正在此对象监视器上等待的单个线程。
  • public final native void notifyAll():唤醒正在此对象监视器上等待的所有线程。
  • public final native void wait(long timeout) throws InterruptedException:导致当前线程等待,直到另一个线程调用此对象的 notify() 方法或 notifyAll() 方法,或者已经过了指定的时间量。
  • public final void wait(long timeout, int nanos) throws InterruptedException:导致当前线程等待,直到另一个线程为此对象调用 notify() 方法或 notifyAll() 方法,或者某个其他线程中断当前线程,或者已经过了一定量的实时。
  • public final void wait() throws InterruptedException:导致当前线程等待,直到另一个线程为此对象调用 notify() 方法或 notifyAll() 方法。

 

二、简述 Java 中内部类

(一) 为什么使用内部类?

最吸引人的原因就是每个内部类都能独立地继承一个(接口的)实现,所以无论外部类是否已经继承了某个(接口的)实现,对于内部类都没有影响。

使用内部类最大的有点就在于它能够非常好的解决多重继承的问题,使用内部类还能够为我们带来如下特性:

  • 内部类可以用多个实例,每个实例都有自己的状态信息,并且与其他外部类的信息相互独立;
  • 在单个外部类中,可以让多个内部类以不同的方式现实同一个接口,或者继承同一个类;
  • 创建内部类对象的时刻并不依赖于外部类对象的创建;
  • 内部类并没令人迷惑的 “is a” 的关系,他就是一个独立的实体;
  • 内部类提供了更好的封装,除了该外部类,其他类都不能访问。

(二) 成员内部类

  1. Inner 类定义在 Outer 类的内部,相当于 Outer 类的一个成员变量的位置,Inner 类可以使用任意访问控制符,如 public、protected、private 等。

  2. Inner 类中定义的 show() 方法可以直接访问 Outer 类中的数据,而不受访问控制符的影响 (包括 private)。

  3. 定义了成员内部类后,必须使用外部类对象来创建内部类对象,而不能直接去 new 一个内部类对象,即:内部类 对象名 = 外部类对象.new 内部类()
  4. 编译后会产生两个 .class 文件:Outer.classOuter$Inner.class{}
  5. 成员内部类中不能存在任何 static 的变量和方法,可以定义常量。
    • 因为非静态内部类是要依赖于外部类的实例,而静态变量和方法是不依赖于对象的,仅与类相关。简而言之:在加载静态域时,根本没有外部类,所在的非静态内部类中不能定义静态域或方法,编译不通过;非静态内部类的作用域是实例级别。
    • 常量是在编译器就确定的,放到了所谓的常量池中。

注意:

  • 外部类是不能直接访问内部类的方法和成员变量的,但是可以先创建内部类对象,然后通过内部类的对象来访问其成员变量和方法;
  • 如果外部类和内部类具有相同的成员变量或方法,内部类默认访问自己的成员变量或方法,如果要访问外部的,需要加 this 关键字,如 Outer.this.name

(三) 静态内部类

静态内部类是 static 修饰的内部类。特点:

  1. 静态内部类不能直接访问外部非静态类,但是可以通过 new 外部类().成员 的方式访问。
  2. 如果外部类的静态成员与内部类的成员名称相同,可以通过 类名.静态成员 访问外部的静态成员;如果不相同,可直接 成员名 访问外部的静态成员。
  3. 创建静态内部类的对象时,不需要外部类的对象,可直接创建 内部类 对象名 = new 内部名()

(四) 方法内部类

访问权限仅限于方法内或者该作用域内。

  1. 方法内部类就像是方法里面的一个局部变量,不能有 public、protected、private 以及 static 修饰符。
  2. 只能访问方法中定义的 final 类型的局部变量,因为:
    • 当方法被调用完毕后,局部变量就已消亡,但是内部类可能还存在,直到没有被引用的时才消亡。此时就会出现一种情况,内部类要访问一个不存在的局部变量。
    • 使用 final 修饰符不仅会保持对象的引用不会改变,而且编译器还会持续维护这个对象在回调方法中的生命周期。
    • 局部内部类并不是直接调用方法传进来的参数,而是内部类将传进来的参数通过自己的构造器备份到自己的内部。
    • 自己内部的方法调用的实际是自己的属性而不是外部类方法的参数。
    • 防止被篡改数据,而导致内部类得到的值不一致。

(五) 匿名内部类

  1. 匿名内部类是直接使用 new 来生成一个对象的引用。
  2. 对于匿名内部类的使用它是存在一个缺陷的,就是它只能被使用一次,创建匿名内部类时它会立即创建一个该类的实例,该类的定义会立即消失,所以匿名内部类是不能被重复使用的。
  3. 使用匿名内部类时,我们必须是继承一个类或实现一个接口,但是两者不可兼得,同时也只能继承一个类或者实现一个接口。
  4. 匿名内部类中是不能定义构造函数的,匿名内部类中不存在任何的静态成员变量和静态方法。
  5. 匿名内部类不能是抽象的,它必须要实现继承的类或者实现的接口的所有抽象方法。
  6. 匿名内部类初始化:使用构造代码块!利用构造代码块能够达到为匿名内部类创建一个构造器的效果。

 

三、char 可以存储汉字吗?

char 类型的变量是用来存储 Unicode 编码的字符的,Unicode 编码字符集中包含了汉字,所以,char 类型变量是可以存储汉字的。不过如果某个特殊的汉字没有包含在 Unicode 编码字符集中,那这个 char 类型变量就不能存储这个特殊的汉字。补充说明: Unicode 编码占用两个字节,所以 char 类型的变量也是占用两个字节。

 

四、Java 中的异常结构

 

1、Error 和 Exception

Error 是程序无法处理的错误,比如 OutOfMemmoryError、ThreadDeath 等。这些异常发生时,Java 虚拟机一般会选择线程终止。

Exception 是程序本身可以处理的异常,这种异常分两大类:运行时异常 (RunTimeException) 和非运行时异常。程序需要尽可能去处理这些异常。

2、运行时异常和非运行时异常

运行时异常都是 RunTimeException 类及其子类异常,如 NullPointerException、IndexOutOfBoundsException 等,这些异常是不检查异常,程序中是可以选择捕获处理的,也可以不处理。这些异常一般是由程序逻辑错误引起的,程序应该错逻辑角度尽可能避免这些异常的发生

非运行时异常是 RunTimeException 以外的异常,是受检查的异常 (checked exceptions),其必须被 try{} catch 语句捕获,或者在方法签名里通过 throws 子句声明。受检查的异常必须在编译时被捕捉处理,否则程序就不能编译通过。如 IOException、SQLException 等以及用户自定义的 Exception 异常,一般情况下不自定义检查异常。

 

posted @ 2019-06-12 21:00  意无尽  阅读(207)  评论(0编辑  收藏  举报