1、什么时候用断言(assert)?

答:

断言在软件开发中是一种常用的调试方式,很多开发语言中都支持这种机制。一

般来说,断言用于保证程序最基本、关键的正确性。断言检查通常在开发和测试

时开启。为了保证程序的执行效率,在软件发布后断言检查通常是关闭的。断言

是一个包含布尔表达式的语句,在执行这个语句时假定该表达式为 true;如果表

达式的值为 false,那么系统会报告一个 AssertionError。

断言的使用如下面的代码所示:

assert(a > 0); // throws an AssertionError if a <= 0

断言可以有两种形式:

assert Expression1;

assert Expression1 : Expression2 ;

Expression1 应该总是产生一个布尔值。

Expression2 可以是得出一个值的任意表达式;这个值用于生成显示更多调试信

息的字符串消息。

要在运行时启用断言,可以在启动 JVM 时使用-enableassertions 或者-ea 标记。

要在运行时选择禁用断言,可以在启动 JVM 时使用-da 或者-disableassertions

标记。要在系统类中启用或禁用断言,可使用-esa 或-dsa 标记。还可以在包的基

础上启用或者禁用断言。

注意: 断言不应该以任何方式改变程序的状态。简单的说,如果希望在不满足某

些条件时阻止代码的执行,就可以考虑用断言来阻止它。

2、Error 和 Exception 有什么区别?

答:

Error 表示系统级的错误和程序不必处理的异常,是恢复不是不可能但很困难的情

况下的一种严重问题;比如内存溢出,不可能指望程序能处理这样的情况;

Exception 表示需要捕捉或者需要程序进行处理的异常,是一种设计或实现问题;

也就是说,它表示如果程序运行正常,从不会发生的情况。

3、Java 语言如何进行异常处理,关键字:throws、throw、try、catch、finally 分别如何使用?

答:

Java 通过面向对象的方法进行异常处理,把各种不同的异常进行分类,并提供了

良好的接口。在 Java 中,每个异常都是一个对象,它是 Throwable 类或其子类

的实例。当一个方法出现异常后便抛出一个异常对象,该对象中包含有异常信息,

调用这个对象的方法可以捕获到这个异常并可以对其进行处理。Java 的异常处理

是通过 5 个关键词来实现的:try、catch、throw、throws 和 finally。一般情况

下是用 try 来执行一段程序,如果系统会抛出(throw)一个异常对象,可以通过

它的类型来捕获(catch)它,或通过总是执行代码块(finally)来处理;try 用

来指定一块预防所有异常的程序;catch 子句紧跟在 try 块后面,用来指定你想要

捕获的异常的类型;throw 语句用来明确地抛出一个异常;throws 用来声明一个

方法可能抛出的各种异常(当然声明异常时允许无病呻吟);finally 为确保一段

代码不管发生什么异常状况都要被执行;try 语句可以嵌套,每当遇到一个 try 语

句,异常的结构就会被放入异常栈中,直到所有的 try 语句都完成。如果下一级的

try 语句没有对某种异常进行处理,异常栈就会执行出栈操作,直到遇到有处理这

种异常的 try 语句或者最终将异常抛给 JVM。

4、运行时异常与受检异常有何异同?

答:

异常表示程序运行过程中可能出现的非正常状态,运行时异常表示虚拟机的通常

操作中可能遇到的异常,是一种常见运行错误,只要程序设计得没有问题通常就

不会发生。受检异常跟程序运行的上下文环境有关,即使程序设计无误,仍然可

能因使用的问题而引发。Java 编译器要求方法必须声明抛出可能发生的受检异常,

但是并不要求必须声明抛出未被捕获的运行时异常。

异常和继承一样,是面向对象程序设计中经常被滥用的东西,在 Effective Java 中对异常的使用给出了以下指导原则:

不要将异常处理用于正常的控制流(设计良好的 API 不应该强迫它的调用者为了正常的控制流而使用异常)

