`
BruceChan_GY
  • 浏览: 7284 次
社区版块
存档分类
最新评论

《Java并发编程的艺术》读书笔记二

    博客分类:
  • JVM
 
阅读更多
java的对象构成


对于JVM来说,构造JAVA对象时,是以oops-klass二分模型来构建的,其中oops表示对象的相关信息。

基本结构如下:

class oopDesc {
  friend class VMStructs;
private:
  volatile markOop  _mark;
  union _metadata {
    wideKlassOop    _klass;
    narrowOop       _compressed_klass;
  } _metadata;


对于数组来说,其结构如下:
class arrayOopDesc : public oopDesc

从源码上没找到length字段,我们只能看注释:
// The layout of array Oops is:
//
//  markOop
//  klassOop  // 32 bits if compressed but declared 64 in LP64.
//  length    // shares klass memory or allocated after declared fields.

还有一段:
  // The _length field is not declared in C++.  It is allocated after the
  // declared nonstatic fields in arrayOopDesc if not compressed, otherwise
  // it occupies the second half of the _klass field in oopDesc.


这里的mark字段,用来表示一些JVM内部需要的相关信息

从OpenJDK源码中可以看到:

看下注释:

//  32 bits:
//  --------
//             hash:25 ------------>| age:4    biased_lock:1 lock:2 (normal object)

//  64 bits:
//  --------
//  unused:25 hash:31 -->| unused:1   age:4    biased_lock:1 lock:2 (normal object)

//  unused:25 hash:31 -->| cms_free:1 age:4    biased_lock:1 lock:2 (COOPs && normal object)

我们运行的时候,默认是使用UseCompressedOops的,所以对应COOPs

我的机器是64bits的,测试下 normal object的mark:

源码如下:

public class HashCodeMemViewDemo {

                 public static void main(String[] args) throws IOException {
                                DataEntity data = new DataEntity();
                                 data.setDataId(0x01020304);
                                
                                System. out.println(data .hashCode());
                                
                                System. in.read();
                }
}

class DataEntity {
                 private int dataId;
                 public int getDataId() {
                                 return dataId ;
                }
                 public void setDataId(int dataId) {
                                 this.dataId = dataId;
                }
}


运行后输出的值为:1359740460
(对应的二进制值为:1010001000010111111111000101100)

查看对象DataEntity的内存布局:

hsdb> mem 0xf5980a58 2
0x00000000f5980a58: 0x000000510bfe2c01
0x00000000f5980a60: 0x01020304dbc99c30

0x000000510bfe2c01 就是DataEntity对象的mark字段值,对应的二进制值为(已拆分):

0000000000000000000000000(unused:25)
1010001000010111111111000101100(hash:31)
0(cms_free:1)
0000(age:4)
0(biased_lock:1)
01(lock:2)

这里的(hash:31),根据 程序中输出的值 对比, 就是DataEntity对应的hash code值了。

另外,这里的hash是 System.identityHashCode

其中一份PPT里提到:

● First word of every object is the mark word
● Used for synchronization and garbage collection
     ● Also holds identity hash code if computed



因为mark字段中会存储对象的age,做个测试:

同样的代码,增加GC。

GC前:

identity hash code: 1359740460
DataEntity addr: 0xf5980b48

hsdb> mem 0xf5980b48 2
0x00000000f5980b48: 0x000000510bfe2c01
0x00000000f5980b50: 0x01020304dbc99cc0

mark字段解析后:
0000000000000000000000000(unused:25)
1010001000010111111111000101100(hash:31)
0(cms_free:1)
0000(age:4)
0(biased_lock:1)
01(lock:2)

经过1次GC:

DataEntity addr: 0xe0c00838
地址已经变了

hsdb> mem 0xe0c00838 2
0x00000000e0c00838: 0x000000510bfe2c09
0x00000000e0c00840: 0x01020304dbc99930

0000000000000000000000000
1010001000010111111111000101100
0
0001(age:4)
0
01

因为mark字段还用来表示锁相关信息,目前不介绍锁的细节,只用一个例子来说明一点问题。

代码如下:

public class LockMemViewDemo {

                 public static void main(String[] args) throws IOException {
                                DataEntity data = new DataEntity();
                                 data.setDataId(0x01020304);

                                 synchronized (data ) {
                                                System. out.println(System.identityHashCode( data));
                                                System. in.read();
                                }
                }
}

运行后输出的值为:353537765

查看对象DataEntity的内存布局:

hsdb> mem 0xf5980b78 2
0x00000000f5980b78: 0x00007f69cc004c1a
0x00000000f5980b80: 0x01020304dbc99ca0

其中mark字段的值为:0x00007f69cc004c1a ,对应的二进制值为(已拆分):
00000000000000000111111101101001110011000000000001001100000110
10

按照源码中的注释:
//    [ptr             | 10]  monitor            inflated lock (header is wapped out)

因此第一段就是指向monitor的指针,是0x1FDA73001306,具体是什么,指向哪个对象,无法验证。

有一段供参考:
In the uncontended case, a Java mutex consists of just 2 bits in the flags word. The JVM only associates a heavy-weight OS lock object with a Java mutex when the mutex is contended, and then releases the OS lock when the mutex has been exited by all threads.


参考:
http://arturmkrtchyan.com/java-object-header
http://stackoverflow.com/questions/4068562/how-heavy-are-java-monitors
http://www.programmershare.com/2077632/
分享到:
评论

相关推荐

    java并发编程实践pdf笔记

    这本书的读书笔记涵盖了多个关键知识点,旨在帮助读者深入理解Java并发编程的核心概念。 1. **线程和进程的区别** - **线程** 是程序执行的最小单位,一个进程中可以有多个线程同时执行,共享同一块内存空间,通信...

    Java并发编程学习笔记.rar

    这本"Java并发编程学习笔记"可能是作者在深入研究Java并发特性、工具和最佳实践过程中积累的心得体会。下面,我们将根据这个主题,探讨一些关键的Java并发编程知识点。 1. **线程与进程**:在多任务环境中,线程是...

    Java并发编程系列心得笔记

    Java并发编程系列心得笔记,可以参考,欢迎共同交流学习

    java并发编程实践笔记

    ### Java并发编程实践笔记知识点详解 #### 一、保证线程安全的方法 1. **不要跨线程访问共享变量:** 当多个线程共享某个变量时,若其中一个线程修改了该变量,其他线程若没有正确同步,则可能读取到错误的数据。...

    java并发编程的艺术读书笔记

    java并发编程的艺术读书笔记根据章节整理的核心内容,便于自己理解

    Java并发编程学习笔记

    Java并发编程是Java开发中必不可少的一部分,涉及到多线程、同步机制、线程池以及并发工具类等多个核心知识点。以下是对这些主题的详细说明: 1. **线程安全与锁 Synchronized 底层实现原理**: 线程安全是指在多...

    读书笔记:Java并发编程之美笔记.zip

    读书笔记:Java并发编程之美笔记

    Java并发编程笔记

    ### Java并发编程知识点详解 #### 一、线程状态与管理 在Java中,线程具有多种状态,这些状态的变化反映了线程在其生命周期中的不同阶段。理解这些状态及其转换对于编写高效、健壮的并发程序至关重要。 - **NEW**...

    Java并发编程的艺术.md

    《Java并发编程的艺术》笔记 第一章 并发编程的挑战 第二章 Java并发机制的底层实现原理 volatile的两条实现原则: 1. Lock前缀指令会引起处理器缓存回写到内存 2. 一个处理器的缓存回写到内存会导致其他...

    读书笔记-Java并发编程实战-基础篇

    在Java并发编程中,数据的封装与访问控制、线程安全性的考量、同步机制的使用是重要的基础概念和技巧。以下是从给出的文件内容中提取出的详细知识点: 1. 数据封装与访问控制:确保内部私有数据不被轻易访问,并且...

    Java并发编程实战-读书笔记

    《Java并发编程实战》个人读书笔记,非常详细: 1 简介 2 线程安全性 3 对象的共享 4 对象的组合 5 基础构建模块 6 任务执行 7 取消与关闭 8 线程池的使用 9 图形用户界面应用程序 10 避免活跃性危险 11 性能与可...

    java并发编程实践笔记资料.pdf

    Java并发编程实践笔记 Java并发编程实践笔记是一份关于Java并发编程的实践笔记,涵盖了多种关于线程安全、并发编程的实践经验和原则。下面是从笔记中总结的知识点: 1. 保证线程安全的三种方法:不要跨线程访问...

    java并发编程笔记

    ### Java并发编程笔记 #### 实现Runnable接口与继承Thread类 在Java中,实现多线程功能主要有两种途径:一种是通过实现`Runnable`接口,另一种则是通过继承`Thread`类。这两种方式各有优劣,具体选择哪一种取决于...

    《java并发编程实战》读书笔记-第3章-对象的共享

    《java并发编程实战》读书笔记-第3章-对象的共享,脑图形式,使用xmind8制作 包括可见性、发布与逸出、线程封闭、不可变性、安全发布等内容

    java并发编程学习笔记

    ### Java并发编程学习笔记知识点详解 #### 一、Java并发编程概述 Java并发编程是指在Java应用程序中同时执行多个操作的技术。它通过多线程、线程池等机制实现资源的有效利用,提高程序运行效率。Java并发编程的...

    Java并发编程与高并发解决方案-学习笔记-www.itmuch.com.pdf

    本文将基于文档《Java并发编程与高并发解决方案-学习笔记***.pdf》中提供的内容,来详细阐述并发编程和高并发的基本概念、CPU多级缓存与缓存一致性、以及Java内存模型。 ### 并发与高并发概念 在现代多线程编程中...

    读书笔记:Java并发编程之美读书笔记.zip

    读书笔记:Java并发编程之美读书笔记

    《java并发编程实战》读书笔记-第2章-线程安全性

    《java并发编程实战》读书笔记-第2章-线程安全性,脑图形式,使用xmind8制作 包括引言、线程安全性定义、原子性、加锁机制、使用锁保护状态、活跃性与性能等内容

    Java并发编程(学习笔记).xmind

    Java并发编程 背景介绍 并发历史 必要性 进程 资源分配的最小单位 线程 CPU调度的最小单位 线程的优势 (1)如果设计正确,多线程程序可以通过提高处理器资源的利用率来提升系统吞吐率 ...

Global site tag (gtag.js) - Google Analytics