在项目开发中, 或许会碰到JAVA的多线程处理, 为保证业务数据的正常, 必须加上锁机制,  常用的处理方法一般是加上synchronized关键字, 目前JDK版本对synchronized已经做了很好的优化,  我们不用再考虑其性能,  但在实际使用中,  往往由于处理不当,  导致系统性能的严重下降, 那么该如何合理的使用synchronized,  必须对其使用方式有个全面了解, 在网上搜寻的资料, 给出的是四种使用方式, 其实可总结为两种, 一个是同步代码块, 一个是同步方法体, 那么该如何使用, 请看下面的测试

  准备两个方法,  对同一个变量做加法, 再对每个方法, 分别开十个线程执行:

  [java]

  public class ThreadUnit

  {

  private int i = 0;

  private Object obj1 = new Object();

  private Object obj2 = new Object();

  public synchronized Integer doAdd1(Long start) throws Exception

  {

  Thread.sleep(100);

  ++i;

  Thread.sleep(100);

  return i;

  }

  public Integer doAdd2(Long start) throws Exception

  {

  Thread.sleep(100);

  ++i;

  Thread.sleep(100);

  return i;

  }

  }

  相关代码:

  [java]

  // 十个线程同时执行方法2

  for (int i = 0; i < 10; i++)

  {

  new Thread(new MessageThread(1, threadUnit))。start();

  }

  // 十个线程同时执行方法2

  for (int j = 0; j < 10; j++)

  {

  new Thread(new MessageThread(2, threadUnit))。start();

  }

  线程处理:

  [java]

  public void run()

  {

  try

  {

  if (operate == 2)

  {

  long start = System.currentTimeMillis();

  int i = threadUnit.doAdd2(start);

  long takeTime = System.currentTimeMillis() - start;

  System.out.println("doAdd2() => i=" + i + ", spendTime=" + takeTime + "ms");

  spendTime += takeTime;

  }

  else

  {

  long start = System.currentTimeMillis();

  int i = threadUnit.doAdd1(start);

  long takeTime = System.currentTimeMillis() - start;

  System.out.println("doAdd1() => i=" + i + ", spendTime=" + takeTime + "ms");

  spendTime += takeTime;

  }

  }

  catch (Exception e)

  {

  e.printStackTrace();

  }

  }

  运行结果:

  1. 在两个方法体上都加上synchronized

  [java]

  public synchronized Integer doAdd1(Long start) throws Exception

  [java]

  public synchronized Integer doAdd2(Long start) throws Exception

  执行结果:

  [html]

  doAdd1() => i=1, spendTime=203ms

  doAdd2() => i=2, spendTime=406ms

  doAdd2() => i=3, spendTime=609ms

  doAdd2() => i=4, spendTime=796ms

  doAdd2() => i=5, spendTime=1000ms

  doAdd2() => i=6, spendTime=1203ms

  doAdd2() => i=7, spendTime=1406ms

  doAdd2() => i=8, spendTime=1609ms

  doAdd2() => i=9, spendTime=1812ms

  doAdd2() => i=10, spendTime=2015ms

  doAdd2() => i=11, spendTime=2218ms

  doAdd1() => i=12, spendTime=2406ms

  doAdd1() => i=13, spendTime=2609ms

  doAdd1() => i=14, spendTime=2812ms

  doAdd1() => i=15, spendTime=3015ms

  doAdd1() => i=16, spendTime=3218ms

  doAdd1() => i=17, spendTime=3421ms

  doAdd1() => i=18, spendTime=3625ms

  doAdd1() => i=19, spendTime=3828ms

  doAdd1() => i=20, spendTime=4015ms

  花费时间:42226ms

  都是有序执行, 变量值没有产生错乱, 但花费时间42226ms

  2.在doAdd1方法上加上synchronized, doAdd2不加。

  [java]

  public synchronized Integer doAdd1(Long start) throws Exception

  执行结果:

  [java]

  doAdd1方法加上synchronized:

  doAdd1() => i=9, spendTime=204ms

  doAdd2() => i=9, spendTime=188ms

  doAdd2() => i=9, spendTime=188ms

  doAdd2() => i=9, spendTime=188ms

  doAdd2() => i=9, spendTime=188ms

  doAdd2() => i=9, spendTime=188ms

  doAdd2() => i=9, spendTime=188ms

  doAdd2() => i=9, spendTime=188ms

  doAdd2() => i=9, spendTime=188ms

  doAdd2() => i=9, spendTime=188ms

  doAdd2() => i=9, spendTime=188ms

  doAdd1() => i=10, spendTime=391ms

  doAdd1() => i=11, spendTime=610ms

  doAdd1() => i=12, spendTime=813ms

  doAdd1() => i=13, spendTime=1016ms

  doAdd1() => i=14, spendTime=1219ms

  doAdd1() => i=15, spendTime=1422ms

  doAdd1() => i=16, spendTime=1610ms

  doAdd1() => i=17, spendTime=1813ms

  doAdd1() => i=18, spendTime=2016ms

  花费时间:12994ms

  doAdd2方法瞬间执行完成, 之后doAdd1方法则是串行有序执行。 这时doAdd2方法获取的变量值已经错乱,  doAdd1获取的正常。  花费时间:12994ms

  3. 两个方法在都没使用synchronized前的情况:

  执行结果:

  [java]

  doAdd1() => i=16, spendTime=203ms

  doAdd1() => i=16, spendTime=203ms

  doAdd1() => i=16, spendTime=203ms

  doAdd1() => i=16, spendTime=203ms

  doAdd1() => i=16, spendTime=203ms

  doAdd1() => i=16, spendTime=203ms

  doAdd1() => i=16, spendTime=203ms

  doAdd1() => i=16, spendTime=203ms

  doAdd1() => i=16, spendTime=203ms

  doAdd2() => i=16, spendTime=203ms

  doAdd2() => i=16, spendTime=203ms

  doAdd2() => i=16, spendTime=203ms

  doAdd1() => i=16, spendTime=203ms

  doAdd2() => i=16, spendTime=203ms

  doAdd2() => i=16, spendTime=203ms

  doAdd2() => i=16, spendTime=203ms

  doAdd2() => i=16, spendTime=203ms

  doAdd2() => i=16, spendTime=203ms

  doAdd2() => i=16, spendTime=203ms

  doAdd2() => i=16, spendTime=203ms

