解析Java编程之Synchronized锁住的对象

 更新时间:2017年10月23日 15:05:22   作者:zhangjk1993  
这篇文章主要介绍了解析Java编程之Synchronized锁住的对象,具有一定参考价值,需要的朋友可以了解下。

图片上传 密码修改为  synchronized是java中用于同步的关键字,一般我们通过Synchronized锁住一个对象,来进行线程同步。我们需要了解在程序执行过程中,synchronized锁住的到底是哪个对象,否则我们在多线程的程序就有可能出现问题。

看下面的代码,我们定义了一个静态变量n,在run方法中,我们使n增加10,然后在main方法中,我们开辟了100个线程,来执行n增加的操作,如果线程没有并发执行,那么n最后的值应该为1000,显然下面的程序执行完结果不是1000,因为我们没有进行线程同步。

import java.util.concurrent.TimeUnit; 
public class SynchronizedTest1 extends Thread { 
  public static int n = 0; 
  public void run() { 
    try { 
      //使n自加10次 
      for (int i = 0; i < 10; i++) { 
        n = n + 1; 
        TimeUnit.MILLISECONDS.sleep(10); 
      } 
    } catch (InterruptedException e) { 
      e.printStackTrace(); 
    } 
  } 
  public static void main(String[] args) throws InterruptedException { 
    Thread[] threads = new Thread[100]; 
    for (int i = 0; i < threads.length; i++) { 
      threads[i] = new SynchronizedTest1(); 
      threads[i].start(); 
    } 
    //使所有其他线程执行完,再继续执行main线程,这样得出的n是最终的结果 
    for (Thread thread : threads) { 
      thread.join(); 
    } 
    System.out.println(n); 
  } 
} 

为了实现同步,我们修改上面的代码,增加一个increase方法,如下。但是当我们执行下面的代码时,会发现n仍然不是1000.

import java.util.concurrent.TimeUnit; 
public class SynchronizedTest2 extends Thread { 
  public static int n = 0; 
  public synchronized void increase() { 
    n++; 
  } 
  public void run() { 
    try { 
      //使n自加10次 
      for (int i = 0; i < 10; i++) { 
        increase(); 
        TimeUnit.MILLISECONDS.sleep(10); 
      } 
    } catch (InterruptedException e) { 
      e.printStackTrace(); 
    } 
  } 
  public static void main(String[] args) throws InterruptedException { 
    Thread[] threads = new Thread[100]; 
    for (int i = 0; i < threads.length; i++) { 
      threads[i] = new SynchronizedTest2(); 
      threads[i].start(); 
    } 
    //使所有其他线程执行完,再继续执行main线程,这样得出的n是最终的结果 
    for (Thread thread : threads) { 
      thread.join(); 
    } 
    System.out.println(n); 
  } 
} 

其实原因很简单,上面的多个线程在执行时根本就没有竞争同一个对象锁。当我们执行用synchronized修饰的非静态方法时,线程会首先获得调用这个方法的对象的锁,然后才能继续执行代码。那么调用这个方法的到底是哪个对象,是this对象。在上面的例子中,thread[i]所代表的线程获取的锁对象是thread[i]对象,也就是该线程对象本身。因此上面所开辟的100个线程只要获得自身对象就可以执行,这样就使同步失去了作用。

我们再次修改代码:即将increase方法改为i静态的,此时程序执行完后n的值为1000。

import java.util.concurrent.TimeUnit; 
 
public class SynchronizedTest3 extends Thread { 
  public static int n = 0; 
 
  public synchronized static void increase() { 
    n++; 
  } 
  public void run() { 
    try { 
      //使n自加10次 
      for (int i = 0; i < 10; i++) { 
        increase(); 
        TimeUnit.MILLISECONDS.sleep(10); 
      } 
    } catch (InterruptedException e) { 
      e.printStackTrace(); 
    } 
  } 
 
  public static void main(String[] args) throws InterruptedException { 
    Thread[] threads = new Thread[100]; 
    for (int i = 0; i < threads.length; i++) { 
      threads[i] = new SynchronizedTest3(); 
      threads[i].start(); 
    } 
 
    //使所有其他线程执行完,再继续执行main线程,这样得出的n是最终的结果 
    for (Thread thread : threads) { 
      thread.join(); 
    } 
    System.out.println(n); 
  } 
} 

