1.1 线程安全
当多个线程访问某一个类(或对象或方法)时,这个类始终能表现出正确的行为,则称其为线程安全的。
synchronized:可在任意对象或方法上加锁,而加锁的这段代码称为“互斥区”或“临界区”。
锁竞争,就是说多个线程会抢锁,拿到锁才能执行synchronized修饰的。
而当线程试图获取一个内部的对象锁,而该锁被其他线程持有,则该线程进入阻塞状态。
当一个线程被阻塞、等待,另一个线程被调度为运行状态,然后那个线程被重新激活(等待时间到、成功获得一个锁),线程调度器检查它是否具有比运行线程更高的优先级,如果有,则从当前运行的线程中挑选一个,剥夺运行权,选择新线程运行。
1.2 多个线程多个锁
每一个线程都能拿到自己指定的锁,拿到锁后执行synchronized方法体。
synchronized取得的锁都是对象锁,而不是把一个方法当做锁,所以哪个线程先执行的synchronized的方法,哪个线程就持有该方法所属对象的锁。两个对象,线程获得的就是两个不同的锁,互不影响。
特殊的,在静态方法上加synchronized,表示锁定类。
1.3 对象锁的同步与异步
同步:synchronized
异步:asynchronized
同步的目的就是为了线程安全。线程安全满足两个特性:原子性(同步)、可见性。
线程1先持有对象锁,线程2如果访问同步方法,则会等待,就是同步。
线程1先持有对象锁,线程2如果可以以异步的方式访问非同步方法。
1.4 脏读
在对对象的方法加锁的时候要保证业务的整体性,比如set与get同时加锁,要保证业务的原子性。
1.5 synchronized其他
synchronized拥有锁重入功能,就是说在使用synchronized时,当一个线程得到了一个对象锁后,再次请求此对象是可以再次获得该对象锁的。
如果出现异常,锁会自动释放。比如在执行队列,很多线程都等着一个线程执行完释放锁,但是由于出现异常,并没有正常执行完便释放了锁,导致后面的业务逻辑全部出错。
1.6 synchronized代码块
使用synchronized声明的方法在某些情况下是有弊端的,比如线程A使用同步的方法执行了很长时间,线程B就等待了很长时间,这种情况下可以使用synchronized代码块去优化,也就是通常说的减小锁的粒度。
1.7 volatile关键字
volatile可以使变量在多个线程间可见。
在Java中,每个线程都有自己的工作内存区,其中存放着 所有线程可共享的主内存中变量的拷贝。在线程执行的时候,操作的是自己工作区的变量。当线程解锁时保证该工作区的变量的值写回共享的主内存中。
注:system.out.println()也是线程操作,也会抢占式执行。
atomic原子操作,如AtomicInteger,但是多个原子操作在同一个方法中的话就没法保证原子性了,需要配合synchronized来保证。
2.1 线程之间的通信