1.调用javac命令将.java文件编译成.class文件
2.调用java命令运行java程序
- 四个概念
- class文件
+ 硬盘上的.class文件加载入内存的那一块区域
+ 存入直接内存
+ Stream 源码:ClassFileStream * cfs = stream():
+ 解析
* parse_constant_pool 解析常量池
* parse_fields 解析属性
* parse_methods 解析方法
* parse_classfile_attribute
+ 解析玩之后会生成InstanceKlass对象
- javap -verbose
- java类在jvm中的存在形式
- 类的元信息:字面量,常量池,属性信息,方法信息
* class文件中的常量池
* 方法区
* 运行时常量池
- 符号引用:引用指向常量池
- 直接引用:引用指向内存地址
* 字符串常量池 StringTable
+ 反射获取的对象是InstanceMirrorKlass(镜像类)
- 静态变量就是存在镜像类中 - class对象
+ Class<?> clzz =Test.class;
+ 堆区
+ 四种方式:
3.对象在内存中的布局
- class文件
4.方法区是接口,规范
5.永久代,元空间是实现类,具体实现
- .永久代和元空间的区别:
- 永久代:
- jdk8以前方法区的实现
- 堆区
- 字符串
- 元空间
- jdk8及以后方法区的实现
- 直接内存
- 元空间调优原则:
- MetaspaceSize,MaxMetaspaceSize调成一样大
- 一般调成物理内存的1/32
- 或者通过两款工具:
- arthas: 纯黑窗口
- visualvm:有图形界面
-为什么用元空间取代永久代:
- 或者通过两款工具:
- GC算法不好实现
- 容易OOM
- 计算机硬件的升级
6.本地方法栈:
- java调用c,c++的动态链接库的函数需要的栈
- jni
- socket
虚拟机栈:
- 永久代:
- jvm中的虚拟机栈:一个线程一个jvm虚拟机栈
- 栈帧的个数与方法的调用次数有关
- 栈帧:
1. 局部变量表
2. 操作数栈
3. 动态链接,直接地之
- main方法对应的c++对象在方法区的内存地址
- 实现方式:
- 句柄池(windows系统):调用OpenProcess,打开handler
- 直接地址(java的实现方式)
4. 返回地址,恢复现场
-
main函数中调用add方法底层做了什么:
1. 创建栈帧
2. 将main函数的程序计数器保存至add方法的栈中
3. main函数的局部变量指针保存至add方法的栈中
4. add方法的栈帧赋值给线程的局部变量表开始的指针
5. main函数的操作数指针保存至add方法的栈中
6. add方法的操作数指针赋值给线程的操作数栈开始的指针
return做了什么:
1. 将main方法的局部表指针赋值给线程的局部表指针
2. 将main方法的操作数指针赋值给线程的操作数指针
3. 将add方法的返回值写入到main函数的操作数栈中(如果有返回值)
4. 将main函数的程序计数器赋值给执行引擎
5. 销毁add方法的栈帧
new 做了什么:
1.在堆中生成一个不完全对象(未执行默认的构造方法)
2.将对象压入到虚拟机栈中