JVM中内存主要划分为以下几个部分
网站建设哪家好,找创新互联公司!专注于网页设计、网站建设、微信开发、重庆小程序开发、集团企业网站建设等服务项目。为回馈新老客户创新互联还提供了当涂免费建站欢迎大家使用!例如
class Test2 {private int z = 0;
}
public class Test {public int x = 0;
public static int y = 0;
public static Test2 t2 = new Test2;
public static void main(String[] args){Test t = new Test();
}
}
局部变量在栈上
成员变量在堆上
静态变量在方法区中
因此
t是局部变量,在栈上
x,z是成员变量,在堆上
y,t2是静态变量,在方法区中
以下情景会触发类加载:
类加载的过程:
双亲委派模型JVM中包括了下面三个类加载器
例如cat类,先去标准库中找,找不到就去扩展库中找,找不到就去自定义类找,如果还没有找到,类加载就失败了
双亲委派模型避免了自己定义的类名和标准库中的类发生冲突时,无法正常加载标准库中的类的问题
内存回收之前C语言的博客中讲到了动态内存分配的问题,在我们malloc的时候很有可能忘记或者无法执行到free代码,导致内存泄漏。因此JVM及其他语言中有GC(垃圾回收)机制,保证我们申请的内存可以自动释放。相当于我们开车从手动挡变成了自动挡
栈中存放的都是局部变量,当代码块执行完毕后就会释放,因此不用回收,方法区中存放类对象,加载后也不太会卸载,程序计数器中是固定的内存空间,也不必进行回收,因此,GC主要回收堆中引用的对象
对于如何判定一个对象是否是垃圾,有下面几种方式
引用计数给每一个对象都加上一个计数器,当有一个变量引用了这个对象,计数器就加一,如果计数器变成0了,那么对象就该被回收了
缺点:
循环引用就是如下情况
class Test {Test t
}
Test a = new Test();
Test b = new Test();
a.t = b;
b.t = a;
a = null;
b = null;
a和b互相引用,这两个对象没有其他人引用,因此都无法访问到,这两个对象的程序计数器虽然都是1,但是事实上已经是垃圾了,因此程序计数器无法解决这个问题
可达性分析约定一些特定的变量(栈上的变量,常量值引用的对象,方法区引用的静态变量)为“GC roots”,从这些变量出发遍历访问所有的对象,如果能访问到就是“可达”的,否则就是“不可达”的,如果不可达,就是垃圾
先约定好一侧的内存是专门存放变量的(比如左侧),那么另一侧就是负责方便拷贝对象的。当一些对象是“不可达”时,就标记这些对象,然后将这些标记的对象释放掉,这时内存中会出现大量内存碎片,因此将左侧的资源拷贝到右侧,或者将这些内存碎片进行搬运,凑到一起
分代回收依据对象挺过GC可达性扫描的次数,可以分为下面几种
另外,如果创建的是一个很大的对象,那么这个对象会直接进入老年代,因为频繁的复制算法会有很大的开销
你是否还在寻找稳定的海外服务器提供商?创新互联www.cdcxhl.cn海外机房具备T级流量清洗系统配攻击溯源,准确流量调度确保服务器高可用性,企业级服务器适合批量采购,新人活动首月15元起,快前往官网查看详情吧
售后响应及时
7×24小时客服热线数据备份
更安全、更高效、更稳定价格公道精准
项目经理精准报价不弄虚作假合作无风险
重合同讲信誉,无效全额退款