补充:面试常见的问题之 Java 基础
Posted reformdai
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了补充:面试常见的问题之 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” 的关系,他就是一个独立的实体;
- 内部类提供了更好的封装,除了该外部类,其他类都不能访问。
(二) 成员内部类
-
-
Inner 类中定义的 show() 方法可以直接访问 Outer 类中的数据,而不受访问控制符的影响 (包括 private)。
- 定义了成员内部类后,必须使用外部类对象来创建内部类对象,而不能直接去 new 一个内部类对象,即:内部类 对象名 = 外部类对象.new 内部类()。
- 编译后会产生两个 .class 文件:Outer.class,Outer$Inner.class。
- 成员内部类中不能存在任何 static 的变量和方法,可以定义常量。
-
- 因为非静态内部类是要依赖于外部类的实例,而静态变量和方法是不依赖于对象的,仅与类相关。简而言之:在加载静态域时,根本没有外部类,所在的非静态内部类中不能定义静态域或方法,编译不通过;非静态内部类的作用域是实例级别。
- 常量是在编译器就确定的,放到了所谓的常量池中。
注意:
- 外部类是不能直接访问内部类的方法和成员变量的,但是可以先创建内部类对象,然后通过内部类的对象来访问其成员变量和方法;
- 如果外部类和内部类具有相同的成员变量或方法,内部类默认访问自己的成员变量或方法,如果要访问外部的,需要加 this 关键字,如 Outer.this.name。
(三) 静态内部类
静态内部类是 static 修饰的内部类。特点:
- 静态内部类不能直接访问外部非静态类,但是可以通过 new 外部类().成员 的方式访问。
- 如果外部类的静态成员与内部类的成员名称相同,可以通过 类名.静态成员 访问外部的静态成员;如果不相同,可直接 成员名 访问外部的静态成员。
- 创建静态内部类的对象时,不需要外部类的对象,可直接创建 内部类 对象名 = new 内部名()。
(四) 方法内部类
访问权限仅限于方法内或者该作用域内。
- 方法内部类就像是方法里面的一个局部变量,不能有 public、protected、private 以及 static 修饰符。
- 只能访问方法中定义的 final 类型的局部变量,因为:
- 当方法被调用完毕后,局部变量就已消亡,但是内部类可能还存在,直到没有被引用的时才消亡。此时就会出现一种情况,内部类要访问一个不存在的局部变量。
- 使用 final 修饰符不仅会保持对象的引用不会改变,而且编译器还会持续维护这个对象在回调方法中的生命周期。
- 局部内部类并不是直接调用方法传进来的参数,而是内部类将传进来的参数通过自己的构造器备份到自己的内部。
- 自己内部的方法调用的实际是自己的属性而不是外部类方法的参数。
- 防止被篡改数据,而导致内部类得到的值不一致。
(五) 匿名内部类
- 匿名内部类是直接使用 new 来生成一个对象的引用。
- 对于匿名内部类的使用它是存在一个缺陷的,就是它只能被使用一次,创建匿名内部类时它会立即创建一个该类的实例,该类的定义会立即消失,所以匿名内部类是不能被重复使用的。
- 使用匿名内部类时,我们必须是继承一个类或实现一个接口,但是两者不可兼得,同时也只能继承一个类或者实现一个接口。
- 匿名内部类中是不能定义构造函数的,匿名内部类中不存在任何的静态成员变量和静态方法。
- 匿名内部类不能是抽象的,它必须要实现继承的类或者实现的接口的所有抽象方法。
- 匿名内部类初始化:使用构造代码块!利用构造代码块能够达到为匿名内部类创建一个构造器的效果。
三、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 异常,一般情况下不自定义检查异常。
以上是关于补充:面试常见的问题之 Java 基础的主要内容,如果未能解决你的问题,请参考以下文章