存储过程如何在内部定义一个可重复使用的代码块_JVM01类加载过程以及运行时数据区的栈介绍...

简单介绍类加载过程:

JVM的内存结构:

不可共享:

虚拟机栈

本地方法栈

程序计数器

可共享:

方法区

a2d1d7b58323f619c425621cb4307c20.png

e90bf8ce7277914a9543f91fe7f8333e.png

dcfdab4bb1c0bb05ee7b711355d31bb9.png

238d3b50307c54c71920fc09bf2a5272.png

aab7ea034af0bd328229ece59878eb35.png

a2074eadc4474a883df2cbe71e90e01a.png

1. 类加载器的基本使用:加载,验证,准备,解析,初始化。

类加载器具有三个,引导类加载器,扩展类加载器,系统类加载器。

类加载器系统负责从文件系统或者网络中加载Class文件的加载,class文件在文件开头有特定的文件标识。

ClassLoader 只负责加载class文件,至于是否可以运行由 execution Engine 决定

加载后的类信息存放在 方法区中,常量池也存放于方法区中。

1.2 详细的类加载过程

43ca22053d20c861fb16169b68dfda3b.png

2. 双亲委派机制

String类的双亲委派机制,我们通过手写的一个String类的一个静态代码块打印,并通过另一个StringTest类去new一个String对象后,我们发现String的静态代码块没有被执行,因此执行的是Java核心类库中的String类。

2.1  双亲委派机制的第一种模型

双亲委派机制的原理是:

当一个类被加载进去之后交给类加载器,类加载器不会再第一时间进行加载,首先会寻找它的父类加载器,若还有父类则继续向上委托,把这个请求委托给父类的加载器去执行。

若是父类加载器可以完成类加载任务,就成功返回,倘若是父类加载器无法完成此加载任务,自加载器才会自己进行加载,这就是双亲委派机制。

public class Diparents {    public static void main(String[] args) {        String s =new String();        System.out.println("hello,atguigu.com");    } }

package java.llang; public class String {    static{        System.out.println("我是自定义的String类的静态代码块");    } }

直接向上进行委托,到引导类加载器,引导类加载器执行java核心库中的String.

系统类加载器:(AppClassLoader)

java 语言编写,派生与ClassLoader类

它负责加载环境变量classpath或者系统属性 java.class.path 指定路径下的类库。

通过ClassLoader # getSystemClassLoader()方法可以获得该类加载器。

扩展类加载器的加载路径:

派生于ClassLoader类

从java.ext.dirs系统属性指定的目录中进行加载或者从jre/lib/ext子目录下加载类库。

2.2  双亲委派机制的第二种模型】

5b6e1643f7071fc3db26ef1e87a5c962.png

2.3  双亲委派机制的优势是什么:

(1)防止类被重复加载,因为类加载器本身具有层次性,因此不可能存在多次加载。

(2)保护程序的安全性,防止核心API被随意的篡改。

自定义类:java.lang.String   不能拥有main方法,因此不能执行。

自定义类:java.lang.Sasd;   被引导类加载器进行加载后,不能执行。

2.4  什么是沙箱安全机制:

我们在使用引导类加载器对String类进行加载的时候,首先会找到JDK自带的文件(rt.jar)包中java\lang\String.class,报错信息会说没有main方法,就是因为加载的是rt.jar包中的类。可以对源码进行保护、因此叫沙箱安全机制。

2.5  两个类完全一样是怎么定义的

在jvm中定义的两个类完全一样:

1.   所在的包名是完全一样的

2. 所在的包中我们加载所用的类加载器是一样的,称之为两个类是完全一样的。

2.6  类的主动使用与被动使用

3. 运行时数据区:

3.1 PC寄存器

1. 使用PC寄存器存储字节码指令地址有什么用?为什么使用PC寄存器记录当前线程的执行地址呢?

CPU需要不断的切换线程,因此需要PC寄存器来记录到底到底执行到哪里了。

2. PC寄存器为什么设定为线程私有?

为了能够准确的记录当前字节码指令地址,最好的办法是为每一个线程都分配一个PC寄存器。防止之间相互干扰。

将指令地址(偏移地址) 存放在PC寄存器里面,执行引擎开始执行相应的操作。

作用:存放下一个指令的地址,有执行引擎读取下一条指令。

什么是并行跟串行的区别:就是垃圾回收线程跟用户线程不能并发的去执行。垃圾回收线程可以有很多个同时执行,但是此时就不能执行用户线程。

3.2 虚拟机栈

栈跟堆的区别是,栈是来解决程序运行问题,例如它应该怎么执行,如何处理数据,

堆是用来存储问题,数据怎么放,放在哪里。

 3.1   一个栈帧代表一个方法。

12e395c67545faeec1b427b7cd117db2.png

一次方法的调用伴随着栈的使用,声明周期是与线程是相同的。

作用:

保存方法的局部变量,部分结果,并参与方法的调用和返回。

栈不存在垃圾回收但是会存在OOM

3.2 java虚拟机栈中可能出现的异常

1. 可能会出现StackOverflowError异常,我们在java虚拟机中,如果线程请求分配的栈容量超过java虚拟机栈的最大容量后就会出现此异常

2. 如果java虚拟机栈可以动态扩展,并且在尝试扩展后无法申请到足够的内存,或者在创建新的线程时候没有足够的内存去创建对应的虚拟机栈。因此java虚拟机会抛出一个OutOfMemoryError异常。

3.3 如何设置栈内存的大小

我们可以设定参数-Xss 选项来设定线程的最大栈空间,栈的大小决定了函数调用的最大深度。

栈内部的结构:

栈中到底存储什么:

每个线程都有自己的栈,栈的数据存储是以栈帧的方式进行。

在这个线程上每个方法都有自己的栈帧,

栈帧是一个内存区块,是一个数据集,存放着方法执行过程中的各种数据信息。

main 主线程对应栈帧1

595460fb493d0bd6d9aacd232e5206c4.png

局部变量表(本地变量表):

1 局部变量数组,定义为一个数字数组,主要存储方法的形参,以及定义在方法体内的局部变量。这些数据类型包含了各类的基本数据类型,以及对象引用,以及returnAddress类型。

2 由于局部变量表是建立在线程的栈上,是线程的私有数据,因此不存在数据安全问题。

3 局部变量表所需的容量大小是在编译期确定下来的。并保存在方法的Code属性的maximum local variable数据项中。在方法运行期间是不会改变局部变量表的大小的。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值