四、多线程/并发
1.如何创建线程?如何保证线程安全?
创建线程有三种方法:
-
MyThread
继承Thread
, 然后new MyThread().start()
-
MyThread
实现Runnable
,然后new Thread(new MyThread()).start()
-
MyThread
实现Callable
,然后new Thread(new MyThread()).start()
保证线程安全 - 同步代码块,
synchronized(obj){ ... }
- 同步方法,
public synchronized void draw(int money){ ... }
- 同步锁,
Lock
,如ReadWriteLock
,ReentrantLock
2.如何实现一个线程安全的数据结构
- 使用锁,同步代码块,同步方法
- 使用
Collections.synchronizedMap()
等内置方法 - 使用标志位,进入方法了就置true,出去了就置false,这样子保证一个方法中只有一个在执行(有争议)
3.如何避免死锁
- 代码逻辑清晰
- suspend方法容易死锁,不要使用
4.Volatile关键字的作用?
5.HashMap在多线程环境下使用需要注意什么?为什么?
HashMap是线程的容器,在多线程环境中如果和多个线程都有交互,那么应该将其包装成线程安全的容器。Collections.synchronizedMap()
6.Java程序中启动一个线程是用run()还是start()?
new Thread().start()
@Override
public void run(){...}
7.什么是守护线程?有什么用?
Daemon的作用是为其他线程的运行提供服务,比如说GC线程。其实User Thread线程和Daemon Thread守护线程本质上来说去没啥区别的,唯一的区别之处就在虚拟机的离开:如果User Thread全部撤离,那么Daemon Thread也就没啥线程好服务的了,所以虚拟机也就退出了。
8.什么是死锁?如何避免
当两个线程相互等待对方释放同步监视器时就会发生死锁。
- 代码逻辑清晰
- suspend方法容易死锁,不要使用
9.线程和进程的差别是什么?
操作系统可以同时执行多个任务,每个任务就是进程;进程可以同时多个任务,每个任务就是线程。一个程序运行后至少有一个进程,一个进程里可以包含多个线程,但至少包含一个线程。
10.Java里面的Threadlocal是怎样实现的?
每个线程中都保留一个变量值的副本,使得每个线程都可以独立改变自己的副本,而不会和其他线程的副本冲突。
11.ConcurrentHashMap的实现原理是?
分段加锁,比起简单粗暴的Hashtable
效率高很多
12.sleep和wait区别
wait等待别人notify。
sleep自己到时间或者被中断了就继续。
13.notify和notifyAll区别
notify唤醒一个在此同步监视器上等待的线程,选择是任意的
notifyAll唤醒所有在此同步监视器上等待的线程。
14.volatile关键字的作用
简单说来就是,让其他高速缓存直接刷到内存中去。
参考这篇文章
15.ThreadLocal的作用与实现
每个线程中都保留一个变量值的副本,使得每个线程都可以独立改变自己的副本,而不会和其他线程的副本冲突。
16.两个线程如何串行执行
在A线程的run方法中启动B线程。
在A线程中join B线程
17.上下文切换是什么含义
上下文切换是存储和恢复CPU状态的过程,它使得线程执行能够从中断点恢复执行。上下文切换是多任务操作系统和多线程环境的基本特征。
18.可以运行时kill掉一个线程吗?
JPS能不能在进程里找到这个线程,然后kill了,不建议强制kill一个线程,最好是让其自然死亡,或者Interrupt它
19.什么是条件锁、读写锁、自旋锁、可重入锁?
从Java5开始,Java提供了一种功能更强大的线程同步机制,通过显示定义同步锁对象来实现同步,在这种机制下,同步锁由Lock对象充当。
Lock
和ReadWriteLock
是两个接口,ReentrantLock
(可重入锁)是Lock
的实现类,ReentrantReadWriteLock
是ReadWriteLock
的实现类。
ReentrantLock
锁具有可重入性,也就是说,一个线程可以对已被加锁的ReentrantLock
锁再次加锁,内部会维持一个计数器来追踪lock()
方法的潜逃调用。
20.线程池ThreadPoolExecutor的实现原理?
一个线程从被提交(submit)到执行共经历以下流程:
- 线程池判断核心线程池里是的线程是否都在执行任务,如果不是,则创建一个新的工作线程来执行任务。如果核心线程池里的线程都在执行任务,则进入下一个流程
- 线程池判断工作队列是否已满。如果工作队列没有满,则将新提交的任务储存在这个工作队列里。如果工作队列满了,则进入下一个流程。
- 线程池判断其内部线程是否都处于工作状态。如果没有,则创建一个新的工作线程来执行任务。如果已满了,则交给饱和策略来处理这个任务。
- 线程池在执行excute方法时,主要有以下四种情况
1 如果当前运行的线程少于corePoolSize,则创建新线程来执行任务(需要获得全局锁)
2 如果运行的线程等于或多于corePoolSize ,则将任务加入BlockingQueue
3 如果无法将任务加入BlockingQueue(队列已满),则创建新的线程来处理任务(需要获得全局锁)
4 如果创建新线程将使当前运行的线程超出maxiumPoolSize,任务将被拒绝,并调用RejectedExecutionHandler.rejectedExecution()方法。
目录列表
一、数据结构与算法基础
二、Java基础
三、JVM
四、多线程/并发
五、Linux使用与问题分析排查
六、框架使用
七、数据库相关
八、网络协议和网络编程
九、Redis等缓存系统/中间件/NoSQL/一致性Hash等
十、设计模式与重构
本文是针对知乎文章《成为Java顶尖程序员,先过了下面问题》的解答