JVM 的虚内存的标准划分通常会划分为 5 个区域,老师上课给简化了下,理解这三个内存区域就足够了。这里谈到的内存本质上是 JVM 的工作空间,和实际硬件层的 RAM 内存并不完全符合,所以叫做虚内存,但也是那么个意思啦。到这个程度也真没必要了,差不多就行了。
三大区域的细分?
- 栈区是按照线程来划分的,每个线程都拥有一个独立的栈。栈中以栈帧为基本单位,每个栈帧就是一个方法,通常还会存储一些局部变量等。平时用了那么多次递归,不就是这个原理吗?
- 堆区主要用来存储对象, 底层数据结构是采用的二叉树实现,足够稳定,插入和删除速度也很快。对象区分为实例属性区和实例方法区域,实例方法区存储的是类的方法代码在方法区的地址。static 修饰过的属性在静态数据区,这就解释了为什么一个类的所有对象都共享这个类的 static 数据。
- 方法区分代码区和数据区,代码区又可以划分为类代码区和共享代码区(类的方法代码是通用的,所以放在这里)。数据区可以划分为静态数据区和常量数据区( String 类型的数据和 Integer 这类包装类型数据)。
构造函数的执行流程?
public class StudentTest1 {
public static void main(String[] args){
Student s = new Student();
s = new Student("Java00001");
s.setAge(18);
s.age=25;
System.out.println(s.getAge());
}
}
上述创建对象 s 的过程中,系统会在堆区开辟好对应的空间,然后将 this 指针,即该空间的地址作为参数传递,并调用相应的构造函函数来给对象的属性值进行初始化。所以说,构造函数并不是用来创建对象的,它是用来给系统已经创建好的对象的属性值初始化的。
系统是根据参数签名来决定调用那个构造函数进行对象的初始化,参数签名由参数的个数、参数的类型、参数的顺序决定。
创建对象一共有几种方法呢?
- 使用 new 关键字创建对象,一定会调用构造函数
- 使用反射创建构造函数,一定会调用构造函数
- 克隆技术,不会调用构造函数,属于浅拷贝(部分复制)。和被克隆对象是同一块内存空间,也就是说开辟一个引用指向同一个内存空间
- 使用反序列化,不会调用构造函数,深拷贝(完全复制)
为什么没有 static 修饰的构造函数?
构造函数是系统创建对象之后,去调用它,来给对象进行初始化的。从这里就可以看出构造函数是依赖于对象而存在的,static 修饰的方法是和这个类进行静态绑定的,属于这个类所有对象共有的,static 方法不会接收 this 指针,自然也就不可能给实例变量进行初始化。反正 java 中是没有语法的,C# 倒是有。