花费时间:4060ms

  可以看到, 两个方法的变量值获取已经错乱,  但花费时间最少4060ms

  4. 使用同步块, 在两个方法内采用不同的对象锁:

  doAdd1:

  [java]

  synchronized (obj1)

  {

  Thread.sleep(100);

  ++i;

  Thread.sleep(100);

  return i;

  }

  doAdd2:

  [java]

  synchronized (obj2)

  {

  Thread.sleep(100);

  ++i;

  Thread.sleep(100);

  return i;

  }

  执行结果:

  [html]

  doAdd1() => i=2, spendTime=203ms

  doAdd2() => i=2, spendTime=203ms

  doAdd1() => i=4, spendTime=406ms

  doAdd2() => i=4, spendTime=406ms

  doAdd1() => i=6, spendTime=609ms

  doAdd2() => i=6, spendTime=609ms

  doAdd1() => i=8, spendTime=812ms

  doAdd2() => i=8, spendTime=812ms

  doAdd1() => i=10, spendTime=1000ms

  doAdd2() => i=10, spendTime=1015ms

  doAdd1() => i=12, spendTime=1203ms

  doAdd2() => i=12, spendTime=1203ms

  doAdd1() => i=14, spendTime=1406ms

  doAdd2() => i=14, spendTime=1406ms

  doAdd1() => i=16, spendTime=1609ms

  doAdd2() => i=16, spendTime=1609ms

  doAdd1() => i=18, spendTime=1812ms

  doAdd2() => i=18, spendTime=1812ms

  doAdd1() => i=20, spendTime=2015ms

  doAdd2() => i=20, spendTime=2015ms

  花费时间:22165ms

  两个方法有序交替执行, 互不影响,  花费时间:22165ms, 相对加锁同一对象执行的时间缩短。

  5. 使用同步块, 使用方法参数作为对象锁:

  [java]

  public Integer doAdd1(Long start) throws Exception

  {

  synchronized (start)

  {

  Thread.sleep(100);

  ++i;

  Thread.sleep(100);

  return i;

  }

  }

  执行结果:

  [java]

  doAdd1() => i=15, spendTime=203ms

  doAdd1() => i=15, spendTime=203ms

  doAdd1() => i=15, spendTime=203ms

  doAdd1() => i=15, spendTime=203ms

  doAdd1() => i=15, spendTime=203ms

  doAdd1() => i=15, spendTime=203ms

  doAdd1() => i=15, spendTime=203ms

  doAdd2() => i=15, spendTime=203ms

  doAdd2() => i=15, spendTime=203ms

  doAdd1() => i=15, spendTime=203ms

  doAdd1() => i=15, spendTime=203ms

  doAdd2() => i=15, spendTime=203ms

  doAdd2() => i=15, spendTime=203ms

  doAdd2() => i=15, spendTime=203ms

  doAdd2() => i=15, spendTime=203ms

  doAdd2() => i=15, spendTime=203ms

  doAdd2() => i=15, spendTime=203ms

  doAdd2() => i=15, spendTime=203ms

  doAdd1() => i=15, spendTime=203ms

  doAdd2() => i=15, spendTime=203ms

  花费时间:4060ms

  执行效果和第三种情况相同,  每个参数作为不同的对象, 即便加上synchronized也不能起到锁的效果。

  6. 把调用的类改为静态类, 只在一个方法上加锁:

  加锁doAdd1方法:

  [java]

  public static synchronized Integer doAdd1(Long start) throws Exception

  执行结果:

  [html]

  doAdd1() => i=9, spendTime=203ms

  doAdd2() => i=9, spendTime=203ms

  doAdd2() => i=9, spendTime=203ms

  doAdd2() => i=9, spendTime=203ms

  doAdd2() => i=9, spendTime=203ms

  doAdd2() => i=9, spendTime=203ms

  doAdd2() => i=9, spendTime=203ms

  doAdd2() => i=9, spendTime=203ms

  doAdd2() => i=9, spendTime=203ms

  doAdd2() => i=9, spendTime=203ms

  doAdd2() => i=9, spendTime=203ms

  doAdd1() => i=10, spendTime=406ms

  doAdd1() => i=11, spendTime=609ms

  doAdd1() => i=12, spendTime=812ms

  doAdd1() => i=13, spendTime=1015ms

  doAdd1() => i=14, spendTime=1218ms

  doAdd1() => i=15, spendTime=1406ms

  doAdd1() => i=16, spendTime=1609ms

  doAdd1() => i=17, spendTime=1812ms

  doAdd1() => i=18, spendTime=2015ms

  花费时间:13135ms

  和第二种情形类似,  没有加锁的doAdd2方法瞬间执行完成,  doAdd1方法则是串行有序执行。

  总结:

  1. synchronized关键在于锁定的对象, 如果是同一对象, 那么所有执行线程, 必须等待对象锁释放后才能执行,  如果是不同对象,  那么只对各对象所关联的线程生效。

  2. synchronized若加在方法体上,  默认锁定的是对象本身。 对于所有加锁的方法,  都是按照串行规则有序执行,  对于没有加锁的方法, 不受任何影响,  静态类同理。

  3. 合理使用synchronized, 既要保证稳定性, 又要保证性能, 需要在两者间作出权衡,  尽量把synchronized范围细度化, 合理控制业务处理流程;  对象操作原子化, 减少锁的使用;

  不要盲目在方法体上加synchronized关键字, 如果每个方法负责处理的是不同业务,  那么尽量采用第四种情形, 使用不同的对象锁处理, 而不是锁定整个对象。

