java线程学习之synchronized关键字
关键字synchronized的作用是实现线程间的同步。它的任务是对同步的代码加锁。一个代码块同时只能有同一个线程进行读和写操作,从而保证线程间是安全的。
线程安全的概念是:当多个线程访问某一个类(对象或方法)时,这个对象始终都能表现出正确的行为,那么这个类(对象或方法)就是线程安全的。反之就是线程不安全的。
一、为什么要用synchronized?
举个例子,老王有张储蓄卡,里面有一万块钱,老王通过无卡取现要取八千,操作过程中,会先查询储蓄金额,发现是一万,当输入八千的时候系统会让他支取。如果在老王的媳妇在这时拿着卡通过柜台要支取七千块钱,如果取钱的代码没有synchronized同步控制,他们同时进行查询的操作,发现金额是一万,两人输入金额同时进行取钱操作,就有可能都能取出需要的钱来,共计15000,这样的话,银行就会亏了,显示也是不安全的。
如果上个例子理解不了,那么看下面例子:
public class TestSynchorized 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的线程");
}
}
运行结果为:

在上面19行如果加了synchronized 关键字后的结果

为什么这样呢:
先分析没加的时候,主线程创建了线程 t1 和线程 t2 , t2 先执行Timer类,这时执行完 num++,num的值为1此时线程 执行 Thread.sleep(1) 语句, t2 线程休眠1毫秒,此时 t1 线程执行Timer类,同样执行完 num++语句,num的值为2,此时线程 执行 Thread.sleep(1) 语句,t1 线程进行休眠 。 t2休眠结束,继续执行打印输出语句,随后 t1 也休眠结束,继续执行打印输出语句。
加完synchronized后执行情况呢,同样 t1 先执行执行到 add(String name) 方法,发现有 synchronized ,此时就会独占此资源,即使是在休眠的时候也不允许 t2 执行 add(String name) 方法,只有当 t1 把add 方法执行完毕退出去后,才让 t2 执行 add(String name) 方法。
可能会有疑问,为什么没加 synchronized 关键字前 是 t2 先执行,加了 synchronized 后是 t1 先执行? 其实java虚拟机jvm执行线程的时候的顺序跟书写代码的顺序无关,跟cpu执行线程的顺序有关,执行的时候,谁在前,谁就先执行。这个当然不是绝对的 ,它还受到优先级的影响。但优先级一样的时候,是这样的。
二、synchronized用法
1、给指定的对象加锁,进入同步代码前要获得当前实例的锁。
2、直接作用于实例方法,相当于对当前实例加锁,同步代码前要获得实例的锁。
3、直接作用于静态方法,相当于对当前类加锁,同步代码前要获得当前类的锁。
三、synchronized方法与synchronized代码块
synchronized静态方法的锁的示例:

synchronized代码块的示例:

