java同步和互斥【用具体程序说明】

 
         所有对象都自动含有单一的锁,也就是所有对象都有且只有唯一的锁,所以当某个任务(线程)访问一个类A中含有sycnhronized的方法是,那么在这个线程从该方法返回之前(也就是该方法在当前线程执行完之前),类A中的其他被该关键字标记的方法在其他的线程中都会被阻塞。
            通俗点说就是,当调用A的含有synchronized的方法是,A会被枷锁,此时A中其他含有synchronized方法只有等到前一个方法调用完毕释放了锁才能被调用

具体说明见另一篇博客<<java同步和互斥【相关原理】》

下面看看具体的程序,其中,Timer中的字段num是共享资源

  1. package com.bankht.synchronize;
  2. public class TestSync implements Runnable {
  3. Timer timer = new Timer();
  4. public static void main(String[] args) {
  5. TestSync test = new TestSync();
  6. Thread t1 = new Thread(test);
  7. Thread t2 = new Thread(test);
  8. t1.setName("t1");
  9. t2.setName("t2");
  10. t1.start();
  11. t2.start();
  12. }
  13. public void run() {
  14. timer.add(Thread.currentThread().getName());
  15. }
  16. }
  17. class Timer {
  18. private static int num = 0;
  19. public void add(String name) {
  20. num++;
  21. try {
  22. Thread.sleep(1);
  23. } catch (InterruptedException e) {
  24. }
  25. System.out.println(name + ", 你是第" + num + "个使用timer的线程");
  26. }
  27. }
  1. 由于没有同步,所以运行结果如下所示

t1, 你是第2个使用timer的线程

t2, 你是第2个使用timer的线程

也就是说当线程一运行到num++的时候被打线程2打断了,由于java中递增和递减操作均不是原子操作,所以本程序中即使没有调用sleep,也会出现这种被打断的情况

下面看看同步的效果

  1. public void add(String name) {
  2. synchronized (this) {//同步
  3. num++;
  4. try {
  5. Thread.sleep(1000);
  6. } catch (InterruptedException e) {
  7. }
  8. System.out.println(name + ", 你是第" + num + "个使用timer的线程");
  9. }
  10. }

这样运行结果就会出是正确的

t1, 你是第1个使用timer的线程

t2, 你是第2个使用timer的线程

但是,下面为了说明问题把TestSync里面的run方法改成如下所示

  1. public void run() {
  2. /
  3. time.add(Thread.currentThread().getName());
  4. try {
  5. Thread.sleep(1000);//为了显示结果,让其睡眠一秒
  6. } catch (InterruptedException ex) {
  7. Logger.getLogger(TestSync.class.getName()).log(Level.SEVERE, null, ex);
  8. }
  9. System.out.println(Thread.currentThread().getName() + "----");
  10. }

那么在此运行就是

t1, 你是第1个使用timer的线程

t2, 你是第2个使用timer的线程

t1--------

t2--------

而不是你所想象的

t1, 你是第1个使用timer的线程
t1----
t2, 你是第2个使用timer的线程
t2----

原因就是在线程t1在睡眠的时候,线程t2切换进来,执行了一次,怎样得到正确的结果呢,下面把TestSync里面的run方法做如下改进就可以得到上面预期的结果

  1. public void run() {
  2. synchronized(time){
  3. time.add(Thread.currentThread().getName());
  4. try {
  5. Thread.sleep(3000);
  6. } catch (InterruptedException ex) {
  7. Logger.getLogger(TestSync.class.getName()).log(Level.SEVERE, null, ex);
  8. }
  9. System.out.println(Thread.currentThread().getName() + "----");
  10. }
  11. }

因为t1先获得time的锁,所以在执行完run里面的同步块之前,即使sleep(),t2也不会执行,因为t2没有获得time的锁,且sleep()操作也不释放锁(这也是和wait的巨大区别)

附录:TestSync.java全部代码

  1. package com.bankht.synchronize;
  2. import java.util.logging.Level;
  3. import java.util.logging.Logger;
  4. public class TestSync implements Runnable {
  5. Timer timer = new Timer();
  6. public static void main(String[] args) {
  7. TestSync test = new TestSync();
  8. Thread t1 = new Thread(test);
  9. Thread t2 = new Thread(test);
  10. t1.setName("t1");
  11. t2.setName("t2");
  12. t1.start();
  13. t2.start();
  14. }
  15. //  public void run() {
  16. //      timer.add(Thread.currentThread().getName());
  17. //  }
  18. //   public void run() {
  19. //       timer.add(Thread.currentThread().getName());
  20. //            try {
  21. //                Thread.sleep(1000);//为了显示结果,让其睡眠一秒
  22. //            } catch (InterruptedException ex) {
  23. //                Logger.getLogger(TestSync.class.getName()).log(Level.SEVERE, null, ex);
  24. //            }
  25. //          System.out.println(Thread.currentThread().getName() + "----");
  26. //        }
  27. public void run() {
  28. synchronized(timer){
  29. timer.add(Thread.currentThread().getName());
  30. try {
  31. Thread.sleep(3000);
  32. } catch (InterruptedException ex) {
  33. Logger.getLogger(TestSync.class.getName()).log(Level.SEVERE, null, ex);
  34. }
  35. System.out.println(Thread.currentThread().getName() + "----");
  36. }
  37. }
  38. }
  39. class Timer {
  40. private static int num = 0;
  41. //  public void add(String name) {
  42. //
  43. //      num++;
  44. //      try {
  45. //          Thread.sleep(1000);
  46. //      } catch (InterruptedException e) {
  47. //      }
  48. //      System.out.println(name + ", 你是第" + num + "个使用timer的线程");
  49. //
  50. //  }
  51. public void add(String name) {
  52. synchronized (this) {// 同步
  53. num++;
  54. try {
  55. Thread.sleep(1000);
  56. } catch (InterruptedException e) {
  57. }
  58. System.out.println(name + ", 你是第" + num + "个使用timer的线程");
  59. }
  60. }
  61. }