当synchronized 修饰static方法,它锁住的是该类的Class对象,而不是某一个具体对象。在上面的例子中,它锁住的就是SynchronizedTest3.class对象。在程序执行过程中,类的Class对象只有一份,所以上面线程竞争的是同一个对象锁。

下面是对synchronized锁住对象的总结:

(1)对于同步方法,锁当前对象(this)
(2)对于静态同步方法,锁当前类的Class对象
(3)对于同步代码块,锁住的是synchronized括号中的对象

总结

以上就是本文关于解析Java编程之Synchronized锁住的对象的全部内容,希望对大家有所帮助。感兴趣的朋友可以继续参阅本站:Java编程redisson实现分布式锁代码示例Java并发编程之重入锁与读写锁等,有什么问题可以直接留言,小编会及时回复大家的。下面推荐本站基本Java编程相关的书籍,免费下载,供朋友们学习参考。

Java初级开发工程师面试题汇总.PDF

https://www.jb51.net/books/576989.html

Java经典实例(第三版) 完整版 ([美]达尔文) 中文pdf扫描版

https://www.jb51.net/books/577859.html

希望大家能够喜欢。

相关文章

  • spring-retry组件的使用教程

    spring-retry组件的使用教程

    Spring Retry的主要目的是为了提高系统的可靠性和容错性,当方法调用失败时,Spring Retry可以在不影响系统性能的情况下,自动进行重试,从而减少故障对系统的影响,这篇文章主要介绍了spring-retry组件的使用,需要的朋友可以参考下
    2023-06-06
  • jar包打包成exe安装包的实现

    jar包打包成exe安装包的实现

    本文主要介绍了jar包打包成exe安装包的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2022-07-07
  • 浅谈Java中ABA问题及避免

    浅谈Java中ABA问题及避免

    这篇文章主要介绍了浅谈Java中ABA问题及避免,具有一定借鉴价值,需要的朋友可以参考下
    2018-01-01
  • IDEA 非常重要的一些设置项(一连串的问题差点让我重新用回 Eclipse)

    IDEA 非常重要的一些设置项(一连串的问题差点让我重新用回 Eclipse)

    这篇文章主要介绍了IDEA 非常重要的一些设置项(一连串的问题差点让我重新用回 Eclipse),本文通过图文并茂的形式给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2020-08-08
  • Java自定义注解实现Redis自动缓存的方法

    Java自定义注解实现Redis自动缓存的方法

    本篇文章主要介绍了Java自定义注解实现Redis自动缓存的方法,具有一定的参考价值,感兴趣的小伙伴们可以参考一下。
    2017-04-04
  • Java多线程之synchronized同步代码块详解

    Java多线程之synchronized同步代码块详解

    这篇文章主要为大家详细介绍了Java多线程之synchronized同步代码块,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下,希望能够给你带来帮助
    2022-03-03
  • Maven编译错误:程序包com.sun.*包不存在的三种解决方案

    Maven编译错误:程序包com.sun.*包不存在的三种解决方案

    J2SE中的类大致可以划分为以下的各个包:java.*,javax.*,org.*,sun.*,本文文章主要介绍了maven编译错误:程序包com.sun.xml.internal.ws.spi不存在的解决方案,感兴趣的可以了解一下
    2024-02-02
  • idea中启动项目弹出 IDEA out of memory窗口的解决方案

    idea中启动项目弹出 IDEA out of memory窗口的解决方案

    这篇文章主要介绍了idea中启动项目弹出 IDEA out of memory窗口的解决方案,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2024-01-01
  • java 反射 动态调用不同类的静态方法(推荐)

    java 反射 动态调用不同类的静态方法(推荐)

    下面小编就为大家带来一篇JAVA 反射 动态调用不同类的静态方法(推荐)。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2016-08-08
  • 一文吃透Spring集成MyBatis

    一文吃透Spring集成MyBatis

    spring能集成很多的框架,是spring一个优势功能,通过集成功能,让开发人员使用其他框架更方便,本文将给大家详细介绍Spring如何集成MyBatis,,需要的朋友可以参考下
    2023-05-05

最新评论