1、泛型出现之前存在的问题:
所有对象的类型都继承自Object,虚拟机只有到运行时才能知道这个Object具体是什么类型,在编译期是无法检查这个Object是否强制转型成功,会将ClassCaseException的风险转移到程序运行期。
2、泛型的作用:
通过泛型,编译器可以在编译阶段发现类型不一致的问题
3、泛型擦除:
将Java代码编译成Class文件,通过反编译发现泛型都不见了,被替换为原生类型,并插入强制转型的代码。
//泛型擦除前 List<String> list = new ArrayList<>(); list.add("hello"); System.out.println(list.get(0)); //泛型擦除后 List list1 = new ArrayList(); list1.add("hello"); System.out.println((String) list1.get(0)); 复制代码
1、公共子表达式消除 在程序基本块中,如果一个表达式E已经被计算过了,下次再次使用的时候,如果表达式的变量值都没发生改变,就可以直接拿表达式的结果来代替E。
int x = 1; int y = 2; int z = x + y; int w1 = x + y +2; //编译器对公共子表达式(x+y)进行消除 int w2 = z + 2; 复制代码
类加载解析节点,将一部分符号引用转化为直接引用。前提是程序运行前有可确定的调用版本,并且在运行期不可变。这些编译期可知、运行期不可变的方法调用就是解析。
1、静态分派:
根据静态类型来定位方法的分派叫做静态分派,发生在编译阶段。
//父类 public class Parent { } //子类 public class Son extends Parent { } //调用 public class MyTest { public void say(Parent parent) { System.out.println("parent say"); } public void say(Son son) { System.out.println("son say"); } public static void main(String[] args) { MyTest myTest = new MyTest(); //实际类型为Parent Parent parent = new Parent(); //实际类型为Son Parent son = new Son(); myTest.say(parent); myTest.say(son); } } 复制代码
返回结果:
Parent为变量的静态类型,Son为实际类型。其中静态类型是在编译期可知的,而实际类型是在运行期确定下来的,编译器在编译阶段不知道某个对象的实际类型是什么,所以是用静态类型作为判定依据来选择使用哪个重载版本的,所以选择了say(Parent)作为调用目标。
2、动态分派
public class Parent { public void say() { System.out.println("parent...."); } } public class Son extends Parent { public void say() { System.out.println("son...."); } } //调用 Parent parent = new Parent(); Parent son = new Son(); parent.say(); son.say(); 复制代码
结果:
虚拟机根据实际类型的不同来分派方法
基本步骤:
将变量从主内存拷贝到工作内存中,将工作内存同步到主内存中。定义了8中操作,每步操作都是原子的、不可再分。
注 :
1、可见性:
private volatile static int x; public static void main(String[] args) { for (int i = 0; i < 20; i++) { Thread thread = new Thread(new Runnable() { @Override public void run() { for (int i1 = 0; i1 < 1000; i1++) { x++; } } }); thread.start(); } System.out.println("x="+x); } 复制代码
最终的结果不是20000,说明volatile修饰的变量也没实现正确并发的目的。
原因:
x++ 是由多条字节码指令构成的,包括取值,+1,赋值操作,volatile只能保证最后变量取到操作栈顶时该变量的同步性,但是在这之前其他线程是可以修改该变量的值。
2、volatile适用的场景:
3、禁止指令重排序优化 普通变量只能保证执行过程所有依赖赋值结果的地方都能得到正确的结果,不能保证变量赋值的顺序与代码中执行顺序一致,
实现方式:在多线程访问同一内存时,相当于通过一个内存屏障,保证不能把后面的指令重排序到内存屏障之前的位置。