java同步和互斥【用具体程序说明】
java同步和互斥【用具体程序说明】
通俗点说就是,当调用A的含有synchronized的方法是,A会被枷锁,此时A中其他含有synchronized方法只有等到前一个方法调用完毕释放了锁才能被调用
具体说明见另一篇博客<<java同步和互斥【相关原理】》
下面看看具体的程序,其中,Timer中的字段num是共享资源
- package com.bankht.synchronize;
- public class TestSync implements Runnable {
- Timer timer = new Timer();
- public static void main(String[] args) {
- TestSync test = new TestSync();
- Thread t1 = new Thread(test);
- Thread t2 = new Thread(test);
- t1.setName("t1");
- t2.setName("t2");
- t1.start();
- t2.start();
- }
- public void run() {
- timer.add(Thread.currentThread().getName());
- }
- }
- class Timer {
- private static int num = 0;
- public void add(String name) {
- num++;
- try {
- Thread.sleep(1);
- } catch (InterruptedException e) {
- }
- System.out.println(name + ", 你是第" + num + "个使用timer的线程");
- }
- }
- 由于没有同步,所以运行结果如下所示
t1, 你是第2个使用timer的线程
t2, 你是第2个使用timer的线程
也就是说当线程一运行到num++的时候被打线程2打断了,由于java中递增和递减操作均不是原子操作,所以本程序中即使没有调用sleep,也会出现这种被打断的情况
下面看看同步的效果
- public void add(String name) {
- synchronized (this) {//同步
- num++;
- try {
- Thread.sleep(1000);
- } catch (InterruptedException e) {
- }
- System.out.println(name + ", 你是第" + num + "个使用timer的线程");
- }
- }
这样运行结果就会出是正确的
t1, 你是第1个使用timer的线程
t2, 你是第2个使用timer的线程
但是,下面为了说明问题把TestSync里面的run方法改成如下所示
- public void run() {
- /
- time.add(Thread.currentThread().getName());
- try {
- Thread.sleep(1000);//为了显示结果,让其睡眠一秒
- } catch (InterruptedException ex) {
- Logger.getLogger(TestSync.class.getName()).log(Level.SEVERE, null, ex);
- }
- System.out.println(Thread.currentThread().getName() + "----");
- }
那么在此运行就是
t1, 你是第1个使用timer的线程
t2, 你是第2个使用timer的线程
t1--------
t2--------
而不是你所想象的
t1, 你是第1个使用timer的线程
t1----
t2, 你是第2个使用timer的线程
t2----
原因就是在线程t1在睡眠的时候,线程t2切换进来,执行了一次,怎样得到正确的结果呢,下面把TestSync里面的run方法做如下改进就可以得到上面预期的结果
- public void run() {
- synchronized(time){
- time.add(Thread.currentThread().getName());
- try {
- Thread.sleep(3000);
- } catch (InterruptedException ex) {
- Logger.getLogger(TestSync.class.getName()).log(Level.SEVERE, null, ex);
- }
- System.out.println(Thread.currentThread().getName() + "----");
- }
- }
因为t1先获得time的锁,所以在执行完run里面的同步块之前,即使sleep(),t2也不会执行,因为t2没有获得time的锁,且sleep()操作也不释放锁(这也是和wait的巨大区别)
附录:TestSync.java全部代码
- package com.bankht.synchronize;
- import java.util.logging.Level;
- import java.util.logging.Logger;
- public class TestSync implements Runnable {
- Timer timer = new Timer();
- public static void main(String[] args) {
- TestSync test = new TestSync();
- Thread t1 = new Thread(test);
- Thread t2 = new Thread(test);
- t1.setName("t1");
- t2.setName("t2");
- t1.start();
- t2.start();
- }
- // public void run() {
- // timer.add(Thread.currentThread().getName());
- // }
- // public void run() {
- // timer.add(Thread.currentThread().getName());
- // try {
- // Thread.sleep(1000);//为了显示结果,让其睡眠一秒
- // } catch (InterruptedException ex) {
- // Logger.getLogger(TestSync.class.getName()).log(Level.SEVERE, null, ex);
- // }
- // System.out.println(Thread.currentThread().getName() + "----");
- // }
- public void run() {
- synchronized(timer){
- timer.add(Thread.currentThread().getName());
- try {
- Thread.sleep(3000);
- } catch (InterruptedException ex) {
- Logger.getLogger(TestSync.class.getName()).log(Level.SEVERE, null, ex);
- }
- System.out.println(Thread.currentThread().getName() + "----");
- }
- }
- }
- class Timer {
- private static int num = 0;
- // public void add(String name) {
- //
- // num++;
- // try {
- // Thread.sleep(1000);
- // } catch (InterruptedException e) {
- // }
- // System.out.println(name + ", 你是第" + num + "个使用timer的线程");
- //
- // }
- public void add(String name) {
- synchronized (this) {// 同步
- num++;
- try {
- Thread.sleep(1000);
- } catch (InterruptedException e) {
- }
- System.out.println(name + ", 你是第" + num + "个使用timer的线程");
- }
- }
- }
java同步和互斥【用具体程序说明】的更多相关文章
- 【Java并发基础】并发编程领域的三个问题:分工、同步和互斥
前言 可以将Java并发编程抽象为三个核心问题:分工.同步和互斥. 这三个问题的产生源自对性能的需求.最初时,为提高计算机的效率,当IO在等待时不让CPU空闲,于是就出现了分时操作系统也就出现了并发. ...
- 监视锁——Java同步的基本思想
翻译人员: 铁锚翻译时间: 2013年11月13日原文链接: Monitors – The Basic Idea of Java synchronization如果你上过操作系统课程,你就知道监视锁( ...
- 死磕 java同步系列之zookeeper分布式锁
问题 (1)zookeeper如何实现分布式锁? (2)zookeeper分布式锁有哪些优点? (3)zookeeper分布式锁有哪些缺点? 简介 zooKeeper是一个分布式的,开放源码的分布式应 ...
- 死磕 java同步系列之ReentrantReadWriteLock源码解析
问题 (1)读写锁是什么? (2)读写锁具有哪些特性? (3)ReentrantReadWriteLock是怎么实现读写锁的? (4)如何使用ReentrantReadWriteLock实现高效安全的 ...
- Java同步与异步
一.关键字: thread(线程).thread-safe(线程安全).intercurrent(并发的) synchronized(同步的).asynchronized(异步的). volatile ...
- 转载自~浮云比翼:Step by Step:Linux C多线程编程入门(基本API及多线程的同步与互斥)
Step by Step:Linux C多线程编程入门(基本API及多线程的同步与互斥) 介绍:什么是线程,线程的优点是什么 线程在Unix系统下,通常被称为轻量级的进程,线程虽然不是进程,但却可 ...
- exec函数族,守护进程,线程同步和互斥
2015.3.2 进程和程序有三点不同:1,存在位置不同,程序:硬盘,磁盘.进程:内存2. 程序是静态的,进程是动态的 执行./a.out -->bash->bash程序调用fork()- ...
- Step by Step:Linux C多线程编程入门(基本API及多线程的同步与互斥)
介绍:什么是线程,线程的优点是什么 线程在Unix系统下,通常被称为轻量级的进程,线程虽然不是进程,但却可以看作是Unix进程的表亲,同一进程中的多条线程将共享该进程中的全部系统资源,如虚拟地址空间, ...
- Windows下C++多线程同步与互斥简单运用
1. 互斥量,Mutex #include <Windows.h> #include <iostream> using namespace std; DWORD WINAPI ...
随机推荐
- redis单线程为什么速度那么快?
1.redis是存储在内存上的,读写的话不会受到硬盘 I/O 速度的限制 如图: (1).硬盘数据库的工作模式: (2).内存数据库的工作模式 2.数据结构简单,对数据操作也简单 3.多路IO复用模型 ...
- hive -- 分区,分桶(创建,修改,删除)
hive -- 分区,分桶(创建,修改,删除) 分区: 静态创建分区: 1. 数据: john doe 10000.0 mary smith 8000.0 todd jones 7000.0 boss ...
- 刷题upupup【Java中HashMap、HashSet用法总结】
HashMap: 常用操作 1. containsKey() 判断HashMap是否包含key 2. containsValue() 判断HashMap是否包含“值为value”的元素 3. get( ...
- NullReferenceException 的可恨之处
通常我们在取数据库记录或者字段时,获取不存在的值时,会出现 NullReferenceException 如果根据某个键值去LoadById, 我们通常会检查一下这个键值是否在数据库里存在. 但如果I ...
- 二分查找方法和printk打印级别
人生就是一个茶几,上面摆满了杯具.内核也是一个大茶几,不过它上面的杯具是一个个的bug.确定bug什么时候被引入是一个很关键的步骤,在这个定位bug的过程中,不论有意或无意,都会很自然地用到二分查找的 ...
- webbench安装使用
简介 运行在linux上的一个性能测试工具 官网地址:http://home.tiscali.cz/~cz210552/webbench.html 如果不能打开的话,也可以直接到网盘下载:http:/ ...
- 使用eclipse新建一个c项目
一.打开eclipse并新建项目 1.快捷键:字体放大:Ctrl+Shift+“+” 字体缩小:Ctrl+“-”
- js 两数的最大公约数
function gcd(a,b){ if (b == 0){ return a; } var r = parseInt(a % b) ; return gcd(b, r);}gcd(12,5);
- 使用HM16.0对视频编码
1.编译HM16.0源码: 步骤参照:https://www.vcodex.com/hevc-and-vp9-codecs-try-them-yourself/(可设置pq等参数) [编译过程中遇到l ...
- 链接中 href='#' 和 href='###' 的区别
<a> 标签 + onclick='{jscode}' 是很常用的一种 js 运用方式,而不使用 href='javascript:{jscode}' 是为了兼容多种浏览器对 <a& ...