JVM虚拟机
一.简述JVM虚拟机的内存结构
JVM内存结构大致分为五个部分,分别为方法区、堆、虚拟机栈、程序计数器、本地方法栈。如下图:
1.程序计数器:
程序计数器主要作用是记录下一条要执行的二进制字节码指令地址。属于线程私有,不会出现内存溢出。
2.虚拟机栈:
- 定义:虚拟机栈内部是一个个的栈帧,每一个栈帧对应着一个方法的调用。栈是线程私有的,每个线程在创建的时候,都会创建一个虚拟机栈。
- 特性:正是因为java是基于栈设计的,才实现类跨平台的特性。
- 生命周期:栈的生命周期与线程一致。
- 保存数据:栈中会保存方法调用的栈帧,栈帧中包括局部变量表,操作数栈,动态链接,方法返回地址等。
- 垃圾回收:栈内存不会涉及垃圾回收。因为方法在执行完毕后,会被弹出栈空间。
- 内存溢出:栈是可能内存溢出的,一般是递归操作没有正确推出,因为栈帧过多会导致栈内存溢出;或者栈帧过大,也会导致内存溢出。
- 局部变量是否线程安全:如果没有逃离方法的作用域,那么就是线程安全的。如果变量引用了对象,并且逃离了方法作用域,那么要考虑线程安全问题。
3.本地方法栈
- 定义:JVM使用C/C++编写的,java代码通过本地方法native间接的调用实现,如clone(),hashCode();查看java源码用native修饰的;当调用native方法时,所需内存就是在本地方法栈分配。
- 说明:调用了本地方法后,内存是不受虚拟机控制的。
- 调用:当线程调用Java方法时,虚拟机会创建一个新的栈帧并压入Java栈。然而当他调用的是本地方法时,虚拟机会保持Java栈不变,不再在线程的Java栈中压入新的栈帧,虚拟机只是简单的动态连接并直接调用指定的本地方法。
4.堆
- 定义:堆是被所有线程共享的一块内存区域。通过new关键字创建的对象,都会使用堆内存。该内存区域唯一的目的,就是存放对象实例。
- 垃圾回收:是GC垃圾回收的重点区域。
- 说明:堆空间细分,还会分为老年代,新生代;新生代还可以分为Eden区,From Surivor区,ToSurivor区。
5.方法区(元空间)
- 定义:存储每个类的构造信息,譬如运行时的常量池,字段,方法数据以及方法和构造方法的代码,包含一些在类和实例初始化和接口初始化时候使用的特殊方法。
- 运行时常量池:常量池就是一张表,虚拟机指令根据这张常量表找到要执行的类名、方法名、参数类型,字面量等信息。常量池是*.class文件中的,当该类被加载,它的常量池信息就会放入运行时常量池,并把里面的符号地址变为真实地址。
- 位置:jdk1.8之前,直接使用的是堆内存;1.8之后使用的是操作系统内存。
- 作用:常量池 的作用是避免频繁的创建和销毁相同的对象,节省内存空间,节省运行时时间。比如:需要已存在的字符串,直接从常量池中取即可。
6.直接内存
常用于NIO操作,用于数据缓冲区,内存不受JVM内存管理,分配回收成本高,但是读写性能高。
二.Java中,堆和栈之间的区别
- 分配方式:堆内存是动态分配的,是由Java虚拟机JVM自动管理。栈内存是静态分配的,由编译器自动管理。
- 存储内容:堆内存用于存储对象实例和数组等引用类型数据。栈内存用于存储方法调用,局部变量和方法参数等基本类型数据。
- 内存管理:堆内存的分配和回收是由Java虚拟机JVM的垃圾回收机制负责的。栈内存的分配和释放是由编译器自动处理,当一个方法执行后,它所占用的栈内存会自动被释放。
- 空间大小:堆内存的空间较大,可以动态扩展。栈内存的空间较小,大小是固定的。
- 存储速度:堆内存的分配速度较慢,因为需要在运行时动态分配内存空间。栈内存的分配速度较快,因为它使用指针来定位数据。
三.简述J啊v啊中对象的深拷贝和浅拷贝
1.浅拷贝:
浅拷贝是指在对一个对象进行拷贝时,只拷贝对象本身和其中的基本数据类型,而不拷贝对象内部的引用类型。因此,在浅拷贝的对象中,引用类型的变量指向的依旧是原始对象中的引用。
在Java中,实现浅拷贝可以使用Object类的clone()方法,或者手动重写类的clone()方法。下面是一个示例:
class Person implements Cloneable {
private String name;
private Address address;
public Person(String name, Address address) {
this.name = name;
this.address = address;
}
@Override
protected Object clone() throws CloneNotSupportedException {
return super.clone();
}
}
class Address {
private String city;
public Address(String city) {
this.city = city;
}
}
public class Test {
public static void main(String[] args) throws Exception {
Address address = new Address("Beijing");
Person person1 = new Person("Tom", address);
Person person2 = (Person) person1.clone();
System.out.println(person1 == person2); // false
System.out.println(person1.getAddress() == person2.getAddress()); // true
}
}
在上述代码中,我们定义了一个Person类和Address类。Person类中包含一个引用类型的成员变量address。使用Object类的clone()方法进行拷贝,并且在Person类中实现了Cloneable接口,并重写了clone()方法,使得Person类可以进行拷贝。
在main函数中,我们创建了一个Person对象person1,其中包含一个Address对象。接着,我们使用person1的clone()方法创建了一个新的Person对象person2,并使用“”判断person1和person2是否是同一对象。结果为false,证明两个对象是不同的。接下来,我们使用“”判断person1和person2中的address是否是同一对象,结果为true,即两个对象的address成员变量指向的是同一个地址。
总的来说,浅拷贝只是将原始对象的引用类型成员变量复制到新的对象中,因此新对象中的引用类型成员变量与原始对象中的成员变量指向同一对象。如果原始对象中的引用类型成员变量发生变化,新对象中的引用类型成员变量也会随之变化。
2.深拷贝:
深拷贝是指在对一个对象进行拷贝时,不仅拷贝对象本身和其中的基本数据类型,同时也拷贝对象内部的引用类型。因此,在深拷贝的对象中,引用类型的变量指向的是全新的对象。
在Java中,实现深拷贝的方式比较多,可以使用对象的序列化、手动编写clone()方法等。下面是一个使用对象序列化来实现深拷贝的例子:
import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import java.io.Serializable;
public class Test {
public static void main(String[] args) throws Exception {
Address address = new Address("Beijing");
Person person1 = new Person("Tom", address);
Person person2 = (Person) deepCopy(person1);
System.out.println(person1 == person2); // false
System.out.println(person1.getAddress() == person2.getAddress()); // false
}
private static Object deepCopy(Object obj) throws Exception {
ByteArrayOutputStream bos = new ByteArrayOutputStream();
ObjectOutputStream oos = new ObjectOutputStream(bos);
oos.writeObject(obj);
ByteArrayInputStream bis = new ByteArrayInputStream(bos.toByteArray());
ObjectInputStream ois = new ObjectInputStream(bis);
return ois.readObject();
}
}
class Person implements Serializable {
private String name;
private Address address;
public Person(String name, Address address) {
this.name = name;
this.address = address;
}
public Address getAddress() {
return address;
}
}
class Address implements Serializable {
private String city;
public Address(String city) {
this.city = city;
}
}
在上述代码中,我们定义了一个Person类和Address类,并且让它们都实现了Serializable接口。在main函数中,我们创建了一个Person对象person1,其中包含一个Address对象。接着,我们使用deepCopy()方法创建了一个新的Person对象person2,并使用“”判断person1和person2是否是同一对象。结果为false,证明两个对象是不同的。接下来,我们使用“”判断person1和person2中的address是否是同一对象,结果为false,即两个对象的address成员变量指向的是不同的地址。
在上述代码中,我们使用了一个deepCopy()方法来实现对象的深拷贝。该方法使用对象的序列化和反序列化来实现深拷贝。首先,将原始对象序列化成字节数组,然后再将字节数组反序列化成新的对象。这样可以保证复制出的新对象与原始对象完全独立,不会相互影响。
3.总结:
浅拷贝和深拷贝是Java中常用的两种对象拷贝方式。浅拷贝只会复制对象的本身和内部的基本数据类型,而深拷贝会将对象内部所有的基本数据类型和引用类型都复制一份。使用clone()方法个重写clone()方法可以实现拷贝,使用对象序列化或手动赋值可以实现深拷贝。在实现对象拷贝时需要根据具体情况选择合适的拷贝方式,以保证复制出的新的对象能够满足应用需求。
标签:对象,虚拟机,Address,Person,内存,JVM,address,拷贝 From: https://www.cnblogs.com/chenlei210162701002/p/18394075