Java多线程同步机制

来源:互联网 发布:远程监控软件下载 编辑:程序博客网 时间:2024/05/21 19:22

一段sysnchronized的代码被一个线程执行之前,它要先拿到执行这段代码的权限,在java里面就是拿到某个同步对象的锁(一个对象一把锁),如果这个时候同步对象的锁被其他线程拿走了,这个线程就只能等了(线程阻塞在锁池 等待队列中)。取到锁后,它就开始执行同步代码(被synchronized修饰的代码),线程执行完同步代码后马上就把锁还给同步对象,其他在锁池中 等待某个线程就可以拿到锁执行同步代码了。这样就保证了同步代码在同一时刻只有一个线程在执行。
众所周知,在Java多线程编程中,一个非常重要的方面就是线程的同步问题。
关于线程的同步,一般有一下解决办法:

  1. 在需要同步的方法的方法名中加入synchronized关键字
  2. 使用synchronized块对需要进行同步的代码进行同步
  3. 使用JDK5中提供的java.util.concurrent.lock包中的Lock对象
    另外,为了解决多个线程对同一变量进行访问时可能发生的安全性问题,我们不仅可以采用同步机制,更可以通过使用JDK1.2中加入的ThreadLocal来保证更好的并发性。
    本文大致目录结构如下:

线程的先来后到——问题的提出:为什么要有多线程同步?Java多线程同步的机制是什么?
给我一把锁,我能创造一个规矩——传统的多线程同步变成方法有哪些?它们有什么异同?
Lock来了,大家让开——Java并发框架中的Lock详解。
你有我有全都有——ThreadLocal如何解决并发安全性?
总结——Java线程安全的几种方法对比

## 线程的先来后到 ##

我们来举一个例子:
某餐厅的卫生间很小,几乎只能容纳一个人。为了保证不受干扰,有人进入就需要锁上门。我们可以把卫生间想象是共享资源,而众多需要卫生间的人可以被视为多个线程。加入卫生间当前有人,那么其他人就必须等待。 知道这个人打开门出来为止,这就好比是多个线程共享一个资源的时候,是一定要分出先来后到的。

有人说:那如果没有这道门会怎么样呢?让两个线程相互竞争,谁抢先,谁就可以先工作么,这样多好。但是我们知道:如果厕所没有门的话,那么所有人一起通向卫生间,必然会发生争执。正常的上卫生间就会被打乱,很有可能会发生意想不到的事情,例如某人被迫在不正确的地方“施肥”~

正是因为有这道门,任何一个单独进入卫生间的人都可以顺利的完成他们的上卫生间过程,而不会被干扰,甚至发生意外的结果,这就是说上卫生间的时候要讲究先来后到。

那么在java多线程程序中,当多个线程竞争同一个资源的时候,如何能够保证他们不会产生“打架”的情况呢?有人说使用同步机制。没错,像上面的这个例子,就是典型的同步案例,一旦第一位开始,第二位就必须等待,一个线程一旦进入某一过程,必须等待正常的返回,并退出一个过程,下一个线程才能开始这个过程,这里最关键的是卫生间的这道门,其实这道门担任的是资源锁的角色,只要进入的人锁上门,就相当于获得了这个锁,而当它打开锁出来以后,就相当于释放了这个锁。

也就是说,多线程的线程同步机制实际上是靠锁的概念控制的,那么在java程序当中,锁是如何体现的呢?
让我们从JVM的角度来看看锁这个概念:
在Java程序运行时环境中,JVM需要对两类线程共享的数据进行协调:
1. 保存在堆中的实例变量
2. 保存在方法去中的类变量

这两类数据是被所有线程共享的。
(程序不需要协调保存在Java栈中的数据,因为这些数据是属于拥有该栈的线程所私有的)
在Java虚拟机中,每个对象和类在逻辑上都是和一个监听器相关联的。
对于对象来说,相关联的监视器保护对象的实例变量。
对于类来说,监视器保护类的类变量。
(如果一个对象没有实例变量,或者一个类没有变量,相关联的监视器就什么也不监视,为了实现监视器的监视能力,java虚拟机为每一个对象和类都关联一个锁,代表任何时候只允许一个线程拥有特权,线程访问实例变量或者类变量不需要锁)
但是如果线程获取了锁,那么在它释放这个锁之前,就没有其他线程可以获取同样的数据的锁了。(锁住一个对象就是获取对象相关联的监视器)
类锁实际上是用对象锁来实现,当虚拟机装菜一个class文件的时候,它就会创建一个java.lang.Class类的实例,当锁住一个对象的时候,实际上锁住的是那个类的Class对象。
一个线程可以多次对同一个对象上锁,对于每一个对象,java虚拟机维护一个加锁计数器,线程每获得一次对象,计数器就加1,没释放一次,计数器就减1,当计数器值为0的时候,锁就被完全释放了。
java编程人员不需要自己动手加锁,对象锁是java虚拟机内部使用的。
在java程序中,只需要使用synchronized块或者synchronized方法就可以标志一个监视区域。当每次进入一个监视区域时,java虚拟机都会自动锁上对象或者类。
看到这里,我想你们一定都泼来了吧,我们休息一下。但这之前,你们一定要记着:当一个有限的资源被多个线程共享的时候,为了保证对共享资源的互斥访问,我们一定要给他们排出一个先来后到,而做到这一点,对象锁在这里起着重要作用。

