使用volatile修饰int型变量i,多个线程同时进行i++操作,这样可以实现线程安全吗
Posted
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了使用volatile修饰int型变量i,多个线程同时进行i++操作,这样可以实现线程安全吗相关的知识,希望对你有一定的参考价值。
参考技术A 不安全,可以使用临界区。这样才安全。volatile不是用来线程同步的,只是用来到内存中读,而不在寄存器中读。
volatile与synchronized区别
volatile与synchronized区别
仅靠volatile不能保证线程的安全性。(原子性)
①volatile轻量级,只能修饰变量。synchronized重量级,还可修饰方法
②volatile只能保证数据的可见性,不能用来同步,因为多个线程并发访问volatile修饰的变量不会阻塞。
synchronized不仅保证可见性,而且还保证原子性,因为,只有获得了锁的线程才能进入临界区,从而保证临界区中的所有语句都全部执行。多个线程争抢synchronized锁对象时,会出现阻塞。
什么是Volatile
Volatile 关键字的作用是变量在多个线程之间可见。
线程安全性
线程安全性包括两个方面,①可见性。②原子性。
1.Java内存模型
共享内存模型指的就是Java内存模型(简称JMM),JMM决定一个线程对共享变量的写入时,能对另一个线程可见。从抽象的角度来看,JMM定义了线程和主内存之间的抽象关系:线程之间的共享变量存储在主内存(main memory)中,每个线程都有一个私有的本地内存(local memory),本地内存中存储了该线程以读/写共享变量的副本。本地内存是JMM的一个抽象概念,并不真实存在。它涵盖了缓存,写缓冲区,寄存器以及其他的硬件和编译器优化。
什么是ThreadLocal
ThreadLocal提高一个线程的局部变量,访问某个线程拥有自己局部变量。
当使用ThreadLocal维护变量时,ThreadLocal为每个使用该变量的线程提供独立的变量副本,所以每一个线程都可以独立地改变自己的副本,而不会影响其它线程所对应的副本。
ThreadLocal的接口方法
ThreadLocal类接口很简单,只有4个方法,我们先来了解一下:
1.void set(Object value)设置当前线程的线程局部变量的值。
2.public Object get()该方法返回当前线程所对应的线程局部变量。
3.public void remove()将当前线程局部变量的值删除,目的是为了减少内存的占用,该方法是JDK 5.0新增的方法。需要指出的是,当线程结束后,对应该线程的局部变量将自动被垃圾回收,所以显式调用该方法清除线程的局部变量并不是必须的操作,但它可以加快内存回收的速度。
4.protected Object initialValue()返回该线程局部变量的初始值,该方法是一个protected的方法,显然是为了让子类覆盖而设计的。这个方法是一个延迟调用方法,在线程第1次调用get()或set(Object)时才执行,并且仅执行1次。ThreadLocal中的缺省实现直接返回一个null。
以上是关于使用volatile修饰int型变量i,多个线程同时进行i++操作,这样可以实现线程安全吗的主要内容,如果未能解决你的问题,请参考以下文章