Java线程及线程安全
什么是线程
线程是进程的一个最小执行单位
一个程序下最少有一个进程,一个进程下至少有一个线程,或多个线程来增加程序的执行舒服
什么是守护线程
守护线程是运行在后台的一种特殊线程,独立于控制端并且周期性地指定某种任务或等待处理某些发生的事件
在java中垃圾回收线程就是特殊的守护线程
创建线程的方式
1.继承Thread重写run方法
2.实现Runnable接口,实现run方法,没有返回值
3.实现Callable接口,实现call方法,有返回值
线程的状态
NEW 尚未启动
RUNNABLE 就绪
RUNNING 运行
BLOCKED 阻塞的(被同步锁或者IO锁阻塞)
WAITING 永久等待
TIMED_WAITING 等待指定的时间重新被唤醒的状态
TERMINATED 执行完成
sleep和wait的区别
sleep:来自Thread,不释放锁,时间到会自动恢复
wait:来自object,释放锁,可以使用notify、notifyAll直接唤醒
在java程序中怎么保证多线程的运行安全
- 使用安全类:比如java.util.concurrent下的类
- 使用自动锁synchronized
- 使用手动锁Lock
悲观锁和乐观锁
悲观锁:总是假设最坏的情况,每次去拿数据的时候都认为别人会修改,所以每次在拿数据的时候都会上锁,这样别人想拿这个数据就会阻塞直到拿到锁(共享资源每次只给一个线程使用,其他线程阻塞,用完后再把资源转让给其他线程)。
适合用在写操作比较频繁的场景
Java中synchronized和ReentrantLock等独占锁就是悲观锁思想的实现
乐观锁: 总是假设最好的情况,每次去拿数据都认为别人不会修改,所以不会上锁,但是更新的时候会判断一下在此期间别人有没有去更新这个数据,可以使用版本号机制和CAS算法实现。
适用于多读的场景,这样可以提高吞吐量
Java中java.util.concurrent.atomic包下面的原子变量类就是使用了乐观锁的一种实现方式CAS实现的
Synchronized关键字
Java对象头里存在一个monitor, 叫监视器锁
Synchronized里面有两个指令:monitorenter monitorexit
monitorenter:每个对象有一个监视器锁(monitor),当monitor被占用时就会处于锁定状态,线程执行monitorenter指令时尝试获取monitor的所有权,过程如下:
- 如果monitor的进入数为0,则线程进入monitor,然后将进入数设置为1,该线程即为monitor的所有者
- 如果线程已经占用该monitor,只是重新进入,则进入monitor的进入数加1
- 如果其他线程已经占用了monitor,则该线程进入阻塞状态,直到monitor的进入数为0,再重新尝试获取monitor的所有权
Monitorexit:执行monitorexit的线程必须是objectref所对应的monitor的所有者,指令执行时,monitor的进入数减1,如果减1后进入数为0,那线程退出monitor,不再是这个monitor的所有者,其他被这个monitor阻塞的线程可以尝试去获取这个monitoer的所有权