我们来看一看传统的同步实现方式以及这背后的原理。
很多人都知道,在Java多线程编程中,有一个重要的关键字synchronized,但是很多人看到这个东西会感到困惑“都说同步机制是通过对象锁来实现的,但是这么一个关键字,我也看不出来Java程序锁住了哪个对象啊?”

没错,我开始也不知道,来看下面这个示例:

public class ThreadTest extends Thread{    private int threadNo;    public ThreadTest(int threadNo){        this.threadNo = threadNo;    }    public static void main(String[]args){        for(int i=1;i<10;i++){            new ThreadTest(i).start();            Thread.sleep(1);        }    }    @Override    public synchronized void run(){        for(int i=1;i<100;i++){            System.out.println("No"+threadNo+":"+i);        }    }}

这个程序其实是让10线程在控制台上数数,从1到100,理想情况下,我们希望看到一个线程数完,然后另一个才开始,但是实际上它是没有顺序的。我知道你不信,Ctrl + C,然后Ctrl + v去试试。^_^

我们上面提到了,对于一个成员方法加上synchronized关键字,这实际上是以这个成员方法所在的对象本身作为对象锁。
在本示例中,就是ThreadTest类的一个具体对象,也就是该线程自身作为对象锁,一共10个线程,每个线程持有自己线程对象的那个对象锁,这必然不能产生同步效果。换句话说,如果对这些线程进行同步 ,那么这些线程所持有的对象锁应该是共享唯一的才对。
我们来看下面的示例:

public class ThreadTest extends Thread{    private int threadNo;    private String lock;    public Thread(int threadNo,String lock){        this.threadNo = threadNo;        this.lock = lock;    }    public static void main(String[]args) throws Exception{        String lock = new String("lock");        for(int i=0;i<10;i++){            new ThreadTest(i,lock).start();        }    }    public void run(){        synchronized(lock){            System.out.println("No"+threadNo+":"+i);        }    }}

我们注意到,改程序通过main方法启动10个线程之前,创建了一个String类型的对象,并通过ThreadTest的构造函数,将这个对象赋值,给每一个ThreadTest线程对象中的私有变量Lock,根据Java方法的传值特点,我们知道,这些线程的lock变量实际上指向的是堆内存中的同一个区域,即存放main函数中的lock变量的区域。
程序将原来run方法前的synchronized关键字去掉,换用了run方法中的一个synchronized块,这个同步块的对象锁,就是mian方法中创建的那个String对象,换句话说,他们指向的是同一个String类型的对象,对象锁是共享且唯一的。
于是我们就看到了,报数的效果了。是一个接一个。
再来看下面的例子:

public class ThreadTest extends Thread{    private int threadNo;    private String lock;    public Thread(int threadNo){        this.threadNo = threadNo;    }    public static void main(String[]args) throws Exception{        String lock = new String("lock");        for(int i=0;i<10;i++){            new ThreadTest(i).start();        }    }    public static synchronized void abc(int threadNo){        for(int i=1;i<100;i++){            System.out.println("No"+threadNo+":"+i);        }    }    public void run(){        abc(threadNo);    }}

细心的读者可能发现了,这段代码没有使用main方法中创建String对象作为这10个线程的线程锁,而是通过在run方法中调用本线程中一个静态同步方法abc实现了线程的同步,我想看到这里,你们应该很困惑,这里synchronized静态方法是用什么来做对象锁的呢?
我们知道,对于同步静态方法,对象锁就是该静态方法所在类的Class实例,由于在JVM中,所有被加载的类都有唯一的类对象,具体到本示例,就是唯一的ThreadTest.class对象,不管我们创建了该类的多少实例,但是它的类实例仍然是一个!

这样我们就知道了:

  1. 对于同步的方法或者代码块来说,必须获得对象锁才能够进入同步方法或者代码进行操作
  2. 如果采用method级别的同步,则对象锁为method所在的对象,如果是静态方法,对象锁就是值method所在的Class对象(唯一)
  3. 对于代码块,对象锁就是synchronized(abc)中的abc
  4. 因为第一种情况,对象锁为每一个线程对象,因此多个,所以同步失效,第二种情况是共用一个对象锁Lock,因此同步生效。第三个因为是static因此对象锁为ThreadTest的Class对象,所以同步生效。
    如上述正确,那么同步有两种形式,同步块和同步方法
    如果是同步代码块,则对象锁需要编程人员自己指定,一般有些代码为synchronized(this)只有在单态模式才生效。
    如果是同步方法,则分静态和非静态两种。
    静态方法则一定会同步,非静态方法需在单例模式才生效,推荐使用静态方法(不用担心是否单例)
    所以说,在Java多线程编程中,最常见的synchronized关键字实际上是依靠对象锁的机制来实现线程同步的。
    我们似乎可以听到synchronized在向我们说“给我一把锁,我能创造一个规矩”。
原创粉丝点击