1. 进程和线程之间有什么不同?
- 一个进程是一个独立的运行环境, 它可以被看作一个程序或者一个应用. 而线程是在进程中执行一个任务. Java运行环境是一个包含了不同类和程序的单一进程. 线程可以以被成为轻量级进程. 线程需要较少的资源来创建和驻留在进程中, 并且可以共享进程中的资源.
2. 多线程编程的好处是什么?
- 在多线程程序中, 多个线程并发的执行以提高程序的效率, CPU不会因为某个线程需要等待资源而进入空闲状态. 多个线程共享堆内存,因此创建多线程去执行一些任务会哔创建多个进程更好. 举个栗子, Servlet比CGI更好, 因为Servlet支持多线程而CGI不支持.
3. 用户线程和守护线程有什么区别?
- 当我们在Java程序中创建一个线程, 它就被成为用户线程. 一个守护线程是在后台执行并且不会阻止JVM终止的线程. 当没有用户线程在运行的时候, JVM关闭程序并且退出. 一个守护线程创建的自线程依然是守护线程
setDaemon(true)
4. 如何创建一个线程?
- 实现
Runnable
接口, 然后将他传递给Thread
的构造函数, 创建一个Thread
对象 - 直接继承
Thread
类
5. 有哪些不同的线程生命周期?
- 当我们在Java程序中创建一个新的线程时, 它的状态是New, 当我们调用线程的
start()
方法时, 状态改变为Runnable, 线程调度器会为Runnable线程池中的线程分配CPU时间并且将他们的状态改变为Running. 其他的线程状态还有Waitng, Blocked, 和Dead
6. 可以直接调用Thread类的run()
方法吗?
- 可以, 但是如果我们调用了
Thread
的run()
方法, 它的行为就会和普通的方法一样, 为了在新的线程执行我们的代码, 必须使用Thread.start()
方法
7. 如何让正在运行的线程暂停一段时间?
- 可以使用
Thread.sleep()
方法让线程暂停一段时间. 需要注意的是, 这并不会让线程终止, 一旦从休眠中唤醒线程, 线程的状态将会被改变为Runnable, 并且根据线程调度, 它将得到执行
8. 你对线程优先级的理解是什么?
- 每一个线程都是有优先级的, 一般来说, 高优先级的线程在运行时会具有优先权, 但这依赖于线程调度的实现, 这个实现是和操作系统相关的. 我们可以定义线程的优先级
getPriority() setPriority()
, 但是这并不能保证高优先级的线程会在低优先级的线程前执行. 县城优先级是一个int变量, 1代表最低, 10代表最高
9. 什么是线程调度器(Thread Scheduler)和时间分片(Time Slicing)?
- 线程调度器是一个操作系统服务, 它负责为Runnable状态的线程分配CPU时间, 一旦我们创建一个线程并启动它, 它的执行便依赖于线程调度器的实现. 时间分片是指将可用的CPU时间分配给可用的Runnable线程的过程. 分配的CPU时间可以基于线程的优先级或者线程的等待时间, 线程调度并不受到Java虚拟机控制, 所以由应用程序来控制它是更好的选择(也就是说不要让你的程序依赖于线程优先级)
10. 在多线程中, 什么是上下文切换?
- 上下文切换是存储和回复CPU状态的过程, 它使得线程执行能够从中断点回复执行.上下文切换是多任务操作系统和多线程环境的基本特征.
11. 如何确保main()
方法所在的线程是Java程序最后结束的线程?
- 我们可以使用
Thread
类的joint()
方法来确保所有程序创建的线程在main()
方法退出前结束.
12. 线程之间是如何通信的?
- 当线程间是可共享资源时, 线程间通信是协调他们的重要手段.
Object
类中wait()
/notify()
/notifyAll()
方法可以用于线程间通信关于资源的锁的状态
13. 为什么线程通信的方法wait()
/notify()
/notifyAll()
被定义在Object
类里?
- Java中每个对象中都有一个锁(monitor, 也称为监视器) 并且
wait()
/notify()
/notifyAll()
方法用于等待对象的锁或者通知其他线程对象的监视器可用. 在Java线程中并没有可供任何对象使用的锁和同步器. 这就是为什么这些方法是Object
类的一部分, 这样Java的每一个类都有用于线程间通信的基本方法.
14. 为什么wait()
/notify()
/notifyAll()
必须在同步方法或者同步块中被调用?
- 当一个线程需要调用对象的
wait()
方法的时候, 这个线程必须拥有该对象的锁, 接着它就会释放这个对象锁并进入等待状态知道其他线程调用这个对象的notify()
方法时, 它会释放这个对象的锁, 以便其他在等待的线程可以得到这个对象锁. 由于所有的这些方法都需要线程持有对象的锁, 这样就只能通过同步来实现, 所以他们只能在同步方法或者同步块中被调用.
15. 为什么Thread
类的sleep()
和yield()
方法是静态的?
-
Thread
类的sleep()
和yield()
方法将在当前正在执行的线程上运行, 所以在其他处于等待状态的线程上调用这些方法是没有意义的, 这就是为什么这些方法是静态的. 他们可以在当前正在执行的线程中工作, 避免程序员错误的认为可以在其他非运行线程调用这些方法.
16. 如何确保线程安全?
- 在Java中可以有很多方法来保证线程安全---同步, 使用原子类(atomic concurrent classes), 实现并发锁, 使用
volatile
关键字, 使用不变类和线程安全类
17. volatile
关键字在Java中有什么作用?
- 当我们使用
volatile
关键字去修饰变量的时候, 所有线程都会直接读取该变量并且不缓存它, 这就确保了线程读取到的变量是同内存中一致的.
18. 同步方法和同步块, 哪个是更好的选择?
- 同步块是更好的选择, 因为它不会锁住整个对象(当然你也可以让它锁住整个对象) . 同步方法会锁住整个对象, 哪怕这个类中有多个不相关联的同步块, 这通常会导致他们停止执行并且需要等待获得这个对象上的锁.
19. 如何创建守护线程?
- 使用
Thread
类的setDaemon(true)
方法可以讲线程设置为守护线程, 需要注意的是, 需要在调用start()
方法前调用这个方法, 否则会抛出IllegalThreadStateException
异常.
20. 什么是ThreadLocal?
ThreadLocal用于创建线程的本地变量, 我们知道一个独享的所有线程会共享它的全局变量, 所以这些变量不是线程安全的, 我们可以使用同步技术,. 但当我们不想使用同步的时候, 我们可以选择ThreadLocal变量.
每个线程都会拥有他们自己的Thread变量, 它们可以使
get()/set()
方法获取他们的默认值或者在线程内部改变他们的值, ThreadLocal实例通常是希望他们
同线程状态关联起来是private static属性.
21. 什么是ThreadGroup
? 为什么不建议使用它?
ThreadGroup
是一个类, 它的目的是提供关于线程组的信息.ThreadGroup
API比较薄弱, 它并没有哔Thread提供了更多的功能. 它主要有两个功能: 一是获取县城组中处于活跃状态线程的列表; 二是甚至为线程设置未捕获异常处理器(uncaught exception handler) . 但在java 1.5中Thread
类也添加了setUncaughtExceptionHandler(UncaughtExceptionHandler eh)
方法, 所以ThreadGroup
是已经过时的, 不建议使用.
22. 什么是死锁(Deadlock)? 如何分析和避免死锁?
- 死锁是指两个线程永远阻塞的情况, 这种情况产生至少需要两个以上的线程和两个以上的资源.
- 分析死锁, 我们需要查看Java应用程序的线程转储, 我们需要找出那些状态为Blocked的线程和他们等待的资源. 每个资源都有 一个唯一的id, 用这个id我们可以找出哪些线程已经拥有了它的对象锁.
- 避免嵌套锁, 只在需要的地方使用锁和避免无限期等待是避免死锁的通常方法.
23. 什么是Java Timer类?如何创建一个有特定时间间隔的任务?
-
java.util.Timer
是一个工具类, 可以用于安排一个线程在未来的某个特定时间执行.Timer
类可以用安排一次性任务或者周期性任务. -
java.util.TimerTask
是一个实现了Runnable
接口的抽象类, 我们需要去继承这个类来创建我们自己的定时任务并使用Timer
安排它的执行.
24. 什么是线程池?如何创建一个Java线程池?
- 一个线程池管理了一组工作线程, 同时它还包括了一个用于放置等待执行的任务的队列.
-
java.util.concurrent.Executors
提供了一个java.util.concurrent.Executor
接口用于创建线程池.