关于JAVA多线程并发synchronized的测试与合理使用的更多相关文章

  1. java 多线程并发 synchronized 同步机制及方式

    2. 锁机制 3. 并发 Excutor框架 4. 并发性与多线程介绍 1. synchronized  参考1. synchronized 分两种方式进行线程的同步:同步块.同步方法 1. 方法同步 ...

  2. Java 多线程并发编程一览笔录

    Java 多线程并发编程一览笔录 知识体系图: 1.线程是什么? 线程是进程中独立运行的子任务. 2.创建线程的方式 方式一:将类声明为 Thread 的子类.该子类应重写 Thread 类的 run ...

  3. Java多线程并发技术

    Java多线程并发技术 参考文献: http://blog.csdn.net/aboy123/article/details/38307539 http://blog.csdn.net/ghsau/a ...

  4. java 多线程并发问题总结

    java 多线程并发主要通过关键字synchronized实现 Java语言的关键字,当它用来修饰一个方法或者一个代码块的时候,能够保证在同一时刻最多只有一个线程执行该段代码. 一.当两个并发线程访问 ...

  5. Java多线程并发05——那么多的锁你都了解了吗

    在多线程或高并发情境中,经常会为了保证数据一致性,而引入锁机制,本文将为各位带来有关锁的基本概念讲解.关注我的公众号「Java面典」了解更多 Java 相关知识点. 根据锁的各种特性,可将锁分为以下几 ...

  6. Java多线程并发07——锁在Java中的实现

    上一篇文章中,我们已经介绍过了各种锁,让各位对锁有了一定的了解.接下来将为各位介绍锁在Java中的实现.关注我的公众号「Java面典」了解更多 Java 相关知识点. 在 Java 中主要通过使用sy ...

  7. Java多线程-并发容器

    Java多线程-并发容器 在Java1.5之后,通过几个并发容器类来改进同步容器类,同步容器类是通过将容器的状态串行访问,从而实现它们的线程安全的,这样做会消弱了并发性,当多个线程并发的竞争容器锁的时 ...

  8. Java多线程-同步:synchronized 和线程通信:生产者消费者模式

    大家伙周末愉快,小乐又来给大家献上技术大餐.上次是说到了Java多线程的创建和状态|乐字节,接下来,我们再来接着说Java多线程-同步:synchronized 和线程通信:生产者消费者模式. 一.同 ...

  9. 从火箭发场景来学习Java多线程并发闭锁对象

    从火箭发场景来学习Java多线程并发闭锁对象 倒计时器场景 在我们开发过程中,有时候会使用到倒计时计数器.最简单的是:int size = 5; 执行后,size—这种方式来实现.但是在多线程并发的情 ...

