java.jvm.自动内存管理机制.hotspot虚拟机对象.对象创建

  • 对象创建

这里的对象仅仅是普通java对象,不包括数组、Class对象

package com.wkh;

public class Test {
   public Test(int i, String bb) {
       
   }
}
package com.wkh;

public class Main {

   public static void main(String[] args) {
       new Test(23,"bb");
   }
}
  • public static void main(String[] args)的class字节码
        0: new           #2                  // class com/wkh/Test
        3: dup
        4: bipush        23
        6: ldc           #3                  // String bb
        8: invokespecial #4                  // Method com/wkh/Test."<init>":(ILjava/lang/String;)V
        11: pop
        12: return


  • 对象创建过程

当虚拟机遇到new指令的时候

  1. 首先去检查这个指令的参数(即#2)是否能在常量池中定义到一个类的符号引用
  2. 检查这个符号引用代表的类是否已经被加载、解析和初始化过,如果没有,那必须先执行相应的类加载过程
  3. 类加载检查通过后,接下来虚拟机将在为新生对象分配内存。对象所需内存的大小在类加载完成后便可以确定,为对象分配空间的任务等同于把一块确定大小的内存从java堆中划分出来。
  4. 内存分配完成后,虚拟机需要将分配到的内存空间都初始化为零(不包括对象头),如果使用TLAB,这一操作也可以提前至TLAB分配时进行。这一步操作保证了对象的实例字段在java代码中不赋初值就可以直接使用,程序能访问到这些字段的数据类型所对应的零值。
  5. 接下来,虚拟机对对象进行必要的设置,例如这个对象是哪个类的实例、如果才能找到类的元数据信息、对象的哈希码、对象的GC分代年龄等信息。这些信息存放在对象的对象头(Object Header)之中。根据虚拟机当前运行状态的不同,如是否启动偏向锁等,对象头会有不同的设置方式。

说明:1-5new指令的过程

  1. 上面的工作完成后,从虚拟机的视角来看,一个新的对象已经产生了,但从java程序的视角来看,对象创建才刚刚开始,因为<init>方法还没有执行,所有的字段都是零值。所以,一般来说,执行new指令之后,还会接着执行<init>方法,如上面的8: invokespecial #4 // Method com/wkh/Test."<init>":(ILjava/lang/String;)V
  • 下面是对象创建过程new指令的源码

等同于对象创建过程1-5

//jdk7u-dev/hotspot/src/share/vm/interpreter/bytecodeInterpreter.cpp
//确保常量池中存放的是已解释的类。
if (!constants->tag_at(index).is_unresolved_klass()) {
 // 断言确保是 klassOop 和 instanceKlassOop 
 oop entry = constants->slot_at(index).get_oop();
 assert(entry->is_klass(), "Should be resolved klass");
 klassOop k_entry = (klassOop) entry;
 assert(k_entry->klass_part()->oop_is_instance(), "Should be instanceKlass");
 instanceKlass* ik = (instanceKlass*) k_entry->klass_part();
 // 确保对象所属类已经经过初始化阶段
 if ( ik->is_initialized() && ik->can_be_fastpath_allocated() ) {
  // 取对象长度
   size_t obj_size = ik->size_helper();
   oop result = NULL;
   // 如果TLAB没有把内存中的值置零,那么这里就需要置零
   bool need_zero = !ZeroTLAB;
  // 是否使用TLAB,是否在TLAB中分配对象
   if (UseTLAB) {
     result = (oop) THREAD->tlab().allocate(obj_size);
   }
   if (result == NULL) {
     need_zero = true;
     // 直接在垃圾收集器中的eden空间分配对象
retry:
     HeapWord* compare_to = *Universe::heap()->top_addr();
     HeapWord* new_top = compare_to + obj_size;
    // cmpxchg是x86中的CAS指令,这里是一个C++方法,通过CAS方式(同步)分配空间,如果并发失败,转到retry中重新尝试,直至成功分配为止。
     if (new_top <= *Universe::heap()->end_addr()) {
       if (Atomic::cmpxchg_ptr(new_top, Universe::heap()->top_addr(), compare_to) != compare_to) {
         goto retry;
       }
       result = (oop) compare_to;
     }
   }
   if (result != NULL) {
     // 如果需要,则为对象初始化零值
     if (need_zero ) {
       HeapWord* to_zero = (HeapWord*) result + sizeof(oopDesc) / oopSize;
       obj_size -= sizeof(oopDesc) / oopSize;
       if (obj_size > 0 ) {
         memset(to_zero, 0, obj_size * HeapWordSize);
       }
     }
    // 根据是否启用偏向锁来设置对象头信息
     if (UseBiasedLocking) {
       result->set_mark(ik->prototype_header());
     } else {
       result->set_mark(markOopDesc::prototype());
     }
     result->set_klass_gap(0);
     result->set_klass(k_entry);
    // 将对象引入栈,继续执行下一条指令
     SET_STACK_OBJECT(result, 0);
     UPDATE_PC_AND_TOS_AND_CONTINUE(3, 1);
   }
 }
}
  • java堆分配内存的2种方式
  1. 指针碰撞(Bump the Point)
    如果java堆中内存是绝对规整的,用过的内存放一边,空闲的内存放另一边,中间放着指针作为边界,那么分配内存仅仅是把指针向着空闲那边挪动一段与对象大小相等距离,这种分配方式叫做指针碰撞
  2. 空闲列表(Free List)
    如果java堆不是绝对规整的,已使用和空闲的内存交错,那么虚拟机就必须维护列表,上面记录哪些内存块是可用的,在分配的时候找一块足够大空间划分给对象实例,并且更新表上记录,这种分配方式叫做空闲列表
  3. 采用哪种内存分配方式
    选择哪种分配方式是由java堆是否规整决定的,而java堆是否规整由垃圾收集器是否带有压缩整理功能决定。因此,在使用Serial、ParNew等带Compact过程的收集器时,系统采用指针碰撞;而使用CMS这种基于Mark-Sweep算法的收集器时,通常采用空闲列表。
  • 关于java堆分配内存时线程不安全的的问题
  1. 例子
    正在给对象A分配内存,指针还未进行修改,B同时又使用了原来的指针分配内存。
  2. 解决办法1-同步
    对分配内存空间的动作进行同步处理,实际上虚拟机采用CAS配上失败重试的方式保证更新操作的原子性
  3. 解决办法2-TLAB
    把内存分配的动作按照线程划分在不同的空间中进行,即每个线程在java堆中预先分配一小块内存,称为线程私有分配缓冲(Thread Local Allocation Buffer,TLAB)。
    哪个线程需要分配内存,就在哪个线程的TLAB上分配,只有TLAB用完并分配新的TLAB时,才需要同步锁定。虚拟机是否使用TLAB,可以通过-XX:+/-UseTLAB参数设定。
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,923评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,154评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,775评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,960评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,976评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,972评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,893评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,709评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,159评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,400评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,552评论 1 346
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,265评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,876评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,528评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,701评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,552评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,451评论 2 352

推荐阅读更多精彩内容

  • oop-klass模型 Hotspot 虚拟机在内部使用两组类来表示Java的类和对象。 oop(ordinary...
    CodeKing2017阅读 3,696评论 1 6
  • jstack-- 如果java程序崩溃生成core文件,jstack工具可以用来获得core文件的java sta...
    not_null阅读 7,967评论 2 62
  • 在java语言层面的一个对象实例对应一个JVM中的instanceOopDesc ;参考/openjdk/hots...
    橡树人阅读 619评论 0 1
  • 第二天一早陆寻就从一阵疼痛中醒来,他看向自己的胳膊发现血管的纹路已经往外扩散呈紫色,他忽然想起老道士的那句话,三日...
    滕y阅读 166评论 0 0
  • 一、函数的概念 代码: 1.print(" _ooOoo_ ") 2.print(" o8888888o ") 3...
    __晴天___阅读 269评论 0 2