volatile and transient

来源:互联网 发布:淘宝小二布置的作业 编辑:程序博客网 时间:2024/06/10 20:03


转自:http://www.blogjava.net/fhtdy2004/archive/2009/06/20/286112.html


   volatile修饰的成员变量在每次被线程访问时,都强迫从主内存中重读该成员变量的值。而且,当成员变量发生变化时,强迫线程将变化值回写到主内存。这样在任何时刻,两个不同的线程总是看到某个成员变量的同一个值。   
  
Java语言规范中指出:为了获得最佳速度,允许线程保存共享成员变量的私有拷贝,而且只当线程进入或者离开同步代码块时才与共享成员变量的原始值对比。   
  
这样当多个线程同时与某个对象交互时,就必须要注意到要让线程及时的得到共享成员变量的变化。   
  
而volatile关键字就是提示VM:对于这个成员变量不能保存它的私有拷贝,而应直接与共享成员变量交互。   
  
使用建议:在两个或者更多的线程访问的成员变量上使用volatile。当要访问的变量已在synchronized代码块中,或者为常量时,不必使用。   
  
由于使用volatile屏蔽掉了VM中必要的代码优化,所以在效率上比较低,因此一定在必要时才使用此关键字。   

   

   当且仅当满足以下所有条件时,才应该使用volatile变量

   1.对变量的写入操作不依赖变量的当前值,或者你能确保只有单个线程更新变量的值

   2.该变量不会与其他状态变量一起纳入不可变性条件中;

   3.在访问变量时不需要加锁。

   注意:volatile 只能保证内存可见性,并不能保证原子性,也即不保证其修饰的变量线程安全

   关于volatile 可参见:http://www.ibm.com/developerworks/cn/java/j-jtp06197.html
  
java关键字Transient   
  
转自http:
//horst.sun.blog.163.com/blog/static/348849612007614494492/   
  
翻译自http:
//www.devx.com/tips/Tip/13726。   
  
Java的serialization提供了一种持久化对象实例的机制。当持久化对象时,可能有一个特殊的对象数据成员,我们不想   
用serialization机制来保存它。为了在一个特定对象的一个域上关闭serialization,可以在这个域前加上关键字transient。   
transient是Java语言的关键字,用来表示一个域不是该对象串行化的一部分。当一个对象被串行化的时候,transient型变量的值不包括在串行化的表示中,然而非transient型的变量是被包括进去的。  
注意static变量也是可以串行化的 
  
首先,让我们看一些Java serialization的代码:   
public class LoggingInfo implements java.io.Serializable   
{   
    
private Date loggingDate = new Date();   
    
private String uid;   
    
private transient String pwd;   
      
    LoggingInfo(String user, String password)   
    
{   
        uid 
= user;   
        pwd 
= password;   
    }
   
    
public String toString()   
    
{   
        String password
=null;   
        
if(pwd == null)   
        
{   
        password 
= "NOT SET";   
        }
   
        
else  
        
{   
            password 
= pwd;   
        }
   
        
return "logon info: \n   " + "user: " + uid +   
            
"\n   logging date : " + loggingDate.toString() +   
            
"\n   password: " + password;   
    }
   
}
   
  
现在我们创建一个这个类的实例,并且串行化(serialize)它 ,然后将这个串行化对象写如磁盘。   
  
LoggingInfo logInfo 
= new LoggingInfo("MIKE""MECHANICS");   
System.out.println(logInfo.toString());   
try  
{   
   ObjectOutputStream o 
= new ObjectOutputStream(   
                
new FileOutputStream("logInfo.out"));   
   o.writeObject(logInfo);   
   o.close();   
}
   
catch(Exception e) {//deal with exception}   
  
To read the object back, we can write   
  
try  
{   
   ObjectInputStream in 
=new ObjectInputStream(   
                
new FileInputStream("logInfo.out"));   
   LoggingInfo logInfo 
= (LoggingInfo)in.readObject();   
   System.out.println(logInfo.toString());   
}
   
catch(Exception e) {//deal with exception}   
  
如果我们运行这段代码,我们会注意到从磁盘中读回(read——back (de
-serializing))的对象打印password为"NOT SET"。这是当我们定义pwd域为transient时,所期望的正确结果。   
现在,让我们来看一下粗心对待transient域可能引起的潜在问题。假设我们修改了类定义,提供给transient域一个默认值,   
代码如下:   
  
public class GuestLoggingInfo implements java.io.Serializable   
{   
    
private Date loggingDate = new Date();   
    
private String uid;   
    
private transient String pwd;   
      
    GuestLoggingInfo()   
    
{   
        uid 
= "guest";   
        pwd 
= "guest";   
    }
   
    
public String toString()   
    
{   
        
//same as above   
     }
   
}
   
现在,如果我们穿行化GuestLoggingInfo的一个实例,将它写入磁盘,并且再将它从磁盘中读出,我们仍然看到读回的对象打印password 为 
"NOT SET"。当从磁盘中读出某个类的实例时,实际上并不会执行这个类的构造函数,而是载入了一个该类对象的持久化状态,并将这个状态赋值给该类的另一个对象。  


0 0
原创粉丝点击
热门问题 老师的惩罚 人脸识别 我在镇武司摸鱼那些年 重生之率土为王 我在大康的咸鱼生活 盘龙之生命进化 天生仙种 凡人之先天五行 春回大明朝 姑娘不必设防,我是瞎子 5岁宝宝老尿床怎么办 五岁了还尿床怎么办 2岁宝宝晚上尿床怎么办 胃疼腹泻稀水怎么办 网购成瘾怎么办 知乎 军人被警察打了怎么办 对高院再审不服怎么办 对高院判决不服怎么办 白色砂锅烧黑了怎么办 手机被偷关机了怎么办 狗被眼镜蛇咬了怎么办 穿高跟鞋脚趾头长茧怎么办 穿皮鞋脚底板疼怎么办 鞋子穿了脚趾痛怎么办 脚被鞋子磨肿了怎么办 脚趾磨出茧子怎么办疼 脚掌长茧走路疼怎么办 脚底磨出茧子疼怎么办 手指上写字有茧怎么办 写字磨的茧子疼怎么办 脚上的大脚骨疼怎么办 脸特别烫 又红怎么办 6岁宝宝发音不准怎么办 3岁宝宝发音不准怎么办 5岁宝宝发音不准怎么办 4岁宝宝发音不准怎么办 被螃蟹夹出血了怎么办 苹果6主板坏了怎么办 键盘掉了一个键怎么办 汽车屏幕砸坏了怎么办 电视的屏幕坏了怎么办 高三孩子早恋该怎么办 高一早恋家长该怎么办 高二早恋家长该怎么办 200斤新娘抱不动怎么办 160斤新娘抱不动怎么办 科目二挂了5次怎么办 纹身纹完了肿了怎么办 在宿舍被孤立了怎么办 b站视频不见了哟怎么办 站立时间长了脚疼怎么办