标签:png 没有 ade volatile 构造 main方法 热点 安全相关 loader
jvm简称java虚拟机。jvm使用软件模拟java字节码的指令集。
jvm规范:定义了
---Class文件类型
---运行时数据
---帧栈
----虚拟机的启动
----虚拟机的指令集
vm指令集:类型转换,出栈入栈操作,运算,流程控制,函数调用。
jvm需要对java Library提供以下支持:
--反射java.lang.reflect
--ClassLoader
--初始化class和interface
--安全相关 java.security
--多线程
--弱引用
jvm编译:javap进行反汇编
jvm启动的流程:在当前路径和系统版本找jvm.cfg文件,根据配置找到JVM.dll,初始化JVM虚拟机获得JNIEnv接口,找到main方法开始运行。
jvm基本结构:
pc寄存器:
每个线程拥有一个pc寄存器。
在线程创建时创建。
指向下一条指令的地址。
执行本地方法时,pc的值为undefined。
方法区:
保存装载的类信息,通常和永久区关联在一起。
类型的常量池
字段,方法信息
方法字节码
java堆:
和程序开发密切相关,new出来的对象都在这里。
应用系统对象都保存在java堆中。
所有线程共享java堆。
对分代GC来说,堆也是分代的。
GC主要工作是区间。
java栈:
线程私有。
栈由一系列帧组成(因此栈也叫帧栈)
栈保存一个方法的局部变量,炒作数栈,常量池指针。
每一次方法调用创建一个帧,并压栈。
栈上分配:
Java栈 – 栈上分配,只能分配小对象
小对象(一般几十个bytes),在没有逃逸的情况下,可以直接分配在栈上
直接分配在栈上,可以自动回收,减轻GC压力
大对象或者逃逸对象无法栈上分配
内存模型:
每个线程有一个工作内存和主存独立
工作内存存放主存中变量值得拷贝
线程共享:
线程总是在自己的本地内存中存储变量,本地内存存储共享内存变量的一个副本,共享内存在主存当中的,共享变量存在时延和误差。
volatile 不能代替锁 一般认为volatile 比锁性能好(不绝对) 选择使用volatile的条件是: 语义是否满足应用
volatile 不是线程安全的 可见性 一个线程修改了变量,其他线程可以立即知道 保证可见性的方法 volatile synchronized (unlock之前,写变量值回主存) final(一旦初始化完成,其他线程就可见)
另外参考:http://www.cnblogs.com/43726581Gavin/p/9066080.html
有序性
在本线程内,操作都是有序的
在线程外观察,操作都是无序的。(指令重排 或 主内存同步延时)
指令重排 线程内串行语义 写后读 a = 1;b = a; 写一个变量之后,再读这个位置。 写后写 a = 1;a = 2; 写一个变量之后,再写这个变量。 读后写 a = b;b = 1; 读一个变量之后,再写这个变量。 以上语句不可重排,有依赖性的代码不可以重排 编译器不考虑多线程间的语义 可重排: a=1;b=2;
指令重排原则:
指令重排的基本原则
程序顺序原则:一个线程内保证语义的串行性
volatile规则:volatile变量的写,先发生于读
锁规则:解锁(unlock)必然发生在随后的加锁(lock)前
传递性:A先于B,B先于C 那么A必然先于C
线程的start方法先于它的每一个动作
线程的所有操作先于线程的终结(Thread.join())
线程的中断(interrupt())先于被中断线程的代码
对象的构造函数执行结束先于finalize()方法
解释运行
解释执行以解释方式运行字节码
解释执行的意思是:读一句执行一句
编译运行(JIT)
将字节码编译成机器码
直接执行机器码
运行时编译
编译后性能有数量级的提升
Java代码经过javac编译成class文件(字节码) class(字节码)文件经过JVM编译成机器码进行解释执行(解释执行) 对于热点代码,JIT(JustInTime)编译器会在运行时将其编译为机器码执行(编译运行) 从上面我们可以看出,在JVM默认的mixed模式下JAVA既不完全是解释运行也不完全是编译运行。
标签:png 没有 ade volatile 构造 main方法 热点 安全相关 loader
原文地址:https://www.cnblogs.com/q1359720840/p/10586583.html