`
DavidIsOK
  • 浏览: 75921 次
社区版块
存档分类
最新评论

java 的synchronized 机制详细介绍

    博客分类:
  • java
阅读更多



一、线程的先来后到

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

有人说:那如果我没有这道门会怎样呢?让两个线程相互竞争,谁抢先了,谁就可以先干活,这样多好阿?但是我们知道:如果厕所没有门的话,如厕的人一起涌向厕所,那么必然会发生争执,正常的如厕步骤就会被打乱,很有可能会发生意想不到的结果,例如某些人可能只好被迫在不正确的地方施肥……

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

那么在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 虚拟机都会自动锁上对象或者类。

看到这里,我想你们一定都疲劳了吧?o(∩_∩)o...哈哈。让我们休息一下,但是在这之前,请你们一定要记着:
当一个有限的资源被多个线程共享的时候,为了保证对共享资源的互斥访问,我们一定要给他们排出一个先来后到。而要做到这一点,对象锁在这里起着非常重要的作用。

在上一篇中,我们讲到了多线程是如何处理共享资源的,以及保证他们对资源进行互斥访问所依赖的重要机制:对象锁。
本篇中,我们来看一看传统的同步实现方式以及这背后的原理。

很多人都知道,在Java多线程编程中,有一个重要的关键字,synchronized。但是很多人看到这个东西会感到困惑:都说同步机制是通过对象锁来实现的,但是这么一个关键字,我也看不出来Java程序锁住了哪个对象阿?

没错,我一开始也是对这个问题感到困惑和不解。不过还好,我们有下面的这个例程:

public class ThreadTest extends Thread {

    private int threadNo;

    public ThreadTest(int threadNo) {

        this.threadNo = threadNo;

    }

    public static void main(String[] args) throws Exception {

        for (int i = 1; i < 10; i++) {

           new ThreadTest(i).start();

            Thread.sleep(1);

        }

     }

 

    @Override

     public synchronized void run() {

        for (int i = 1; i < 10000; i++) {

            System.out.println("No." + threadNo + ":" + i);

        }

     }

 }

这个程序其实就是让10个线程在控制台上数数,从1数到9999。理想情况下,我们希望看到一个线程数完,然后才是另一个线程开始数数。但是这个程序的执行过程告诉我们,这些线程还是乱糟糟的在那里抢着报数,丝毫没有任何规矩可言。

但是细心的读者注意到:run方法还是加了一个synchronized关键字的,按道理说,这些线程应该可以一个接一个的执行这个run方法才对阿。

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

我们来看下面的例程:

public class ThreadTest2 extends Thread {

 private int threadNo; private String lock;

 public ThreadTest2(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 = 1; i < 10; i++) { 

  new ThreadTest2(i, lock).start();

      Thread.sleep(1);

     }

  } 

public void run() { 

 synchronized (lock) {

      for (int i = 1; i < 10000; i++) {

       System.out.println("No." + threadNo + ":" + i);

    }  

 } 

 }

 }

我们注意到,该程序通过在main方法启动10个线程之前,创建了一个String类型的对象。并通过ThreadTest2的构造函数,将这个对象赋值给每一个ThreadTest2线程对象中的私有变量lock。根据Java方法的传值特点,我们知道,这些线程的lock变量实际上指向的是堆内存中的同一个区域,即存放main函数中的lock变量的区域。

程序将原来run方法前的synchronized关键字去掉,换用了run方法中的一个synchronized块来实现。这个同步块的对象锁,就是 main方法中创建的那个String对象。换句话说,他们指向的是同一个String类型的对象,对象锁是共享且唯一的!

于是,我们看到了预期的效果:10个线程不再是争先恐后的报数了,而是一个接一个的报数。

再来看下面的例程:

public class ThreadTest3 extends Thread {  

   

     private int threadNo;  

     private String lock;  

   

     public ThreadTest3(int threadNo) {  

         this.threadNo = threadNo;  

     }  

     public static void main(String[] args) throws Exception {  

         for (int i = 1; i < 20; i++) {  

             new ThreadTest3(i).start();  

             Thread.sleep(1);  

         }  

     }  

     public static synchronized void abc(int threadNo) {  

         for (int i = 1; i < 10000; i++) {  

              

                 System.out.println("No." + threadNo + ":" + i);          

         }  

     }  

     public void run() {  

         abc(threadNo);  

     }  

 }

细心的读者发现了:这段代码没有使用main方法中创建的String对象作为这10个线程的线程锁。而是通过在run方法中调用本线程中一个静态的同步方法abc而实现了线程的同步。我想看到这里,你们应该很困惑:这里synchronized静态方法是用什么来做对象锁的呢?

我们知道,对于同步静态方法,对象锁就是该静态放发所在的类的Class实例,由于在JVM中,所有被加载的类都有唯一的类对象,具体到本例,就是唯一的 ThreadTest3.class对象。不管我们创建了该类的多少实例,但是它的类实例仍然是一个!
这样我们就知道了:
1
、对于同步的方法或者代码块来说,必须获得对象锁才能够进入同步方法或者代码块进行操作;

2
、如果采用method级别的同步,则对象锁即为method所在的对象,如果是静态方法,对象锁即指method所在的
Class
对象(唯一)

3
、对于代码块,对象锁即指synchronized(abc)中的abc

4
、因为第一种情况,对象锁即为每一个线程对象,因此有多个,所以同步失效,第二种共用同一个对象锁lock,因此同步生效,第三个因为是
static
因此对象锁为ThreadTest3class 对象,因此同步生效。

如上述正确,则同步有两种方式,同步块和同步方法(为什么没有waitnotify?这个我会在补充章节中做出阐述)

如果是同步代码块,则对象锁需要编程人员自己指定,一般有些代码为synchronized(this)只有在单态模式才生效;
(本类的实例有且只有一个)

如果是同步方法,则分静态和非静态两种 
静态方法则一定会同步,非静态方法需在单例模式才生效,推荐用静态方法(不用担心是否单例)

所以说,在Java多线程编程中,最常见的synchronized关键字实际上是依靠对象锁的机制来实现线程同步的。
我们似乎可以听到synchronized在向我们说:给我一把 锁,我能创造一个规矩

 

 

  • 大小: 57.8 KB
分享到:
评论

相关推荐

    java锁机制Synchronizedjava锁机制Synchronized

    "Java 锁机制 Synchronized" Java 锁机制 Synchronized 是 Java 语言中的一种同步机制,用于解决多线程并发访问共享资源时可能出现的一些问题。 Java 锁机制 Synchronized 的概念 在 Java 中,每个对象都可以被...

    Javasynchronized机制.pdf

    以下是对`synchronized`机制的详细解释: 1. **同步方法**: 当`synchronized`关键字应用于方法时,它会锁定调用该方法的对象。这意味着,如果两个线程分别使用不同的对象实例调用同一个同步方法,它们可以同时...

    Java synchronized详细解读.docx

    总的来说,`synchronized`是Java中实现线程安全的关键工具,它通过锁机制保证了对共享资源的有序访问,防止了数据竞争问题。然而,正确使用`synchronized`需要对并发编程有深入理解,以确保性能和正确性之间的平衡。...

    Java synchronized使用案例

    Java中的`synchronized`关键字是多线程编程中的一个重要概念,用于控制并发访问共享资源,以保证数据的一致性和完整性。这个关键词提供了互斥锁机制,防止多个线程同时执行同一段代码,确保了线程安全。 一、`...

    java锁机制Synchronized.pdf

    java锁机制Synchronized.pdf

    java synchronized demo

    本示例"java synchronized demo"旨在探讨`synchronized`关键字的用法及其作用机制。下面将详细阐述`synchronized`的相关知识点。 1. **synchronized的作用**: - `synchronized`关键字主要用于解决多线程环境中的...

    java同步synchronized关键字用法示例

    Java中的`synchronized`关键字是多线程编程中的一个重要概念,用于控制并发访问共享资源时的同步机制。在Java中,当多个线程试图同时访问和修改同一块代码或数据时,可能会导致数据不一致的问题。为了解决这个问题,...

    生产者消费者Java—synchronized 机制

    总的来说,这个项目提供了一个学习和实践Java多线程同步机制的好例子,尤其是`synchronized`关键字的使用,以及如何通过线程间通信解决并发问题。通过分析和运行这个项目,可以加深对生产者消费者模型和`...

    java锁机制Synchronized[归纳].pdf

    java锁机制Synchronized[归纳].pdf

    技术分享-java synchronized锁机制.pptx

    由浅入深解析synchronized锁的机制,各种锁的概念的介绍,膨胀过程,基于redis的分布式锁demo。

    [JAVA][synchronized的使用]

    在Java编程语言中,`synchronized`关键字是一个至关重要的概念,它主要用于实现线程同步,以确保多线程环境下的数据一致性与安全性。本篇文章将深入探讨`synchronized`的使用,包括其基本原理、使用方式以及实际应用...

    java锁机制Synchronized参考.pdf

    java锁机制Synchronized参考.pdf

    Java synchronized关键_动力节点Java学院整理

    3. **监视器锁**:`synchronized`关键字实际上基于Java的监视器锁(Monitor)机制,由JVM实现。当线程进入`synchronized`代码块或方法时,会获取锁并进入临界区,执行完毕后释放锁,其他线程才能继续执行。 **...

    Java-synchronized详解.docx

    这篇文章将详细介绍 Java 中的 synchronized 机制,通过实例代码,演示如何使用 synchronized解决多线程同步问题。 Java 中的同步机制 在 Java 中,同步机制是通过 synchronized 关键字实现的。synchronized 可以...

    详细解读java同步之synchronized解析

    Java中的`synchronized`关键字是实现多线程同步的重要机制,它确保了在并发环境中对共享资源的访问是线程安全的。以下是对`synchronized`的详细解读: ### 1. synchronized的特性 - **互斥性**:当一个线程进入一...

    java_synchronized详解

    #### 二、synchronized的作用机制 `synchronized`主要通过对象锁(也称为监视器锁)来实现其功能。当一个线程进入某个对象的一个`synchronized`代码块时,它会自动获得该对象的锁;离开该代码块时,则自动释放锁。...

Global site tag (gtag.js) - Google Analytics