<1> final类不能被继承,没有子类,final类中的方法默认是final的。
 final方法不能被子类的方法覆盖,但可以被继承。
 final成员变量表示常量,只能被赋值一次,赋值后值不再改变。(引用不可变,内容仍可变)
 final不能用于修饰构造方法。
 注意:父类的private成员方法是不能被子类方法覆盖的,因此private类型的方法默认是final类型的。

<2>volatile:

  当线程访问某一个对象时候值的时候,首先通过对象的引用找到对应在堆内存的变量的值,然后把堆内存变量的具体值

  load到线程本地内存中,建立一个变量副本,之后线程就不再和对象在堆内存变量值有任何关系,而是直接修改副本变量的值,

  在修改完之后的某一个时刻(线程退出之前),自动把线程变量副本的值回写到对象在堆中变量

  read and load 从主存复制变量到当前工作内存
  use and assign  执行代码,改变共享变量值 
  store and write 用工作内存数据刷新主存相关内容

对于volatile修饰的变量,jvm虚拟机只是保证从主内存加载到线程工作内存的值是最新的

<3>并发编程的几个问题:

原子性:即一个操作或者多个操作 要么全部执行并且执行的过程不会被任何因素打断,要么就都不执行。

可见性: 指当多个线程访问同一个变量时,一个线程修改了这个变量的值,其他线程能够立即看得到修改的值。

有序性:即程序执行的顺序按照代码的先后顺序执行

要想并发程序正确地执行,必须要保证原子性、可见性以及有序性。只要有一个没有被保证,就有可能会导致程序运行不正确

原子性保证:

Java内存模型只保证了基本读取和赋值是原子性操作,如果要实现更大范围操作的原子性,可以通过synchronized和Lock来实现。

由于synchronized和Lock能够保证任一时刻只有一个线程执行该代码块,那么自然就不存在原子性问题了,从而保证了原子性

可见性保证:

Java提供了volatile关键字来保证可见性。synchronized和Lock能保证同一时刻只有一个线程获取锁然后执行同步代码,并且在释放锁

之前会将对变量的修改刷新到主存当中。因此可以保证可见性。

有序性保证:

通过volatile关键字来保证一定的“有序性”,synchronized和Lock保证每个时刻是有一个线程执行同步代码,相当于是让线程顺序执行

同步代码,自然就保证了有序性

volatile的原理和实现机制:

1)它确保指令重排序时不会把其后面的指令排到内存屏障之前的位置,也不会把前面的指令排到内存屏障的后面;即在执行到内存屏

    障这句指令时,在它前面的操作已经全部完成;

2)它会强制将对缓存的修改操作立即写入主存;

3)如果是写操作,它会导致其他CPU中对应的缓存行无效。

<3>static修饰的变量被所有类的实例共享