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修饰的方法或代码 ...
随机推荐
- ESLint 规范项目代码
ESLint 由 JavaScript 红宝书 作者 Nicholas C. Zakas 编写, 2013 年发布第一个版本. NCZ 以可扩展.每条规则独立.不内置编码风格为理念编写了一个 lint ...
- 1 ERP管理系统概念
1 ERP管理系统概念 一.ERP是什么? ERP是企业资源计划(Enterpise Resource Planning)的简称,蕴含现代企业管理理念,其核心是在制造资源计划基础上进一步发展而成的面向 ...
- CSS3实现1前端常用Loading效果
此页动画效果都是gif图的,不想用代码写的话,下载图片就可使用. 第1种效果: 代码如下 <div class="loading"> <span></ ...
- Bootstrap 有一个 class 属性叫做 well,它的作用是为设定的列创造出一种视觉上的深度感
Bootstrap 有一个 class 属性叫做 well,它的作用是为设定的列创造出一种视觉上的深度感
- URLLib库使用
Date: 2019-06-19 Author: Sun urllib 在Python 3以后的版本中,urllib2这个模块已经不单独存在(也就是说当你import urllib2时,系统提示你 ...
- java 1.8 内存告警问题
Java HotSpot(TM) 64-Bit Server VM warning: ignoring option PermSize=512m; support was removed in 8.0 ...
- Zabbix系列-REHL6.10离线方式安装Zabbix 4.0 LTS
环境 第零步:关闭系统默认防火墙 setenforce 0 sed -i -r "/^SELINUX=/c SELINUX=disabled" /etc/selinux/confi ...
- Corn Fields 状压动归入门题
#include<cstdio> using namespace std; const int N=15; int dp[N][1<<N],v[1<<N],M[N] ...
- mmap,malloc分配随机内存
随机数1G #cat malloc_rand_1g.c #include <stdio.h> /* printf, scanf, NULL */ #include <stdlib.h ...
- Code VS 1002 搭桥
题目描述 Description 有一矩形区域的城市中建筑了若干建筑物,如果某两个单元格有一个点相联系,则它们属于同一座建筑物.现在想在这些建筑物之间搭建一些桥梁,其中桥梁只能沿着矩形的方格的边沿搭建 ...