对可以恢复的情况使用受检异常,对编程错误使用运行时异常

避免不必要的使用受检异常(可以通过一些状态检测手段来避免异常的发生)

优先使用标准的异常

每个方法抛出的异常都要有文档

保持异常的原子性

不要在 catch 中忽略掉捕获到的异常

5、列出一些你常见的运行时异常?

ArithmeticException(算术异常)

ClassCastException (类转换异常)

IllegalArgumentException (非法参数异常)

IndexOutOfBoundsException (下标越界异常)

NullPointerException (空指针异常)

SecurityException (安全异常)

6、阐述 final、finally、finalize 的区别。

答:

final:

修饰符(关键字)有三种用法:如果一个类被声明为 final,意味着它不能再派生出新的子类,即不能被继承,因此它和 abstract 是反义词。将变量声明为 final,可以保证它们在使用中不被改变,被声明为 final 的变量必须在声明时给定初值,而在以后的引用中只能读取不可修改。被声明为 final 的方法也同样只能使用,不能在子类中被重写。

finally:

通常放在 try…catch…的后面构造总是执行代码块,这就意味着程序无论正常执行还是发生异常,这里的代码只要 JVM 不关闭都能执行,可以将释放外部资源的代码写在 finally 块中。

finalize:

Object 类中定义的方法,Java 中允许使用 finalize()方法在垃圾收集器将对象从内存中清除出去之前做必要的清理工作。这个方法是由垃圾收集器在销毁对象时调用的,通过重写 finalize()方法可以整理系统资源或者执行其他清理工作。

7、阐述 ArrayList、Vector、LinkedList 的存储性能和特性。

答:

ArrayList 和 Vector 都是使用数组方式存储数据,此数组元素数大于实际存储的

数据以便增加和插入元素,它们都允许直接按序号索引元素,但是插入元素要涉

及数组元素移动等内存操作,所以索引数据快而插入数据慢,Vector 中的方法由

于添加了 synchronized 修饰,因此 Vector 是线程安全的容器,但性能上较

ArrayList 差,因此已经是 Java 中的遗留容器。LinkedList 使用双向链表实现存

储(将内存中零散的内存单元通过附加的引用关联起来,形成一个可以按序号索

引的线性结构,这种链式存储方式与数组的连续存储方式相比,内存的利用率更

高),按序号索引数据需要进行前向或后向遍历,但是插入数据时只需要记录本

项的前后项即可,所以插入速度较快。Vector 属于遗留容器(Java 早期的版本中

提供的容器,除此之外,Hashtable、Dictionary、BitSet、Stack、Properties

都是遗留容器),已经不推荐使用,但是由于 ArrayList 和 LinkedListed 都是非

线程安全的,如果遇到多个线程操作同一个容器的场景,则可以通过工具类

Collections 中的 synchronizedList 方法将其转换成线程安全的容器后再使用(这

是对装潢模式的应用,将已有对象传入另一个类的构造器中创建新的对象来增强

实现)。

补充: 遗留容器中的 Properties 类和 Stack 类在设计上有严重的问题,Properties

是一个键和值都是字符串的特殊的键值对映射,在设计上应该是关联一个

Hashtable 并将其两个泛型参数设置为 String 类型,但是 Java API 中的

Properties 直接继承了 Hashtable,这很明显是对继承的滥用。这里复用代码的方式应该是 Has-A 关系而不是 Is-A 关系,另一方面容器都属于工具类,继承工具类本身就是一个错误的做法,使用工具类最好的方式是 Has-A 关系(关联)或Use-A 关系(依赖)。同理,Stack 类继承 Vector 也是不正确的。Sun 公司的工程师们也会犯这种低级错误,让人唏嘘不已。

8、Collection 和 Collections 的区别?

答:

Collection 是一个接口,它是 Set、List 等容器的父接口;Collections 是个一个