java同步和互斥【用具体程序说明】的更多相关文章

  1. 【Java并发基础】并发编程领域的三个问题:分工、同步和互斥

    前言 可以将Java并发编程抽象为三个核心问题:分工.同步和互斥. 这三个问题的产生源自对性能的需求.最初时,为提高计算机的效率,当IO在等待时不让CPU空闲,于是就出现了分时操作系统也就出现了并发. ...

  2. 监视锁——Java同步的基本思想

    翻译人员: 铁锚翻译时间: 2013年11月13日原文链接: Monitors – The Basic Idea of Java synchronization如果你上过操作系统课程,你就知道监视锁( ...

  3. 死磕 java同步系列之zookeeper分布式锁

    问题 (1)zookeeper如何实现分布式锁? (2)zookeeper分布式锁有哪些优点? (3)zookeeper分布式锁有哪些缺点? 简介 zooKeeper是一个分布式的,开放源码的分布式应 ...

  4. 死磕 java同步系列之ReentrantReadWriteLock源码解析

    问题 (1)读写锁是什么? (2)读写锁具有哪些特性? (3)ReentrantReadWriteLock是怎么实现读写锁的? (4)如何使用ReentrantReadWriteLock实现高效安全的 ...

  5. Java同步与异步

    一.关键字: thread(线程).thread-safe(线程安全).intercurrent(并发的) synchronized(同步的).asynchronized(异步的). volatile ...

  6. 转载自~浮云比翼:Step by Step:Linux C多线程编程入门(基本API及多线程的同步与互斥)

    Step by Step:Linux C多线程编程入门(基本API及多线程的同步与互斥)   介绍:什么是线程,线程的优点是什么 线程在Unix系统下,通常被称为轻量级的进程,线程虽然不是进程,但却可 ...

  7. exec函数族,守护进程,线程同步和互斥

    2015.3.2 进程和程序有三点不同:1,存在位置不同,程序:硬盘,磁盘.进程:内存2. 程序是静态的,进程是动态的 执行./a.out -->bash->bash程序调用fork()- ...

  8. Step by Step:Linux C多线程编程入门(基本API及多线程的同步与互斥)

    介绍:什么是线程,线程的优点是什么 线程在Unix系统下,通常被称为轻量级的进程,线程虽然不是进程,但却可以看作是Unix进程的表亲,同一进程中的多条线程将共享该进程中的全部系统资源,如虚拟地址空间, ...

  9. Windows下C++多线程同步与互斥简单运用

    1.  互斥量,Mutex #include <Windows.h> #include <iostream> using namespace std; DWORD WINAPI ...

随机推荐

  1. LAMP环境配置安装注意安装步骤及说明事项

    一.安装gcc shell># yum -y install gcc 二.安装zlib压缩库 shell>## cd /home/hsk/tar shell># tar –zxvf ...

  2. OO的奇妙冒险——OOP入门与字符串处理

    OO的奇妙冒险 ~OOP入门与字符串处理~ 总体分析 公测 中测(基础与进阶): 其实在我看来,从完成作业的角度来说,中测的基础与进阶并没有任何区别,都不能挂,都不太难,都对得分没有什么影响.中测的样 ...

  3. C++标准模板库(STL)之Queue

    1.Queue的常用用法 queue:队列,实现的一个先进先出的容器. 1.1.queue的定义 使用queue,首先要加头文件#include<queue>和using namespac ...

  4. 学号 20175212 《Java程序设计》第九周学习总结

    学号 20175212 <Java程序设计>第九周学习总结 教材学习内容总结 一.MySQL数据库管理系统 1.在官网上下载并安装MySQL 2.在IDEA中输入测试代码Connectio ...

  5. Oracle通用维、父子维相互转换

    所谓通用维即维度层级1.2.3均作为字段展示为列,父子维即维度id+父级维度+维度层级字段 通用维 lvl_id1 lvl_name1 lvl_id2 lvl_name2 lvl_id3 lvl_na ...

  6. REST(Representational state transfer)的四个级别以及HATEOAS介绍

    Rest RES(Representational state transfer):表现层状态转移.其实它省略了主语,「表现层」其实指的是「资源」的「表现层」,所以通俗来讲就是:资源在网络中以某种表现 ...

  7. python---文字云

    本文介绍的是数据可视化中的一种常见方式:文字云. 用Python构建文字云主要分为两步: 1)构建文字云 from wordcloud import WordCloud wc = WordCloud( ...

  8. maven配置及使用

    配置maven工程.从官网下载maven工具,然后解压到磁盘某个目录下即可. 计算机->属性->高级系统设置->环境变量. 新建如下变量: 变量名:MAVEN_HOME 变量值:C: ...

  9. Mysql基本操作命令【转载】

    原文链接:http://www.cnblogs.com/rookie-c/p/6425039.html 创建数据库 CREATE DATABASE name; 显示所有数据库 SHOW DATABAS ...

  10. styled-components 背后的魔法

    styled-components 定义组件的风格为 const Button = styled.button` background-color: papayawhip; border-radius ...