虽然网上关于Java并发、多线程的文章已是不胜枚举,但是读起来总感觉晦涩且千篇一律,不是你转载我,就是我复制你。在慕课网学习了一套悟空老师的教程:http://www.imooc.com/learn/1086 ,受益匪浅,因此决定自己总结一下,以便自己常读常新,同时由于并发和多线程这块是面试重点中的重点,也是为了以后的面试做一下准备。如果有需要转载的小伙伴,只需要注明转载来源即可,珍惜博主劳动成果,再次感谢悟空老师提供的教程。
能够保证在同一时刻最多只有一个线程执行该段代码,以达到保证并发安全的效果。
1.Synchronized是Java的关键字,被Java语言原生支持
2.是最基本的互斥同步手段
3.是并发编程中的元老级角色,是并发编程的必学内容
包括方法锁(默认锁对象为this当前实例对象)和同步代码块锁(自己指定锁对象)
方法锁形式:synchronized修饰普通方法,锁对象默认为this
代码块形式:手动指定锁对象
指synchronized修饰静态的方法或指定锁为Class对象
概念(重要):
只有一个Class对象:Java类可能有很多个对象,但只有一个Class对象
本质:所谓的类锁,不过是Class对象的锁而已
用法和效果:类锁只能在同一时刻被一个对象拥有。
形式一:synchronized加在static方法上
形式二:synchronized(*.class)代码块
解决问题:两个线程同时a++,最后结果会比预计的少
原因
count++,它看上去只是一个操作,实际上包含了三个动作:
1.读取count
2.将count加一
3.将count的值写入到内存中
这三个操作,如果不按照原子去执行,就会带来并发问题
解决办法:
1.方法锁形式:在普通方法上加上sychronized关键字:
@Override
public synchronized void run() {
for (int j = 0; j < 100000; j++) {
i++;
}
}
2.同步代码块加锁:
@Override
public void run() {
synchronized (this) {
for (int j = 0; j < 100000; j++) {
i++;
}
}
}
1.静态方法上加锁:
@Override
public void run() {
count();
}
synchronized static void count() {
for (int j = 0; j < 100000; j++) {
i++;
}
}
2.synchronized(*.class)代码块:
@Override
public void run() {
synchronized (DisappearRequest1.class) {
for (int j = 0; j < 100000; j++) {
i++;
}
}
}
这种情况就是对象锁的方法锁情况。会相互等待,只能有一个线程持有锁。
不会加锁,因为访问的是不同的实例
这种情况就是类锁的静态方法锁。
synchronized关键字只作用于当前方法,不会影响其他未加关键字的方法的并发行为。因此非同步方法不受到影响,还是会并发执行。
synchronized关键字虽然没有指定所要的那个锁对象,但是本质上是指定了this这个对象作为它的锁。所以对于同一个实例来讲,两个方法拿到的是同一把锁,因此会出现串行的情况。
前者为类锁,锁为Class类;后者为对象锁,锁为this对象。因此两者的锁不同,会并行执行。
Lock类加锁时,如果出现异常,不显式手动释放锁的话,Lock是不会释放的。
而synchronized不同,一旦出现异常,会自动释放锁。
也就是说当第二个线程等待一个被synchronized修饰的方法时,若第一个线程出现异常退出,这把锁会立刻释放并且被第二个线程所获取到。JVM会自动把锁释放。
答案:不是的。出了本方法后,由于另外的方法没有被synchronized修饰,所以说这个方法可以被多个线程同时访问的。
1.一把锁只能同时被一个线程获取,没有拿到锁的线程必须等待(对应第1、5种情况);
2.每个实例都对应有自己的一把锁,不同实例之间互不影响;
例外:锁对象是*。class以及synchronized修饰的是static方法的时候,所有对象共用同一把类锁,这就是类锁的两种情况(对应第2、3、4、5种情况);
3.无论方法是正常执行完毕或者抛出异常,都会释放锁(对应第7种情况)。
只需要对这三点核心思想理解透彻了,所有情况都是这三点核心思想的实例化的表现。
什么是可重入:指的是同一线程的外层函数获得锁之后,内层函数可以直接再次获取该锁。也叫做递归锁。Java中两大递归锁:Synchronized和ReentrantLock
好比买车摇号,只要摇到一次号,可以给家里的第一辆车、第二辆车…所有车都上牌,直到我不需要上牌为止。这就叫做可重入性。如果每辆车上牌都需要摇一次号,这就叫做不可重入性。
锁的不可重入性:线程拿到一把锁了,如果想再次使用这把锁,必须先将锁释放,与其他线程再次进行竞争。
锁的可重入性:如果线程已经拿到了锁,试图去请求这个已经获得到的锁,而无需提前释放,直接就可以使用手里的这把锁,这就叫做可重入性
好处:避免死锁、提升封装性
1.假如有两个synchronized修饰的方法1和方法2,此时线程A执行到了方法1,并且获得了方法1的锁,此时方法1调用了方法2,由于方法2也是synchronized修饰的,假设synchronized不具备可重入性的话,那么线程A虽然拿到了方法1的锁,但是由于不可重入,它无法使用本身获得的方法1的这把锁。这样一来,它既想拿锁又不释放锁,这样就会永远等待,形成了死锁。所以由于synchronized具备可重入性,就避免了这种情况的发生。
2.避免了一次又一次的解锁加锁的过程,利用其可重入的性质提高了封装性,简化了并发编程的难度。
情况1:证明同一个方法是可重入的
情况2:证明可重入不要求是同一个方法
情况3:证明可重入不要求是同一个类中的
一旦这个锁已经被别的线程获得了,如果当前线程还想获得,只能选择等待或者阻塞,直到别的线程释放这个锁。如果别的线程 永远不释放锁,那么线程只能永远地等下去。
相比之下,Lock类,可以拥有中断的能力。
第一点,如果我觉得我等的时候太长了,有权中断现在已经获取到锁的线程的执行;
第二点,如果我觉得我等待的时间太长了不想再等了,也可以退出。
Lock比synchronized灵活很多,但是编码易出错。
每一个类的实例对应一把锁,而每一个synchronized方法都必须先获得调用该方法的类的实例的锁方能执行,否则线程会阻塞。而方法一旦执行,它就独占了这把锁,直到该方法返回或者是抛出异常,才将锁释放。一旦锁释放之后,之前被阻塞的线程才能获得这把锁,从被阻塞的状态重新进入到可执行的状态。当一个对象中有synchronized修饰的方法或者代码块的时候,要想执行这段代码,就必须先获得这个对象锁,如果此对象的对象锁已经被其他调用者所占用,就必须等待它被释放。所有的Java对象都含有一个互斥锁,这个锁由JVM自动去获取和释放,我们只需要指定这个对象就行了,至于锁的释放和获取不需要我们操心。
我们知道每一个Java对象都可以用作一个同步的锁,这个锁叫做内部锁,或者叫做监视器锁–monitor lock。线程在进入到同步代码块之前,会自动获得这个锁,并且在退出同步代码块的时候会自动的释放这个锁,无论是正常途径退出还是抛出异常退出。获得这个内置锁的唯一途径就是进入这个锁保护的同步代码块或者同步方法中。这样一来就理解了时机。
Lock lock = new ReentrantLock();
public synchronized void method1 () {
System.out.println("我是synchronized形式的锁");
}
public void method2() {
lock.lock();
try {
System.out.println("我是lock形式的锁");
} finally {
lock.unlock();
}
}
第一个方法和第二个方法等价,相当于把synchronized拆分成lock
在进入方法的时候回隐形的获取一把锁,等价于第二个方法的lock.lock();代码所做的事情
而在退出或者抛出异常的时候会释放锁,等价于lock.unlock();代码所做的事情
1.编写一个类:
package demo;
/**
* @Description: 反编译字节码
* @Author: jiangfan
* @CreateTime 2019/1/14 上午10:22
*/
public class Decompliation14 {
private Object object = new Object();
public void insert(Thread thread) {
synchronized (object) {
}
}
}
2.javac编译该类:
javac Decompliation14.java
3.反编译后:
javap -verbose Decompliation14.class
线程既可以在方法完成之后退出,也可以在抛出异常后退出,因此monitorexit数量多于monitorenter。
monitorenter:线程每次进入时,计数器+1。如果重入,继续加
monitorexit:线程退出时,计数器-1.变为0时候,其他线程可以获取锁。
1.每个对象都有一把锁,JVM负责跟踪对象被加锁的次数。
2.线程第一次给对象加锁的时候,计数变为1。每当这个相同的线程再次对象上再次获得锁时,计数会递增。只有首先获得这个锁的线程才能继续的在这个对象上多次的获取这把锁。
3.每当任务离开时,计数递减,当计数为0的时候,锁被完全释放。
线程A向线程B发送数据的两个步骤:
1.线程A修改了本地内存A,并将其存储到主内存中。
2.线程B再从主内存中读取出来。
这个过程是由JMM(Java Memory Model)控制的。JMM通过控制主内存与每个线程的本地内存的交互来为Java程序员提供内存可见性的保证。
一旦一个代码块或者方法被synchronized修饰之后,那么它在执行完毕之后被锁住的对象所做的任何修改都要在释放锁之前从线程内存写回到主内存中。所以下一个线程从主内存中读取到的数据一定是最新的。就是通过这样的原理,synchronized关键字保证了每一次执行都是可靠的,保证了可见性。
1.效率低:
2.不够灵活(读写锁更灵活:读操作的时候不会加锁,写操作的时候才会加锁):
3.无法知道是否成功获取到锁
Lock可以,如果尝试成功了做一些逻辑判断,如果没有成功做另外一些逻辑判断.
lock.lock();
lock.unlock();
通过这两个方法,可以手动加锁和解锁。
lock.tryLock();
lock.tryLock(10, TimeUnit.MINUTES);
可以判断是否加锁,返回类型为boolean
锁对象不能为空:锁的信息保存在对象头里面
作用域不宜过大:synchronized关键字包裹的范围。不需要串行工作的情况下,用并行的方式可以提高运行的效率
避免死锁
1.如果可以的情况下,两者都不要选择,而是使用java.util.concurrent包中的各种各样的类,例如:CountDownLatch等。使用这些类,不需要自己做同步工作,更方便,也更不容易出错。
2.如果synchronized关键字在程序中适用,就优先实用这个关键字。因为这样可以减少需要编写的代码,就减少了出错的几率。
3.如果特别需要Lock这样结构独有的特性的时候,才使用Lock。
以上三点主要是基于减少代码出错为出发点。
看4.多线程访问同步方法的七种情况
锁调度机制。对于synchronized内置锁,不同版本的JVM处理方式不同,blocked和running都有几率
自己实现一个Lock
在之前的JVM版本中,synchronized性能不是特别的好,而经过不断的迭代,synchronized性能已经得到了显著的提高,这里面运用的技术就是偏斜锁、轻量级锁、重量级锁。JVM会根据synchronized关键字使用到的次数或者其他的种种指标对锁进行有效的优化使得性能得到大幅上涨,这里面还涉及到了对象头里面的字段。
一句话介绍synchronized:
JVM会自动通过使用monitor来加锁和解锁,保证了同时只有一个线程可以执行指定代码,从而保证了线程安全,同时具有可重入性和不可中断的性质。