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

指定枷锁对象:对给定对象加锁,进入同步代码前要得到给定对象的锁

做用与静态方法:至关于对当前类加锁(全部实例)

直接做用与实例方法:至关于给当前实例加锁(必须是同一个对象)

相关文章
相关标签/搜索