transient volatile 关键词

读AbstractMap.java中的源码时,遇到transient  volatile 突然忘记了:

transient volatile Set<K>        keySet = null;
 transient volatile Collection<V> values = null;

 

解析:

关于volatile:

 

在虚拟机的实现中,int char等基本类型是一个字长,而long double是两个字长。某些虚拟机的实现中,long的两个字长可能被作为两个原子性的单字长来操作。

如果不以volatile修饰double long,如果多线程访问该变量,由于long操作的整体非原子性而导致结果混乱。
比如:int,一个线程写入4,另一个写入5. 最后肯定是4或者5.而 long型,可能就是个乱七八糟的数值了。

另外如果用 synchronized 则不存在这样的问题了。

volatile变量,一般在多任务多线程情况下使用。

volatile修饰的变量,表明该变量会有不可预知的改变,该变量会被意想不到的改变。精确的说,jvm每次解析到这个变量时,会小心翼翼的读取它,会从主存中读取,而不会直接从寄存器里读取它的拷贝。

关于transient :

如果用transient声明一个实例变量,当对象存储时,它的值不需要维持。

transient修饰的变量,不进行序列化。

 

摘要:

我们知道,在Java中设置变量值的操作,除了long和double类型的变量外都是原子操作,也就是说,对于变量值的简单读写操作没有必要进行同步。
这在JVM 1.2之前,Java的内存模型实现总是从主存读取变量,是不需要进行特别的注意的。而随着JVM的成熟和优化,现在在多线程环境下
volatile关键字的使用变得非常重要。在当前的Java内存模型下,线程可以把变量保存在本地内存(比如机器的寄存器)中,而不是直接在主存中进行
读写。这就可能造成一个线程在主存中修改了一个变量的值,而另外一个线程还继续使用它在寄存器中的变量值的拷贝,造成数据的不一致。要解决这个问题,只需
要像在本程序中的这样,把该变量声明为volatile(不稳定的)即可,这就指示JVM,这个变量是不稳定的,每次使用它都到主存中进行读取。一般说
来,多任务环境下各任务间共享的标志都应该加volatile修饰。


Volatile修饰的成员变量在每次被线程访问时,都强迫从共享内存中重读该成员变量的值。而且,当成员变量发生变化时,强迫线程将变化值回写到共享内存。这样在任何时刻,两个不同的线程总是看到某个成员变量的同一个值。

Java语言规范中指出:为了获得最佳速度,允许线程保存共享成员变量的私有拷贝,而且只当线程进入或者离开同步代码块时才与共享成员变量的原始值对比。

这样当多个线程同时与某个对象交互时,就必须要注意到要让线程及时的得到共享成员变量的变化。

而volatile关键字就是提示VM:对于这个成员变量不能保存它的私有拷贝,而应直接与共享成员变量交互。

使用建议:在两个或者更多的线程访问的成员变量上使用volatile。当要访问的变量已在synchronized代码块中,或者为常量时,不必使用。

由于使用volatile屏蔽掉了VM中必要的代码优化,所以在效率上比较低,因此一定在必要时才使用此关键字。

 

摘要transient:

Transient 
  Java语言的关键字,用来表示一个域不是该对象串行化的一部分。当一个对象被串行化的时候,transient型变量的值不包括在串行化的表示中,然而非transient型的变量是被包括进去的。

        Transient 也是java中的保留字,Java有个特点就是序列化,简单地来说就是可以将这个类存储在物理空间(当然还是以文件的形式存在),那么当你从本地还原这个文件时,你可以将它转换为它本身。这可以极大地方便程序中操作,但同时,因为涉及到安全问题,所以并不希望把类里面所有的属性都能存储(因为那样,别人可以通过序列化知道类里面的内容),那么我们就可以用上transient这个关键字,它的意思是临时的,即不会随类一起序列化到本地,所以当还原后,这个关键字定义的变量也就不再存在。

你可能感兴趣的:(jvm,多线程,虚拟机)