Java 多线程(二)synchronized和volatile
脏读:
脏读指当一个事务正在访问数据,并且对数据进行了修改,而这种修改还没有提交到数据库中,这时,另外一个事务也访问这个数据,然后使用了这个数据。总的来说取到的数据是其实是被更改过的,但还没有保存到数据库的数据。
不可重复读:
不可重复读是指在一个事务内,多次读同一数据。在这个事务还没有结束时,另一个事务也访问该同一数据,那么,在第一个事务中的两次读数据之间,由于第二个事务的修改,那么第一个事务两次读到的数据可能是不一样的。这样造成了在同一个事务中两次读到的数据是不一样的,因此称为是不可重复读。即不能读到相同的数据内容。
幻读:
幻读是指当事务不是独立执行时发生的一种现象,例如第一个事务对一个表中的数据进行了修改,这种修改涉及到表中的全部数据行。同时,第二个事务也修改这个表中的数据,这种修改是向表中插入一行新数据。那么,以后就会发生操作第一个事务的用户发现表中还有没有修改的数据行,就好像发生了幻觉一样。
事务隔离的五种级别:
- TRANSACTION_NONE: 不使用事务
- TRANSACTION_READ_UNCOMMITTED: 允许脏读
- TRANSACTION_READ_COMMITTED: 防止脏读(最常用的隔离级别,并且是大多数数据库的默认隔离级别)
- TRANSACTION_REPEATABLE_READ: 可以防止脏读和不可重复读
- TRANSACTION_SERIALIZABLE: 可以防止脏读,不可重复读和幻读(事务串行化,降低数据库的效率)
注意: 事务的隔离级别受到数据库的限制,不同的数据库支持的隔离级别不一定相同
synchronized关键字
synchronize修饰的方法
关键字synchronized取得的锁都是对象锁或类锁,而不是把一段代码或方法当作锁,哪个线程先执行带有synchronized关键字的方法,哪个线程就持有该方法所属对象的锁,其他线程只能等待,前提是多个线程访问的是同一个对象。
A线程持有Object对象的锁,B线程可以异步调用Object对象中非synchronized的方法
A线程持有Object对象的锁,B线程如调用Object对象中含有synchronized的方法,则需等待,即同步
当一个线程得到一个对象锁之后,再次请求此对象锁可以再次得到该对象锁,这就是锁重入,在父子类的继承当中,也支持锁重入
当一个线程执行代码发生异常时,锁会被自动释放掉
synchronized修饰的同步代码块
当A线程访问对象的用synchronized修饰的代码块时,B线程可以访问该对象方法中其余非synchronized块的部分
当A线程访问对象的用synchronized修饰的代码块,B线程如果要访问这段synchronized块,将会被阻塞
Java还支持对“任意对象”作为对象监视器来实现同步的功能,这个任意对象大多是实例变量和方法的参数
另外只要对象的引用不变,即使改变了对象的属性,运行结果依然是同步的
synchronized的非this对象的三个结论:
- 当多个线程同时执行synchronized(A){ }同步代码块时呈同步效果
- 当其他线程执行A对象中的synchronized同步方法时呈同步效果
- 当其他线程执行A对象方法中的synchronized(this)代码块时也呈同步效果
synchronized修饰静态方法
synchronized用在静态方法上,则代表的时对当前.Java文件对应的Class类加锁,它和非静态的同步方法持有的锁是不同的,前者是类锁,后者是对象锁
volatile关键字
根据Java内存模型(JMM),Java中有一块主内存,不同的线程有自己的工作内存。同一个变量的值在主内存中有一份,如果线程用到了这个变量,自己的工作内存中也会有一份一模一样的的拷贝。每次进入线程从主内存中拿到变量的值,每次执行完线程将变量从工作内存同步回主内存。
volatile关键字修饰变量,每次读取这个变量时,都先从主内存中把变量同步到线程的工作内存中,该变量为当前时刻最新的变量,当修改变量时,都会把这个变量同步到主线程。从而保证了每次读取到的都是最新的值。这就是volatile的可见性
线程安全围绕的是可见性和原子性这两个特性展开的,volatile解决的是变量在多个线程之间的可见性,但无法保证原子性。用synchronized则既保证了原子性,也保证了可见性。synchronized修饰的方法或代码块,都会先把主存中的数据拷贝到工作内存,同步代码结束,会把工作内存的数据更新到主内存,从而保证了主内存的数据始终使最新的
wait()方法
wait()方法的作用是使当前运行代码的线程进入等待状态,并将其线程放置在“预执行队列中”,并且在wait()方法所在的代码处停止执行,直到接到通知或被中断。在调用wait()方法前,线程必须获得该对象的锁,因此只能在同步方法或同步代码块中调用该方法
notify()方法
notify()方法的作用是当多个线程等待,线程规划器会随机挑选出一个wait的线程,将其唤醒,并使它等待获取该对象的对象锁。等待获取该对象的对象锁意味着即使收到通知,wait的线程也不会马上获取对象锁,必须等待调用notify()方法的线程释放对象锁。notify()方法也只能在同步方法或同步代码块中调用
notifyAll()方法
notifyAll()方法唤醒所有等待状态的线程
总结:
- wait()方法会使该线程释放共享资源的锁,从而从运行状态进入等待状态,直到被唤醒
- notify()方法会随机唤醒等待队列中等待共享资源的一个线程,并使该线程从等待状态变为可运行状态,当该线程获取到了该对象锁后,即可运行下去
- notifyAll()方法会唤醒所有等待队列中等待共享资源的线程,并使这些线程从等待状态变为可运行状态
最后,如果wait(),notify(),notifyAll()这三个方法没有在同步方法或同步代码块中调用,将会抛java.lang.IllegalMonitorStateException
注意: wait()方法释放锁,notify()方法不会释放锁
Java 多线程(二)synchronized和volatile的更多相关文章
- Java多线程同步方法Synchronized和volatile
11 同步方法 synchronized – 同时解决了有序性.可见性问题 volatile – 结果可见性问题 12 同步- synchronized synchronized可以在任意对象上加 ...
- Java 多线程之 synchronized 和 volatile 的比較
概述 在做多线程并发处理时,常常须要对资源进行可见性訪问和相互排斥同步操作.有时候,我们可能从前辈那里得知我们须要对资源进行 volatile 或是 synchronized 关键字修饰处理.但是,我 ...
- 【java多线程】synchronized和volatile
文章目录 一.synchronized 1.synchronized使用的方法 2.注意 3.不要以字符串作为锁的对象 4.`synchronized`锁的是什么? 二.volatile 1.引出问题 ...
- java线程安全— synchronized和volatile
java线程安全— synchronized和volatile package threadsafe; public class TranditionalThreadSynchronized { pu ...
- Java多线程-同步:synchronized 和线程通信:生产者消费者模式
大家伙周末愉快,小乐又来给大家献上技术大餐.上次是说到了Java多线程的创建和状态|乐字节,接下来,我们再来接着说Java多线程-同步:synchronized 和线程通信:生产者消费者模式. 一.同 ...
- java 多线程二
java 多线程一 java 多线程二 java 多线程三 java 多线程四 线程中断: /** * Created by root on 17-9-30. */ public class Test ...
- Java多线程编程那些事:volatile解惑--转
http://www.infoq.com/cn/articles/java-multi-thread-volatile/ 1. 前言 volatile关键字可能是Java开发人员“熟悉而又陌生”的一个 ...
- JAVA多线程学习- 三:volatile关键字
Java的volatile关键字在JDK源码中经常出现,但是对它的认识只是停留在共享变量上,今天来谈谈volatile关键字. volatile,从字面上说是易变的.不稳定的,事实上,也确实如此,这个 ...
- Java多线程(二) 多线程的锁机制
当两条线程同时访问一个类的时候,可能会带来一些问题.并发线程重入可能会带来内存泄漏.程序不可控等等.不管是线程间的通讯还是线程共享数据都需要使用Java的锁机制控制并发代码产生的问题.本篇总结主要著名 ...
- Java多线程:synchronized的可重入性
从Java多线程:线程间通信之volatile与sychronized这篇文章中我们了解了synchronized的基本特性,知道了一旦有一个线程访问某个对象的synchronized修饰的方法或代码 ...
随机推荐
- GEF中连接的实现
在GEF绘图笔想象中复杂许多,需要很多组件的依赖与支持,稍微弄错一个引用,或一个操作调试起来就比较麻烦,下面列一下实现一个连接线功能所需要实现的类及添加的方法 建议大图查看. 相关代码:参考<G ...
- 「JavaSE 重新出发」05.03.03 使用反射编写泛型数组代码
Employee[] a = new Employee[100]; // ... // array is full a = Arrays.copyOf(a, 2 * a.length); 如何编写这样 ...
- 避免关注底层硬件,Nvidia将机器学习与GPU绑定
Nvidia释放的一组cuDNN的库,有效的实现了其与多种深度学习框架的整合.基于cuDNN,加速了代码的运行,同时让研究员避免去关心底层硬件性能. 关键字: 编程语言语音识别Nvidia 原文链接: ...
- 2014 Container技术大会:未来Linux Container会是PaaS平台的核心
不应错过2014 Container技术大会的九大理由. 一.Docker官方人员再次来到北京,首次向中国布道Docker技术.2013年Docker高级软件工程师Jerome Petazzoni,曾 ...
- 杭电1019 Least Common Multiple【求最小公倍数】
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1019 解题思路:lcm(a,b)=a*b/gcd(a,b) 反思:最开始提交的时候WA,以为是溢出了, ...
- MongoDB_聚合
MongoDB提供以下聚合工具来对数据进行操作:聚合框架.MapReduce以及几个简单聚合命令:count.distinct.group 聚合框架:可以使用多个构件创建一个管道,上一个构件的结果传给 ...
- 团体程序设计天梯赛-练习集-L1-024. 后天
L1-024. 后天 如果今天是星期三,后天就是星期五:如果今天是星期六,后天就是星期一.我们用数字1到7对应星期一到星期日.给定某一天,请你输出那天的“后天”是星期几. 输入格式: 输入第一行给出一 ...
- Python之进程 进阶 下
在python程序中的进程操作 之前我们已经了解了很多进程相关的理论知识,了解进程是什么应该不再困难了,刚刚我们已经了解了,运行中的程序就是一个进程.所有的进程都是通过它的父进程来创建的.因此,运行起 ...
- WIN 10 增删输入法
第一步: 任务栏右击 “语言——设置” 第二步: 第三步: 删除或者增加就好.
- 路飞学城Python-Day136
列举Http请求中常见的请求方式 根据HTTP标准,HTTP请求可以使用多种请求方法. HTTP1.0定义了三种请求方法: GET, POST 和 HEAD方法. HTTP1.1新增了五种请求方法:O ...