7.final

1. final的指令重排

对于 final 域,编译器和处理器要遵守两个重排序规则。

  1. 在构造函数内对一个final变量赋值,与随后把这个被构造对象的引用赋值给一个变量,这两个操作之间不能重排序
  2. 初次读一个包含final变量的对象,与随后初次读这个final变量,这两个操作之间不能重排序
public class FinalExample {
       int i;                    // 普通变量
       final int j;                // final 变量
       static FinalExample obj;
       public FinalExample () {          // 构造函数
              i = 1;               // 写普通域
              j = 2;               // 写 final 域
       }
       public static void writer () {    // 写线程 A 执行
              obj = new FinalExample ();
       }
       public static void reader () {    // 读线程 B 执行
              FinalExample object = obj;  // 读对象引用
              int a = object.i;      // 读普通域
              int b = object.j;      // 读 final 域
       }
}

1.1. 赋值final属性的指令重排

赋值final属性的重排序规则禁止把final属性的赋值操作重排序到构造函数之外,这个规则的实现包含下面2个方面

  1. JMM 禁止编译器把 final属性的赋值操作重排序到构造函数之外
  2. 编译器会在 final 域的写之后,构造函数 return 之前,插入一个 StoreStore 屏障。这个屏障禁止处理器把 final 域的写重排序到构造函数之外

写 final 域的重排序规则可以确保:在对象引用为任意线程可见之前,对象的 final 域已经被正确初始化过了,而普通域不具有这个保障

1.2. 读取final属性的指令重排

读 final 域的重排序规则是,在一个线程中,初次读对象引用与初次读该对象包含的 final 域,JMM 禁止处理器重排序这两个操作.编译器会在读 final 域操作的前面插入一个 LoadLoad 屏障

1.3. final属性为引用类型

对于引用类型,赋值final属性的重排序规则对编译器和处理器增加了如下约束:
在构造函数内对一个引用类型的final属性的赋值,与随后在构造函数外把这个被构造对象的引用赋值给一个引用变量,这两个操作之间不能重排序

RROR o.r.c.h.ErrorsLoggingHandler: - Exception occured. Channel: [id: 0x534f733e, L:/10.138.46.28:59780 - R:7.210.163.89/7.210.163.89:6379] io.netty.handler.codec.DecoderException: java.io.IOException: Unsupported protocol version 115 at io.netty.handler.codec.ReplayingDecoder.callDecode(ReplayingDecoder.java:421) ~[netty-codec-4.1.118.Final.jar:4.1.118.Final] at io.netty.handler.codec.ByteToMessageDecoder.channelRead(ByteToMessageDecoder.java:290) ~[netty-codec-4.1.118.Final.jar:4.1.118.Final] at io.netty.channel.AbstractChannelHandlerContext.invokeChannelRead(AbstractChannelHandlerContext.java:444) ~[netty-transport-4.1.118.Final.jar:4.1.118.Final] at io.netty.channel.AbstractChannelHandlerContext.invokeChannelRead(AbstractChannelHandlerContext.java:420) ~[netty-transport-4.1.118.Final.jar:4.1.118.Final] at io.netty.channel.AbstractChannelHandlerContext.fireChannelRead(AbstractChannelHandlerContext.java:412) ~[netty-transport-4.1.118.Final.jar:4.1.118.Final] at io.netty.channel.DefaultChannelPipeline$HeadContext.channelRead(DefaultChannelPipeline.java:1357) ~[netty-transport-4.1.118.Final.jar:4.1.118.Final] at io.netty.channel.AbstractChannelHandlerContext.invokeChannelRead(AbstractChannelHandlerContext.java:440) ~[netty-transport-4.1.118.Final.jar:4.1.118.Final] at io.netty.channel.AbstractChannelHandlerContext.invokeChannelRead(AbstractChannelHandlerContext.java:420) ~[netty-transport-4.1.118.Final.jar:4.1.118.Final] at io.netty.channel.DefaultChannelPipeline.fireChannelRead(DefaultChannelPipeline.java:868) ~[netty-transport-4.1.118.Final.jar:4.1.118.Final] at io.netty.channel.nio.AbstractNioByteChannel$NioByteUnsafe.read(AbstractNioByteChannel.java:166) ~[netty-transport-4.1.118.Final.jar:4.1.118.Final] at io.netty.channel.nio.NioEventLoop.processSelectedKey(NioEventLoop.java:796) ~[netty-transport-4.1.118.Final.jar:4.1.118.Final] at io.netty.channel.nio.NioEventLoop.processSelectedKeysOptimized(NioEventLoop.java:732) ~[netty-transport-4.1.118.Final
最新发布
06-26
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值