transient、volatile關鍵字

transient是在對象序列化的時候,不參與序列化的字段。html

如LinkedList實現了Serializable,其中有變量transient int size = 0;java

在Serializable序列化的時候size時不會參與序列化的,如用ObjectOutputStream讀取LinkedList時,size時不會被寫入到流中的,包括json轉換。json

package com.ada.wuliu.mobile.front.action.test;

import java.io.Serializable;

import com.alibaba.fastjson.JSON;

public class TestTransient implements Serializable{
    private transient String name;
    private String address;
    
    public TestTransient(String name, String address) {
        super();
        this.name = name;
        this.address = address;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public String getAddress() {
        return address;
    }

    public void setAddress(String address) {
        this.address = address;
    }
    public static void main(String[] args) {
        String json=JSON.toJSONString(new TestTransient("翟春雨","北京"));
        System.out.println(json);
    }
}

{"address":"北京"}緩存

volatile:安全

volatile 申明變量以後,虛擬機不能隨意變更優化目標指令,適用一個線程寫,多個線程讀(只能肯定一個線程修改數據以後,其餘線程可以看到這個改的。可是兩個線程同事修改數據就會產生衝突,也就是髒讀)。多線程

Java語言提供了一種稍弱的同步機制,即volatile變量,用來確保將變量的更新操做通知到其餘線程。當把變量聲明爲volatile類型後,編譯器與運行時都會注意到這個變量是共享的,所以不會將該變量上的操做與其餘內存操做一塊兒重排序。volatile變量不會被緩存在寄存器或者對其餘處理器不可見的地方,所以在讀取volatile類型的變量時總會返回最新寫入的值。性能

  在訪問volatile變量時不會執行加鎖操做,所以也就不會使執行線程阻塞,所以volatile變量是一種比sychronized關鍵字更輕量級的同步機制。優化

 

 

  當對非 volatile 變量進行讀寫的時候,每一個線程先從內存拷貝變量到CPU緩存中。若是計算機有多個CPU,每一個線程可能在不一樣的CPU上被處理,這意味着每一個線程能夠拷貝到不一樣的 CPU cache 中。this

 

  而聲明變量是 volatile 的,JVM 保證了每次讀變量都從內存中讀,跳過 CPU cache 這一步。spa

 

當一個變量定義爲 volatile 以後,將具有兩種特性:

 

  1.保證此變量對全部的線程的可見性,這裏的「可見性」,如本文開頭所述,當一個線程修改了這個變量的值,volatile 保證了新值能當即同步到主內存,以及每次使用前當即從主內存刷新。但普通變量作不到這點,普通變量的值在線程間傳遞均須要經過主內存(詳見:Java內存模型)來完成。

 

  2.禁止指令重排序優化。有volatile修飾的變量,賦值後多執行了一個「load addl $0x0, (%esp)」操做,這個操做至關於一個內存屏障(指令重排序時不能把後面的指令重排序到內存屏障以前的位置),只有一個CPU訪問內存時,並不須要內存屏障;(什麼是指令重排序:是指CPU採用了容許將多條指令不按程序規定的順序分開發送給各相應電路單元處理)。

 

volatile 性能:

 

  volatile 的讀性能消耗與普通變量幾乎相同,可是寫操做稍慢,由於它須要在本地代碼中插入許多內存屏障指令來保證處理器不發生亂序執行

 value ++這樣的操做並不具備原子性,其實際的過程以下:

 

 

當線程1在步驟2對value進行計算時,恰好其餘線程也對value進行了修改,這時線程1返回的值就不是咱們指望的值了,因而出現線程安全問題,因此volatile不能保證複合操做具備原子性;解決辦法就是給increment方法加鎖(lock/synchronized)或將變量聲明爲原子類類型。

 

 

 

Synchronized與volatile區別 

 

1.volatile只能修飾變量,而synchronized能夠修改變量,方法以及代碼塊
2.volatile在多線程中不會存在阻塞問題,synchronized會存在阻塞問題
3.volatile能保證數據的可見性,但不能徹底保證數據的原子性,synchronized即保證了數據的可見性也保證了原子性
4.volatile解決的是變量在多個線程之間的可見性,而sychroized解決的是多個線程之間訪問資源的同步性

 sychroized

指定枷鎖對象:對給定對象加鎖,進入同步代碼前要得到給定對象的鎖

做用與靜態方法:至關於對當前類加鎖(全部實例)

直接做用與實例方法:至關於給當前實例加鎖(必須是同一個對象)

相關文章
相關標籤/搜索