1.线程安全问题

多个线程同时运行,线程调度由操作系统决定,程序本身无法决定。

如果多个线程同时读写共享变量,就可能出现问题。

假设有AddThread和DecThread,它们分别对同一个共享变量做加和减运算LOOP次,最终结果应该是0。但某些时候比如LOOP为10000时,结果是错误的。

  1. class AddThread extends Thread{
  2. public void run(){
  3. for(int i=0;i<Main.LOOP;i++){
  4. Main.count += 1;
  5. }
  6. }
  7. }
  8. class DecThread extends Thread{
  9. public void run(){
  10. for(int i=0;i<Main.LOOP;i++){
  11. Main.count -= 1;
  12. }
  13. }
  14. }
  15. public class Main {
  16. final static int LOOP = 10000;
  17. public static int count = 0;
  18. public static void main(String[] args) throws InterruptedException{
  19. Thread t1 = new AddThread();
  20. Thread t2 = new DecThread();
  21. t1.start();
  22. t2.start();
  23. //等待这两个线程执行结束
  24. t1.join();
  25. t2.join();
  26. System.out.println(count);
  27. }
  28. }


## 2.原子操作
* 因此对共享变量进行写入时,必须保证是原子操作
* 原子操作是指不能被中断的一个或一系列操作

当执行 n = n +1时,编译器会把它编译为3条字节码指令,分别是ILOAD, IADD, ISTORE。所以对于这个简单的赋值语句,它并不是一个原子操作,这就可能导致两个线程在执行这条语句的时候,会出现问题。

假设1:n=100,Thread1执行语句n为101,Thread2再执行n为102。

假设2:Thread1刚执行完ILOAD指令,就被操作系统暂停了,然后Thread2调度执行,结果n变成了101,此后Thread1再度被操作系统调度执行,结果也是101。即n+1的指令被2个线程调用了2次,最终只加了1.



所以我们要保证当Thread1执行时,Thread2不能执行,直到Thread1执行完毕,Thread2才能开始执行。这样运行的结果就是正确的。

要实现这个效果,就要对ILOAD之前和ISTORE之后进行加锁和解锁。

3.同步代码块

Java使用synchronized对一个对象进行加锁:

  • 为了保证一系列操作作为原子操作,必须保证一系列操作过程中不被其他线程执行
  1. synchronized (lock){
  2. n=n+1;
  3. }

当一个线程想要执行synchronized语句块时,必须首先获得指定对象的锁,这个对象就是synchronized括号里的对象,然后线程再执行synchronized语句块,执行结束以后释放锁。

在执行synchronized语句块时,如果Thread1执行到任何语句时,被操作系统中断。其他线程如Thread2因为无法获取lock对象的锁,从而导致Thread2无法进入synchronized语句块,Thread2就必须等待,直到Thread1再次被调用,并执行完synchronized语句块释放了锁,Thread2才能获得lock对象锁,进入synchronized语句块。

synchronized保证了代码块和任意时刻最多只有一个线程能执行。

  • 因为一个对象的锁只能被一个线程获得,其他线程必须等待。

synchronized的问题:

  • 性能下降。因为synchronized代码块无法并发执行,所以性能会下降。此外加锁和解锁都会消耗一定的时间,所以synchronized会降低程序的执行效率。

如何使用synchronized:

  • 1.找出修改共享变量的线程代码块
  • 2.选择一个实例作为锁
  • 3.使用synchronized(lock Object){...}

注意:

  • 对于同一个变量的修改,必须要获取同一个锁,如果2个线程获取的是不同的锁,它们是没有办法进行同步的。
  • 不用担心异常。无论有无异常,在synchronized结束时都会释放锁。
  1. class AddThread extends Thread{
  2. public void run(){
  3. for(int i=0;i<Main.LOOP;i++){
  4. synchronized (Main.LOCK) {
  5. Main.count += 1;
  6. }
  7. }
  8. }
  9. }
  10. class DecThread extends Thread{
  11. public void run(){
  12. for(int i=0;i<Main.LOOP;i++){
  13. synchronized (Main.LOCK) {//对于同一个变量的修改,要使用同一个锁
  14. Main.count -= 1;
  15. }//无论有无异常,都会在此释放锁
  16. }
  17. }
  18. }
  19. public class Main {
  20. final static int LOOP = 10000;
  21. public static int count = 0;
  22. public static final Object LOCK = new Object();
  23. public static void main(String[] args) throws InterruptedException{
  24. Thread t1 = new AddThread();
  25. Thread t2 = new DecThread();
  26. t1.start();
  27. t2.start();
  28. t1.join();
  29. t2.join();
  30. System.out.println(count);
  31. }
  32. }

4.JVM的原子操作

JVM定义了几种原子操作:

  • 基本类型(long和double除外)赋值
  • 引用类型赋值

注意:

  • 原子操作时不需要同步的。
  • 可以把非原子操作变为原子操作
  • 局部变量不需要同步
  1. //原子操作不需要同步
  2. public void set(int m){
  3. synchronized (obj){
  4. this.value = m;
  5. }
  6. }
  7. //->
  8. public void set(int m){
  9. this.value = m;
  10. }
  11. //对2个int类型进行赋值,它不是一个原子操作。但可以先构造一个int数组,然后利用引用类型赋值,把它变成1个原子操作。
  12. class Pair{
  13. int first;
  14. int last;
  15. public void set(int first,int last){
  16. synchronized (this){
  17. this.first = first;
  18. this.last = last;
  19. }
  20. }
  21. }
  22. //->
  23. class Pair{
  24. int[] pair;
  25. public void set(int first,int last){
  26. int[] ps = new int[]{first,last};
  27. this.pair = ps;
  28. }
  29. }
  30. //a,b,s1,s2,r都是局部变量,各个线程的局部变量是完全独立的,互不影响,所以这个方法不需要同步。
  31. public int avg(int a, int b){
  32. int s1 = a*a + b*b;
  33. int s2 = a + b;
  34. int r = s1/s2;
  35. return r;
  36. }

