13 New一个对象的过程

迈不过友情╰ 2022-11-24 14:08 196阅读 0赞

new一个对象时发生了什么?

  • JVM遇到一条new指令时,首先会先查看对象所属的类有没有被加载到内存,如果没有的话,就会先通过类的全限定名来加载。加载并初始化类完成后,再进行对象的创建工作。
  • 创建对象
  1. 在堆区分配对象需要的内存:分配的内存包括本类和父类的所有实例变量,但不包括任何静态变量。
  2. 对所有实例变量赋默认值:将方法区内对实例变量的定义拷贝一份到堆区,然后赋默认值。
  3. 执行实例初始化代码:初始化顺序是先初始化父类再初始化子类,初始化时先执行实例代码块然后是构造方法
  4. 如果有类似于Cat c = new Cat()形式的c引用的话,在栈区定义Cat类型引用变量c,然后将堆区对象的地址赋值给它

new一个对象时发生了什么?

  • JVM遇到一条new指令时,首先会检查这个new指令的参数是否能在常量池中定位到一个类的符号引用,并且检查这个符号引用代表的类是否已被加载、解析和初始化过,如果没有,那必须先执行相应的类加载过程。
  • 类加载检查通过后,虚拟机将为新生对象分配内存,要把一块确定大小的内存从Java堆中划分出来,有两种分配方式:

    • ①指针碰撞:指的是如果Java堆中内存是绝对规整的,用过的内存放在一边,空闲的内存放在另一边,中间放着一个指针作为分界点的指示器,那分配内存就仅仅是把指针向空闲空间那边挪动一段与对象大小相等的距离,这种分配方式叫做指针碰撞。
    • ②空闲列表:指的是如果Java堆中内存并不是规整的,已使用的内存和空闲的内存相互交错,于是虚拟机维护一个列表来记录哪些内存块是可用的,在分配时就从列表中找出一块足够大的空间划分给对象实例,并更新列表上的记录,这种分配方式成为空闲列表。
    • 选择哪种分配方式由Java堆是否规整决定,而Java堆是否规整由所采用的垃圾收集器是否带有压缩整理功能来决定。在使用Serial, ParNew等带Compact过程的收集器时,系统采用分配算法是指针碰撞,而使用CMS这种基于Mark-Sweep 算法的收集器时,通常采用空闲列表。
  • 还需要考虑并发情况下的线程安全问题,解决这类问题有两种方案:

    • ①对分配内存空间的动作进行同步处理,也就是采用CAS配上失败重试的方式来保证更新操作的原子性。
    • ②把内存分配的动作按照线程划分在不同的空间中进行,也就是每个线程在Java堆中预先分配一小块内存,称为本地线程分配缓存,也叫TLAB,哪个线程要分配内存,就在哪个线程的TLAB上分配。
  • 内存分配完成后,虚拟机将分配到的内存空间都初始化为零值,这一步操作保证了对象的属性不赋初始值就可以直接使用,然后设置对象头(对象头中保存了对象是哪个类的实例,对应的哈希码,对象的GC分代年龄等信息),这样一个新的对象就产生了,一般执行 new 指令后会接着执行 init 方法,将对象初始化,这样一个真正可用的对象才算完成生成了。

new一个对象时给它分配多大的空间?

(JDK64位,创建一个对象占8+16=24字节,JDK32位 占4+16=20字节)

发表评论

表情:
评论列表 (有 0 条评论,196人围观)

还没有评论,来说两句吧...

相关阅读

    相关 13 New一个对象过程

    new一个对象时发生了什么? JVM遇到一条new指令时,首先会先查看对象所属的类有没有被加载到内存,如果没有的话,就会先通过类的全限定名来加载。加载并初始化类完成

    相关 JAVA new对象过程

    我们都知道,创建对象是由 new关键字调用构造方法 返回类实例。先实例化对象,再把对象初始化。 例如 : Person jack = new Person(); 这句话到底