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
1.保證此變量對全部的線程的可見性,這裏的「可見性」,如本文開頭所述,當一個線程修改了這個變量的值,volatile 保證了新值能當即同步到主內存,以及每次使用前當即從主內存刷新。但普通變量作不到這點,普通變量的值在線程間傳遞均須要經過主內存(詳見:Java內存模型)來完成。
2.禁止指令重排序優化。有volatile修飾的變量,賦值後多執行了一個「load addl $0x0, (%esp)」操做,這個操做至關於一個內存屏障(指令重排序時不能把後面的指令重排序到內存屏障以前的位置),只有一個CPU訪問內存時,並不須要內存屏障;(什麼是指令重排序:是指CPU採用了容許將多條指令不按程序規定的順序分開發送給各相應電路單元處理)。
volatile 的讀性能消耗與普通變量幾乎相同,可是寫操做稍慢,由於它須要在本地代碼中插入許多內存屏障指令來保證處理器不發生亂序執行
value ++這樣的操做並不具備原子性,其實際的過程以下:
sychroized
指定枷鎖對象:對給定對象加鎖,進入同步代碼前要得到給定對象的鎖
做用與靜態方法:至關於對當前類加鎖(全部實例)
直接做用與實例方法:至關於給當前實例加鎖(必須是同一個對象)