java容许多线程并发控制,当多个线程同时操做一个可共享的资源变量时,将会致使数据不许确,相互之间产生冲突,所以加入同步锁以免该线程没有完成操做以前,被其余线程调用,从而保证数据的惟一性和准确性。 1.同步方法,即为synchronized关键字修饰的方法。 public sunchronized void save(){} 例如该方法,因为java的每个内置对象都有一个内置锁,当用此关键字修饰方法是,内置锁会保护整个方法。在调用该方法前,须要得到内置锁,不然就处于阻塞状态。 注意:synchronized关键字也能够修饰静态方法,此时若是调用静态方法,将会锁住整个类。java
2.同步代码块,被关键字修饰的语句块会自动被加上内置锁,从而实现同步 synchronized(object){} 同步代码块和同步方法不一样,同步代码块必须有锁。 注意:同步是一种高开销的操做,所以应该尽可能减小同步的内筒,一般咱们优先选择同步代码块来同步关键代码便可。面试
3.使用volatile关键字修饰变量来实现线程同步多线程
peivate volatile int account = 100; a.volatile关键字为域变量的访问提供了一种免锁机制 b.使用volatile修饰域至关于告诉虚拟机该域有可能会被其余线程更新,所以每次使用该域就要从新计算,而不是使用寄存器中的值。volatitle不会提供任何原子操做,他也不能用来修饰final类型的变量。并发
4.使用重入锁实现线程同步 ReentrantLock类能够重入,互斥,实现了Lock接口的锁, 该类的经常使用方法 lock() 得到锁 unlock(): 释放锁ide
class Bank { private int account = 100; //须要声明这个锁 private Lock lock = new ReentrantLock(); public int getAccount() { return account; } //这里再也不须要synchronized public void save(int money) { lock.lock(); try{ account += money; }finally{ lock.unlock(); } } }
注:关于Lock对象和synchronized关键字的选择:线程
a.最好两个都不用,使用一种java.util.concurrent包提供的机制, 可以帮助用户处理全部与锁相关的代码。 b.若是synchronized关键字能知足用户的需求,就用synchronized,由于它能简化代码 c.若是须要更高级的功能,就用ReentrantLock类,此时要注意及时释放锁,不然会出现死锁,一般在finally代码释放锁
5.使用局部变量实现线程同步 若是使用ThreadLocal管理变量,则每个人使用该变量的线程都得到该变量的副本。 副本之间相互独立,这样每个线程均可以随意修改本身的变量副本,而不会对其余线程产生影响。 ThreadLocal 类的经常使用方法 ThreadLocal() : 建立一个线程本地变量code
get() : 返回此线程局部变量的当前线程副本中的值 initialValue() : 返回此线程局部变量的当前线程的"初始值" set(T value) : 将此线程局部变量的当前线程副本中的值设置为value
public class Bank{ //使用ThreadLocal类管理共享变量account private static ThreadLocal<Integer> account = new ThreadLocal<Integer>(){ @Override protected Integer initialValue(){ return 100; } }; public void save(int money){ account.set(account.get()+money); } public int getAccount(){ return account.get(); } }
面试题:一个线程进入对象的synchronized方法A以后,其余线程是否能够进去此对象的synchronized的方法B? 回答:不能够,所对象没有释放,其余线程不能进入方法B。对象