5.总结:

  • 多线程同时修改变量,会造成逻辑错误

    * 需要通过synchronized同步

    * 同步的本质就是给指定对象加锁

    * 注意加锁对象必须是同一个实例
  • 对JVM定义的单个原子操作不需要同步

廖雪峰Java11多线程编程-2线程同步-1同步代码块的更多相关文章

  1. 廖雪峰Java11多线程编程-2线程同步-3死锁

    1.线程锁可以嵌套 在多线程编程中,要执行synchronized块: 必须首先获得指定对象的锁 Java的线程锁是可重入的锁.对同一个对象,同一个线程,可以多次获取他的锁,即同一把锁可以嵌套.如以下 ...

  2. 廖雪峰Java11多线程编程-2线程同步-2synchronized方法

    1.Java使用synchronized对一个方法进行加锁 class Counter{ int count = 0; public synchronized void add(int n){ cou ...

  3. 廖雪峰Java11多线程编程-2线程同步-4wait和notify

    wait和notify synchronized解决了多线程竞争的问题 我们可以在synchronized块中安全的对一个变量进行修改,但是它没有解决多线程协调的问题. 例如设计一个TaskQueue ...

  4. 廖雪峰Java11多线程编程-1线程的概念-1多线程简介

    多任务 现代操作系统(windows,MacOS,Linux)都可以执行多任务: 多任务就是同时运行多个任务,例如同时开启钉钉.百度网盘.火狐.谷歌.ps等 操作系统执行多任务就是让多个任务交替执行, ...

  5. 廖雪峰Java11多线程编程-4线程工具类-1ThreadLocal

    多线程是Java实现多任务的基础: Thread ExecutorService ScheduledThreadPool Fork/Join Thread对象代表一个线程:调用Tread.curren ...

  6. 廖雪峰Java11多线程编程-1线程的概念-2创建新线程

    Java语言内置多线程支持: 一个Java程序实际上是一个JVM进程 JVM用一个主线程来执行main()方法 在main()方法中又可以启动多个线程 1.创建新线程 1.1 方法一:使用Thread ...

  7. 廖雪峰Java11多线程编程-1线程的概念-5中断线程

    1.中断线程: 如果线程需要执行一个长时间任务,就可能需要中断线程.场景:从网络上下载一个100M的文件,用户在下载过程中中断下载任务的执行. 中断线程就是其他线程给该线程发一个信号,该线程收到信号后 ...

  8. 廖雪峰Java11多线程编程-1线程的概念-3线程的状态

    1线程的状态 线程终止的的原因: run()或call()方法执行完成,线程正常结束 线程抛出一个未捕获的Exception或Error 直接调用该线程的stop()方法来结束该线程--该方法容易导致 ...

  9. 廖雪峰Java11多线程编程-3高级concurrent包-2ReadWriteLock

    ReentrantLock保证单一线程执行 ReentrantLock保证了只有一个线程可以执行临界区代码: 临界区代码:任何时候只有1个线程可以执行的代码块. 临界区指的是一个访问共用资源(例如:共 ...

随机推荐

  1. QQ空间删除日志

    按下F12,贴上如下代码 var delay = 2000; function del() { document.querySelector(".app_canvas_frame" ...

  2. JavaScript 数据值校验工具类

    /** * 数据值校验工具类 */ var checkService = { // 不校验 none: function () { return true; }, //非空校验 isEmpty: fu ...

  3. (转载)js引擎的执行过程(一)

    概述 js是一种非常灵活的语言,理解js引擎的执行过程对我们学习javascript非常重要,但是网上讲解js引擎的文章也大多是浅尝辄止或者只局部分析,例如只分析事件循环(Event Loop)或者变 ...

  4. IK分词器插件

    (1)源码 https://github.com/medcl/elasticsearch-analysis-ik  (2)releases https://github.com/medcl/elast ...

  5. Print Article /// 斜率优化DP oj26302

    题目大意: 经典题 数学分析 G(a,b)<sum[i]时 a优于b G(a,b)<G(b,c)<sum[i]时 b必不为最优 #include <bits/stdc++.h& ...

  6. 腾讯bugly接入插件(CocosCreator)

    下载: plugin-bugly.zip (1.4 MB) 插件开源地址: https://github.com/tidys/CocosCreatorPlugins/tree/master/packa ...

  7. [NOIP2019模拟赛]序列(Sequence)

    题目大意 有一个序列$A_i$ • 对于 i ≥ 1,如果有$ A_i > 0.A_{i+1}> 0$ 且存在 $A_{i+2}$,那么法老可以令$ Ai$ 和 $A_{i+1}$ 减一, ...

  8. Bash Specially-crafted Environment Variables Code Injection Vulnerability Analysis

    http://www.cnblogs.com/LittleHann/p/3992778.html

  9. shell脚本实现读取一个文件中的某一列,并进行循环处理

    shell脚本实现读取一个文件中的某一列,并进行循环处理 1) for循环 #!bin/bash if [ ! -f "userlist.txt" ]; then echo &qu ...

  10. java-学习网站推荐

    技术博客: http://c.biancheng.net/view/1390.html (设计模式等等应有尽有,最全教程,强烈推荐!!!) hutool:http://hutool.mydoc.io/ ...