当使用多线程访问同一个资源的时候,非常容易出现线程安全的问题(当多个线程同时对一个数据进行修改的时候,会导致某些线程对数据的修改丢失)。因此,需要采用同步机制来解决这种问题。
synchronized关键字
每个对象都有一个对象锁与之相关联,该锁表明对象在任何时候只允许被一个线程锁拥有,当一个线程调用对象的一段synchronized代码时,需要先获取这个锁,然后去执行相应的代码,执行结束之后,释放锁。
synchronized关键字主要有两种用法: synchronized方法 和 synchronized 块。此外,这个关键字还可以作用于静态方法、类或者某个实例,但这都 对程序的效率有很大的影响
synchronized方法
在方法的声明前主要有synchronized关键字,示例如下:
public synchronized void mutiThreadAccess();
只要把多个线程对类需要被同步的资源的操作放到mutiThreadAccess()方法中,就能保证这个方法在同一时刻只能被同一个线程访问,从而保证了多线程访问的安全性。然而,当一个方法的方法体规模非常大时,把该方法声明为synchronized会大大影响程序的执行效率。为了提高程序的效率,Java提供了synchronized块
synchronized块
synchronized块既可以把任意的代码段声明为synchronized,也可以指定上锁的对象,有非常高的灵活性。其用法如下:
synchronized(syncObject){
//访问synchObject的代码
}
Lock
JDK5新增加了Lock接口以及它的一个实现类ReentrantLock(重入锁),Lock也可以用来实现多线程的同步
它提供了如下一些方法来实现多线程的同步:
- lock():这个方法以阻塞的方式获取锁,也就是说,如果获取到锁了,立即返回;如果别的线程持有锁,当前线程就等待,直到获取到锁之后返回
- tryLock():这个方法与lock()方法不同,它以非阻塞的方式来获取锁。此外,它只是常识性地去获取一下锁,如果获取到了锁,立即返回true,否则立即返回false
- tryLock(long timeout,TimeUnit unit):如果获取到锁,立即返回true,否则会等待参数给定的时间单元,在等待的过程中,如果获取到了锁,就返回true,如果等待超时则返回false
- lockInterruptibly():如果获取了锁,立即返回;如果没有获取到锁,当前线程处于休眠状态,直到获取到锁,或者当前线程被别的线程中断(会受到InterruptedException异常)。它与lock()方法最大的区别在于如果lock()方法获取不到锁就会一直处于阻塞状态,而且还会忽略interrupt()方法
- unlock():用来释放锁
示例如下:
package javatest;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;
public class LockTest {
public static void main(String[] args)throws InterruptedException {
// TODO Auto-generated method stub
final Lock lock=new ReentrantLock();
lock.lock();
Thread t1=new Thread(new Runnable() {
public void run() {
try {
lock.lockInterruptibly();
}catch(InterruptedException e)
{
System.out.println("interrupted");
}
}
});
t1.start();
t1.interrupt();
Thread.sleep(1);
}
}