专栏首页码农小胖哥的码农生涯一起了解下Java多线程基础

一起了解下Java多线程基础

多线程核心:可见性和原子性(工作内存与主内存) 如下图:

所有的共享变量存在于主内存,每个线程独有自己的共享内存副本即工作内存。

volatile与syncronzid

com.prometheus.multiThread.day07

VOLATILE:

使变量在多个线程间可见,没有原子性,通过加入内存屏障和 禁止重排序(as if serial)优化实现可见性。

  • 使用场景:
  • 写入操作不依赖当前值
  • 该变量没有包含在其他变量的不变式中 ##### 原理:当需要使用被volatile修饰的变量时,线程会从主内存中重新获取该变量的值,但当该线程修改完该变量的值写入主内存的时候,并没有判断主内存内该变量是否已经变化,故可能出现非预期的结果。如主内存内有被volatile修饰变量 a,值为3,某线程使用该变量时,重新从主存内读取该变量的值,为3,然后对其进行+1操作,此时该线程内a变量的副本值为4。但此时该线程的时间片时间到了,等该线程再次获得时间片的时候,主存内a的值已经是另外的值,如5,但是该线程并不知道,该线程继续完成其未完成的工作,将线程内的a副本的值4写入主存,这时,主存内a的值就是4了。这样,之前修改a的值为5的操作就相当于没有发生了,a的值出现了意料之外的结果。

SYNCRONZID:

读也好,写也好都是基于对这个变量的加锁操作,全过程中变量是加锁的,其他线程无法对这个变量进行读写操作

参考 :

http://blog.csdn.net/lemon_tree12138/article/details/51062421

代public class VolatileDemo {
    /**
     * volatile只有可见性,无法保证原子性
     */
    private volatile int num=0;

    public int getNum() {
        return this.num;
    }

    public void increase() {
        try {
            //休眠更容易重现结果不为500
            Thread.sleep(100);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        this.num++;
    }

    public static void main(String[] args) {
        final  VolatileDemo volatileDemo = new VolatileDemo();
        for (int i=0;i<500;i++){
            new Thread(new Runnable() {
                @Override
                public void run() {
                    volatileDemo.increase();
                }
            }).start();

        }
        //如果还有子线程在执行,主线程让出cpu资源,直到子线程执行完毕,主线程继续执行
        while (Thread.activeCount()>1){
            Thread.yield();
        }
        System.out.println("-------num:"+volatileDemo.getNum());
    }
}

若解决以上问题,可采用ReentrantLock解决 代码 com.prometheus.multiThread.day07.VolatileDemoReenTrantLock

wait与notify

com.prometheus.multiThread.day08
  • 必须和syncronzid一起使用
  • wait释放锁(让当前对象处于等待状态),notify不释放锁(唤醒其他线程)
  • CountDownLatch通知锁 多用于远程连接等工作

锁优化:

  • 减小锁粒度 ## ThreadLocal:线程局部变量 com.prometheus.multiThread.day10.ConnThreadLocal

  • 共享变量:一个变量再多个线程工作内存中都存在副本,该变量即为共享变量
  • 线程局部变量,保证对当前线程可用。

github : https://github.com/liangliang1259/common-study.git

本文分享自微信公众号 - 码农小胖哥(Felordcn)

原文出处及转载信息见文内详细说明,如有侵权,请联系 yunjia_community@tencent.com 删除。

原始发表时间:2019-09-19

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

我来说两句

0 条评论
登录 后参与评论

相关文章

  • 学并发编程,透彻理解这三个核心是关键

    上一篇文章这次走进并发的世界,请不要错过 给大家带了并发编程的开胃菜,接下来我们逐步上正餐,在吃正餐之前,我还要引用那首诗词: 「横看成岭侧成峰,远近高低各不同...

    码农小胖哥
  • 菜鸟的进阶之路:了解使用多线程

    关于多进程和多线程,教科书上最经典的一句话是“进程是资源分配的最小单位,线程是CPU调度的最小单位”,一般来说进程是独立的而同一进程中的线程是共享的,但是开一个...

    码农小胖哥
  • 让人头大的各种锁,从这里让你思绪清晰

    说到了锁我们经常会联想到生活中的锁,在我们日常中我们经常会接触到锁。比如我们的手机锁,电脑锁,再比如我们生活中的门锁,这些都是锁。

    码农小胖哥
  • java并发中的volatile和synchronized

    今天使用到了volatile关键字,之前了解到这个关键字,但是不知道他的具体作用是什么,下面就来详细解释一下他的作用:

    王念博客
  • 浏览器进程?线程?傻傻分不清楚!

    进程和线程是操作系统的基本概念,许多人会有所了解,但不能较为清晰的分辨。 这里我们需要了解下面几个点。

    IMWeb前端团队
  • 浏览器进程?线程?傻傻分不清楚!

    本文作者:IMWeb coverguo 原文出处:IMWeb社区 未经同意,禁止转载 在生活中,浏览器和我们的工作和生活息息相关。做为前端开发,我们...

    IMWeb前端团队
  • Java面试官最爱问的volatile关键字

    版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。 ...

    用户1161110
  • Java中Volatile和Synchronized关键字的区别

    在上面,我们定义了三个整数变量:i1、i2和i3。我们定义了三个相应的getter方法:geti1()、geti2()和geti3()。

    程序你好
  • 多线程系列问题汇总

    多线程编程中,一般线程的个数都大于 CPU 的核心个数,而一个 CPU 核心在任意时刻只能被一个线程使用,为了让这些线程都能得到有效的执行,通常 CPU 采取的...

    happyJared
  • 一分钟告诉你synchronized和volatile的区别

    先从一道笔试题开始: 9. volatile关键字是否能保证线程安全?() 答案:不能 ---- synchronized和volatile的区别 sync...

    ImportSource

扫码关注云+社区

领取腾讯云代金券