四、synchronized的作用
1、synchronized可确保线程同步,线程安全。正如之上所言。
2、synchronized能保证线程间的可见性。
可见性:一个线程对共享变量值的修改,能够及时的被其他线程看到。
synchronized可以完全替代volative的功能,只是在使用上没有volative使用方便。
3、synchronized保证线程间的有序性。
因为synchronized限制每次只能允许一个线程可以访问代码块,因此无论怎么使用同步块内的代码如何被打乱顺序,只要保证串行语义一致,那么执行结果总是一样的。而其他访问线程,又必须在获得锁后方能进入代码块读取数据,因此,看到的最终结果并不取决于代码的执行过程,从而有序性自然得到了解决。
java线程学习之synchronized关键字的更多相关文章
- java线程学习之volatile关键字
volatile变量的主要作用:是使变量在多个线程间可见. 在java中每一个线程都会有一块工作内存区,其中存放着所有线程共享的主内存的变量值的拷贝.当线程执行时,它在自己的工作内存区操作这些变量,为 ...
- java高并发系列 - 第10天:线程安全和synchronized关键字
这是并发系列第10篇文章. 什么是线程安全? 当多个线程去访问同一个类(对象或方法)的时候,该类都能表现出正常的行为(与自己预想的结果一致),那我们就可以所这个类是线程安全的. 看一段代码: pack ...
- JAVA多线程之Synchronized关键字--对象锁的特点
一,介绍 本文介绍JAVA多线程中的synchronized关键字作为对象锁的一些知识点. 所谓对象锁,就是就是synchronized 给某个对象 加锁.关于 对象锁 可参考:这篇文章 二,分析 s ...
- Java:多线程,线程同步,synchronized关键字的用法(同步代码块、非静态同步方法、静态同步方法)
关于线程的同步,可以使用synchronized关键字,或者是使用JDK 5中提供的java.util.concurrent.lock包中的Lock对象.本文探讨synchronized关键字. sy ...
- 从线程池到synchronized关键字详解
线程池 BlockingQueue synchronized volatile 前段时间看了一篇关于"一名3年工作经验的程序员应该具备的技能"文章,倍受打击.很多熟悉而又陌生的知识 ...
- Java线程同步(synchronized)——卖票问题
卖票问题通常被用来举例说明线程同步问题,在Java中,采用关键字synchronized关键字来解决线程同步的问题. Java任意类型的对象都有一个标志位,该标志位具有0,1两种状态,其开始状态为1, ...
- Java多线程4:synchronized关键字
原文:http://www.cnblogs.com/skywang12345/p/3479202.html 1. synchronized原理在java中,每一个对象有且仅有一个同步锁.这也意味着,同 ...
- 学习java线程学习笔记
线程:代码执行的一个分支 主要作用是提高了效率,cpu能同时执行多个部分的代码. 线程的创建:两种方式 a.继承于thread类,重写run方法. b. ...
- [多线程] 线程中的synchronized关键字锁
为什么要用锁? 在多线程中,难免会出现在多个线程中对同一个对象的实例变量或者全局静态变量进行并发访问的情况,如果不做正确的同步处理,那么产生的后果就是"脏读",也就是取到的数据其实 ...
随机推荐
- js数组指定位置添加删除
示例参考:http://www.w3school.com.cn/jsref/jsref_splice.asp
- racket安装
https://www.cnblogs.com/scige/p/3379447.html
- python经常使用的十进制、16进制、字符串、字节串之间的转换(长期更新帖)
进行协议解析时.总是会遇到各种各样的数据转换的问题,从二进制到十进制,从字节串到整数等等 废话不多上.直接上样例 整数之间的进制转换: 10进制转16进制: hex(16) ==> 0x10 ...
- MTQQ 物联网
这个是学校的SRP项目.去年12月做了大概3周. 直接摘个人总结报告如下: 在本次“学生研究计划”项目,研究课题是“基于JAVA的智能家居公众号”.根据课题要求之一:以微信作为媒介,实现智能设备的远程 ...
- jQuery地图插件jVectorMap的简单使用
1.官网下载jVectorMap插件压缩文件 官网地址:http://www.jvectormap.com/ 2.解压文件包括jVectorMap库及基础样式表,编写Html文件,引入jQuery框架 ...
- Robot Framework 自动化测试 Selenium2Library 库 用法
Robot Framework自动化测试Selenium2Library库详细用法 一.浏览器驱动 通过不同的浏览器执行脚本. Open Browser Htpp://www.xxx.com ...
- win10系统电脑无法识别u盘的解决办法
一些win10系统用户说插入usb设备的时候出现无法识别usb设备的问题,就此问题,接下来是对应的解决方法. win10系统电脑无法识别U盘的应对方法: 右键“计算机”,从弹出的菜单中选择“属性”项: ...
- 迪杰斯特拉(Dijkstra)算法描述及理解
Dijkstra算法是一种计算单源最短无负边路径问题的常用算法之一,时间复杂度为O(n2) 算法描述如下:dis[v]表示s到v的距离,pre[v]为v的前驱结点,用以输出路径,vis[v]表示该点最 ...
- python中闭包的理解
闭包的三个条件: 1.函数(外函数)中定义了内函数:2.内函数使用了外函数的非全局变量:3.外函数最终返回的是内函数的引用. 简单闭包事例: #outerfunc为外函数 def outerfunc( ...
- 《图解HTTP》读书笔记(七:通信数据转发程序-代理/网关/隧道)
HTTP通信时,除客户端和服务器以外,还有一些用于通信数据转发的应用程序,例如代理.网关和隧道,它们可以配合服务器工作.这些服务器和应用程序可以将请求转发给通信线路上的下一站服务器,并且能接收从那台服 ...