锁:
1.synchronized(重量级锁)多线程并发编程。
JDK1.6为了减少获得锁和释放锁带来的性能消耗引入的偏向锁和轻量级锁。
拓展:
1.无锁:不加锁
2.偏向锁:不锁锁,只有一个线程争夺时,偏向某一个线程,这个线程不加锁
3.轻量级锁:少量线程来了之后,向尝试自旋,不挂起线程。
4.重量级锁:排队挂起(暂停)线程。(synchronized)
synchronized有三种方式来加锁,分别是:
1.修饰实例方法,作用于当前实例加锁,进入同步代码前要获得当前实例的锁
2.静态方法,作用于当前类对象加锁,进入同步代码前要获得的当前类对象的锁
3.代码块,指定加锁对象,对给定对象加锁,进入同步代码块之前要获得给定对象的锁
1.实例方法:调用该方法的实例
2.静态方法:类对象
3.this:调用该方法的实例对象
4.类对象:类对象
操作共享数据的代码
共享数据:多个线程共同操作的变量,都可以充当锁
当使用同步方法时,synchronized锁的东西是this(默认的)
1.同步方法:
1.同步方法依然涉及到同步锁对象,不需要我们写出来
2.非静态的同步方法,同步锁就是this
3.静态的同步方法,同步监视器就是类本身
2.同步代码块:
1.选好同步监视器(锁)推荐使用类对象,第三方对象,this
2.在实现接口创建的线程类中,同步代码块不可以用this来充当同步锁
3.同步的方式,解决线程安全的问题。
1.操作同步代码时,只有一个线程能够参与,其他线程等待。
2.相当于一个单线程的过程,效率低。
synchronized只针对于当前JVM可以解决线程安全问题。
synchronized不可以跨JVM解决问题!!!
死锁:
多个线程同时被阻塞,他们中的一个或者全部都在等待某个资源的释放,由于线程无限期的阻塞,程序就不可能正常终止。
死锁产生四个必要条件:
1.互斥使用,当资源被一个线程使用(占用),别的线程不能使用。
2.不可抢占,资源请求者不能强制从占有者中抢夺资源,资源只能从占有者
手动释放
3.请求和保持,
4.循环等待,存在一个等待队列。P1占有P2的资源,P2占有了P3的资源
P3占有P1的资源。形成了一个等待环路。
线程重入:
任意线程在拿到锁之后,再次获取该锁不会被该锁所阻碍
线程不会被自己锁死的。
这就叫线程的重入,synchronized可重入锁。
挂起线程和恢复线程需要转入内核态中完成这些操作,给系统的并发性带来很大的压力。
在许多应用上共享数据的锁定状态,只会持续很短的时间,为了这段时间去挂起和恢复并不值得
我们可以让后面的线程等待一下,不要放弃处理器的执行时间。锁为了让线程等待,我们只需要让线程执行一个循环,自旋。【自旋锁】
等待和唤醒:
wait(): | 等待 |
wait(long timeout): | 当前线程进入等待状态 |
notify(): | 唤醒正在等待的下一个线程 |
notifyAll(): | 唤醒正在等待的所有线程 |
线程间的通信
比如两条线程,共同运行。
线程A如果先走,线程B就要等待。等待线程A走完,唤醒线程B,线程B再走
方法总结:
1.Thread的两个静态方法:
sleep释放CPU资源,但是不会释放锁
yield方法释放CPU执行权,保留了CPU的执行资格,不常用。
2.join方法,yield出让了执行权,join就加入进来。
3.wait:释放CPU资源,释放锁
notify:唤醒等待中的线程
notifyAll:唤醒等待中的所有线程
线程的常用方法:
Thread类中的方法:
方法名称 | 是否static | 方法说明 |
---|---|---|
start() | 否 | 让线程启动,进入就绪状态,等待cpu分配时间片 |
run() | 否 | 重写Runnable接口的方法,线程获取到cpu时间片时执行的具体逻辑 |
yield() | 是 | 线程的礼让,使得获取到cpu时间片的线程进入就绪状态,重新争抢时间片 |
sleep(time) | 是 | 线程休眠固定时间,进入阻塞状态,休眠时间完成后重新争抢时间片,休眠可被打断 |
join()/join(time) | 否 | 调用线程对象的join方法,调用者线程进入阻塞,等待线程对象执行完或者到达指定时间才恢复,重新争抢时间片 |
isInterrupted() | 否 | 获取线程的打断标记,true:被打断,false:没有被打断。调用后不会修改打断标记 |
interrupt() | 否 | 打断线程,抛出InterruptedException异常的方法均可被打断,但是打断后不会修改打断标记,正常执行的线程被打断后会修改打断标记 |
interrupted() | 否 | 获取线程的打断标记。调用后会清空打断标记 |
stop() | 否 | 停止线程运行 不推荐 |
suspend() | 否 | 挂起线程 不推荐 |
resume() | 否 | 恢复线程运行 不推荐 |
currentThread() | 是 | 获取当前线程 |
Object中的方法:
方法名称 | 方法说明 |
---|---|
wait()/wait(long timeout) | 获取到锁的线程进入阻塞状态 |
notify() | 随机唤醒被wait()的一个线程 |
notifyAll(); | 唤醒被wait()的所有线程,重新争抢时间片 |
interrupt方法:中断线程。
调用interrupt方法会抛出InterruptedException异常,
捕获后再做停止线程的逻辑即可。
如果线程while(true)运行的状态,interrupt方法无法中断线程。