工具类,提供了一系列的静态方法来辅助容器操作,这些方法包括对容器的搜索、

排序、线程安全化等等。

9、TreeMap 和 TreeSet 在排序时如何比较元素? Collections 工具类中的 sort()方法如何比较元素?

答:

TreeSet 要求存放的对象所属的类必须实现 Comparable 接口,该接口提供了比

较元素的 compareTo()方法,当插入元素时会回调该方法比较元素的大小。

TreeMap 要求存放的键值对映射的键必须实现 Comparable 接口从而根据键对元

素进行排序。Collections 工具类的 sort 方法有两种重载的形式,第一种要求传入

的待排序容器中存放的对象比较实现 Comparable 接口以实现元素的比较;第二

种不强制性的要求容器中的元素必须可比较,但是要求传入第二个参数,参数是

Comparator 接口的子类型(需要重写 compare 方法实现元素的比较),相当于

一个临时定义的排序规则,其实就是通过接口注入比较元素大小的算法,也是对

回调模式的应用(Java 中对函数式编程的支持)。

例子 1:

public class Student implements Comparable<Student> {
    private String name; // 姓名
    private int age; // 年龄
    public Student(String name, int age) {
        this.name = name;
        this.age = age;
    }
    @Override
    public String toString() {
        return "Student [name=" + name + ", age=" + age + "]";
    }
    @Override
    public int compareTo(Student o) {
        return this.age - o.age; // 比较年龄(年龄的升序)
    }
}
import java.util.Set;
import java.util.TreeSet;
class Test01 {
    public static void main(String[] args) {
        Set<Student> set = new TreeSet<>(); // Java 7 的钻石语法
        (构造器后面的尖括号中不需要写类型)
                set.add(new Student("Hao LUO", 33));
        set.add(new Student("XJ WANG", 32));
        set.add(new Student("Bruce LEE", 60));
        set.add(new Student("Bob YANG", 22));
        for(Student stu : set) {
            System.out.println(stu);
        }
// 输出结果:
// Student [name=Bob YANG, age=22]
// Student [name=XJ WANG, age=32]
// Student [name=Hao LUO, age=33]
// Student [name=Bruce LEE, age=60]
    }
}

10、Thread 类的 sleep()方法和对象的 wait()方法都可以让线程暂停执行,它们有什么区别?

答:

sleep()方法(休眠)是线程类(Thread)的静态方法,调用此方法会让当前线程

暂停执行指定的时间,将执行机会(CPU)让给其他线程,但是对象的锁依然保

持,因此休眠时间结束后会自动恢复(线程回到就绪状态,请参考第 66 题中的线

程状态转换图)。wait()是 Object 类的方法,调用对象的 wait()方法导致当前线

程放弃对象的锁(线程暂停执行),进入对象的等待池(wait pool),只有调用

对象的 notify()方法(或 notifyAll()方法)时才能唤醒等待池中的线程进入等锁池

(lock pool),如果线程重新获得对象的锁就可以进入就绪状态。

补充: 可能不少人对什么是进程,什么是线程还比较模糊,对于为什么需要多线

程编程也不是特别理解。简单的说:进程是具有一定独立功能的程序关于某个数

据集合上的一次运行活动,是操作系统进行资源分配和调度的一个独立单位;线

程是进程的一个实体,是 CPU 调度和分派的基本单位,是比进程更小的能独立运

行的基本单位。线程的划分尺度小于进程,这使得多线程程序的并发性高;进程

在执行时通常拥有独立的内存单元,而线程之间可以共享内存。使用多线程的编

程通常能够带来更好的性能和用户体验,但是多线程的程序对于其他程序是不友

好的,因为它可能占用了更多的 CPU 资源。当然,也不是线程越多,程序的性能

就越好,因为线程之间的调度和切换也会浪费 CPU 时间。时下很时髦的 Node.js

就采用了单线程异步 I/O 的工作模式。