关键字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关键字的更多相关文章

  1. java线程学习之volatile关键字

    volatile变量的主要作用:是使变量在多个线程间可见. 在java中每一个线程都会有一块工作内存区,其中存放着所有线程共享的主内存的变量值的拷贝.当线程执行时,它在自己的工作内存区操作这些变量,为 ...

  2. java高并发系列 - 第10天:线程安全和synchronized关键字

    这是并发系列第10篇文章. 什么是线程安全? 当多个线程去访问同一个类(对象或方法)的时候,该类都能表现出正常的行为(与自己预想的结果一致),那我们就可以所这个类是线程安全的. 看一段代码: pack ...

  3. JAVA多线程之Synchronized关键字--对象锁的特点

    一,介绍 本文介绍JAVA多线程中的synchronized关键字作为对象锁的一些知识点. 所谓对象锁,就是就是synchronized 给某个对象 加锁.关于 对象锁 可参考:这篇文章 二,分析 s ...

  4. Java:多线程,线程同步,synchronized关键字的用法(同步代码块、非静态同步方法、静态同步方法)

    关于线程的同步,可以使用synchronized关键字,或者是使用JDK 5中提供的java.util.concurrent.lock包中的Lock对象.本文探讨synchronized关键字. sy ...

  5. 从线程池到synchronized关键字详解

    线程池 BlockingQueue synchronized volatile 前段时间看了一篇关于"一名3年工作经验的程序员应该具备的技能"文章,倍受打击.很多熟悉而又陌生的知识 ...

  6. Java线程同步(synchronized)——卖票问题

    卖票问题通常被用来举例说明线程同步问题,在Java中,采用关键字synchronized关键字来解决线程同步的问题. Java任意类型的对象都有一个标志位,该标志位具有0,1两种状态,其开始状态为1, ...

  7. Java多线程4:synchronized关键字

    原文:http://www.cnblogs.com/skywang12345/p/3479202.html 1. synchronized原理在java中,每一个对象有且仅有一个同步锁.这也意味着,同 ...

  8. 学习java线程学习笔记

    线程:代码执行的一个分支          主要作用是提高了效率,cpu能同时执行多个部分的代码.      线程的创建:两种方式      a.继承于thread类,重写run方法.      b. ...

  9. [多线程] 线程中的synchronized关键字锁

    为什么要用锁? 在多线程中,难免会出现在多个线程中对同一个对象的实例变量或者全局静态变量进行并发访问的情况,如果不做正确的同步处理,那么产生的后果就是"脏读",也就是取到的数据其实 ...

随机推荐

  1. css 中 zoom和transform:scale的区别(转载)

    一.IE和Chrome等浏览器与zoom 还在几年前,zoom还只是IE浏览器自己私有的玩具,但是,现在,除了FireFox浏览器,其他,尤其Chrome和移动端浏览器已经很好支持zoom属性了: z ...

  2. windows下创建.gitignore

    网上搜索 .gitignore 的创建,很多linux上的,而且还一样,... 尝试了几次,windows可以这样写 .svn/ .settings/ .buildpath .project

  3. sysbench对MySQL的压测

    QPS - query per second TPS - transaction per second 不是特别关注,每个业务场景中事务标准是不一样的 Ⅰ.sysbench测试框架 Ⅱ.常用测试脚本 ...

  4. Python中的引用传参

    Python中函数参数是引用传递(注意不是值传递).对于不可变类型,因变量不能修改,所以运算不会影响到变量自身:而对于可变类型来说,函数体中的运算有可能会更改传入的参数变量. 引用传参一: >& ...

  5. Luogu4113 [HEOI2012]采花

    题目大意:给定一个长度为$n$的序列$a_i$,$m$次询问,每次询问$[l,r]$,求在区间内有多少个数出现了至少2次. 数据范围:$1\leq l\leq r\leq n\leq 2*10^6,1 ...

  6. leetcode 之 two sum (easy)c++

    1.数组的长度 length() .容器vector长度  size() .容器vector vector是C++标准模板库中的部分内容,它是一个多功能的,能够操作多种数据结构和算法的模板类和函数库. ...

  7. main.js中封装全局登录函数

    1. 在 main.js 中封装全局登录函数 通过 vue 对象的原型扩展,可以扩展一个函数,这样这个函数就可以在每一个界面通过类似指向对象的方式,去访问这个函数. 如下是 main.js 扩展的函数 ...

  8. python tkinter entry

    """小白随笔,大佬勿喷""" '''Entry编辑框 收集数据''' import tkinter as tk import tkinte ...

  9. Python从入坑到放弃!

    Python基础  python基础 python基础之 while 逻辑运算符 格式化输出等 python基础之 基本数据类型,str方法和for循环 python基础之 列表,元组,字典 pyth ...

  10. mysql语句,插入id随机生成

    insert into 表名 VALUES(uuid(),…) 还有一个uuid_short(),只有数字 insert into 表名 VALUES(uuid_short(),…)