随机推荐

  1. 富文本是在modal框中弹出显示的问题

    记录一下,在用tinymce富文本的时候,由于是用在modal 上的,始终无法获取焦点,后来才发现问题出在tinymce在modal前创建了,所以导致这个问题,解决方案就是用 v-if="v ...

  2. 牛客网 江西财经大学第二届程序设计竞赛同步赛 D.绕圈游戏-(跳青蛙游戏)找数的所有因子就可以了

    链接:https://ac.nowcoder.com/acm/contest/635/D来源:牛客网 D.绕圈游戏 433为了帮ddd提升智商,决定陪他van特殊的游戏.433给定一个带有n个点的环, ...

  3. Java 中类的初始化过程

    先来一张 JVM 中的内存模型 . 在Java 虚拟机原理这本书中介绍了类会被初始化的 5 种情况 . 1 遇到 new getstatic putstatic 和 invokestatic 这 4 ...

  4. 洛谷P1154 奶牛分厩

    P1154 奶牛分厩 173通过 481提交 题目提供者该用户不存在 标签高性能 难度普及- 时空限制1s / 128MB 提交  讨论  题解 最新讨论更多讨论 测试点3??? 求助!超时了 我抗议 ...

  5. centOS7下SVN的安装和使用

    1. 安装 CentOS通过yum安装subversion. # yum install subversion  subversion安装在/bin目录: # which svnserve #查看目录 ...

  6. Struts2的概念

    Struts2的概念 Struts2是一个基于MVC设计模式的Web应用框架,它本质上相当于一个servlet,在MVC设计模式中,Struts2作为控制器(Controller)来建立模型与视图的数 ...

  7. hdu 4544 优先队列+贪心

    题意:最近,减肥失败的湫湫为发泄心中郁闷,在玩一个消灭免子的游戏.游戏规则很简单,用箭杀死免子即可.箭是一种消耗品,已知有M种不同类型的箭可以选择,并且每种箭都会对兔子造成伤害,对应的伤害值分别为Di ...

  8. bzoj 2998 第k小字串

    这道题用后缀数组貌似会T. 后缀自动机做法: t==0:第k小的本质不同字串 首先把后缀自动机建出来,我们会得到一个DAG,并且只存在一个点入度为0(我们称之为根),可以证明字符串的任意一个本质不同的 ...

  9. Python168的学习笔记4

    关于普通文本文件的读写 python2.7中,未注明的字符都是以acsii来编码的,而要让字符能够通用,必须声明为unicode. s=u'你好',s.encode('utf8')就是指用utf8来进 ...

  10. BZOJ 4521 CQOI 2016 手机号码 数位DP

    4521: [Cqoi2016]手机号码 Time Limit: 10 Sec  Memory Limit: 512 MBSubmit: 539  Solved: 325[Submit][Status ...