面试官:一个Java对象占用多大内存?

2024-04-17 22:20

本文主要是介绍面试官:一个Java对象占用多大内存?,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!


程序员的公众号:源1024获取更多资料,无加密无套路!

最近整理了一波电子书籍资料,包含《Effective Java中文版 第2版》《深入JAVA虚拟机》,《重构改善既有代码设计》,《MySQL高性能-第3版》,《Java并发编程实战》等等
获取方式: 关注公众号并回复 电子书 领取,更多内容持续奉上


关于new Object()的灵魂追问,一个Java对象占用多大内存?被面试官这么一问是不是有点蒙?

我们都知道Java是面向对象开发,几乎每天都在new对象,但却不知道对象的体重,实属有点尴尬,今天我们重新来认识下这个陪伴我们职业生涯的对象!

Java对象由对象头、类型指针、实例数据、填充数据组成,虚拟机要求对象大小必须是8字节的整数倍,为了保证对象中的成员变量都能够自然对齐。

图片

mark word

占固定8个字节,主要用于标记对象的锁信息、hashcode、GC标记(三色标记)。

class pointer

类型指针占位4个字节或者8个字节(看是否开启类型指针压缩),开启类型指针压缩占位4个字节,反之占位8个字节。JDK1.8默认是开启类型指针压缩的。

通过以下命令查看java命令默认的启动参数

java -XX:+PrintCommandLineFlags -version

图片

-XX:+UseCompressedClassPointers //开启压缩类指针-XX:-UseCompressedClassPointers //关闭压缩类指针-XX:+UseCompressedOops  //开启压缩普通对象指针-XX:-UseCompressedOops  //关闭压缩普通对象指针

instance data

实例数据是存储对象真正的有效信息,如果是基本类型,直接存储下来,如果是引用类型,存储的是指向堆中对象的引用指针。非静态属性,生成对象时就是实例数据。

java分为两种数据类型:基本类型和引用类型

8种基本类型

类型占用空间(byte)
boolean1
byte1
short2
char2
int4
float4
long8
double8

引用类型

对应的就是oops(Ordinary Object Pointers, OOPs)—— 普通对象指针,JVM开启普通对象指针压缩时为4个字节,反之为8个字节。

空对象Object
内容32位64位
mark word4byte8byte
类型指针4byte未开启压缩为8bit,开启压缩为4byte
实例数据00
填充数据(将最后的大小填充为8的倍数)0~7byte0~7byte
总计8byte开启压缩后,则8+4=12,对齐填充为8的倍数,最后为 16byte

下面我们通过代码进行验证,使用 JOL 工具分析 Java 对象

maven依赖

<dependency><groupId>org.openjdk.jol</groupId><artifactId>jol-core</artifactId><version>0.2</version>
</dependency>
 

新建Demo类

public class Demo {public static void main(String[] args){ClassLayout classLayout = ClassLayout.parseInstance(new Demo());System.out.println(classLayout.toPrintable());}
}

输出结果


com.lly835.bestpay.model.Demo object internals:OFFSET  SIZE   TYPE DESCRIPTION                               VALUE0     4        (object header)                           01 00 00 00 (00000001 00000000 00000000 00000000) (1)4     4        (object header)                           00 00 00 00 (00000000 00000000 00000000 00000000) (0)8     4        (object header)                           05 c1 00 f8 (00000101 11000001 00000000 11111000) (-134168315)12     4        (loss due to the next object alignment)
Instance size: 16 bytes
Space losses: 0 bytes internal + 4 bytes external = 4 bytes total

我们看到Demo对象占16字节,对象头12字节+填充(4字节)

新建Demo2类

定义4个类型属性:boolean、int、double、Integer


public class Demo2 {private boolean flag = true;private int a = Integer.MAX_VALUE;private double b = 0.01;private Integer c = 5;public static void main(String[] args){ClassLayout classLayout = ClassLayout.parseInstance(new Demo2());System.out.println(classLayout.toPrintable());}
}

输出结果


com.lly835.bestpay.model.Demo2 object internals:OFFSET  SIZE                TYPE DESCRIPTION                               VALUE0     4                     (object header)                           01 00 00 00 (00000001 00000000 00000000 00000000) (1)4     4                     (object header)                           00 00 00 00 (00000000 00000000 00000000 00000000) (0)8     4                     (object header)                           05 c1 00 f8 (00000101 11000001 00000000 11111000) (-134168315)12     4                 int Demo2.a                                   214748364716     8              double Demo2.b                                   0.0124     1             boolean Demo2.flag                                true25     3                     (alignment/padding gap)                  28     4   java.lang.Integer Demo2.c                                   5
Instance size: 32 bytes
Space losses: 3 bytes internal + 0 bytes external = 3 bytes total

我们看到Demo2对象占32字节,对象头12字节+int(4字节)+double(8字节)+boolean(1字节)+填充(3字节)+ Integer(4字节)

一个Java对象占用多大内存空间,你学会了吗?


 系列文章索引

MyBatis的插件能在哪些地方进行拦截?

了解MyBatis的缓存机制吗

面试官:谈谈对volatile的理解

Spring中用到了哪些设计模式

面试官:说一下SQL的执行过程

线程池的工作原理


 

这篇关于面试官:一个Java对象占用多大内存?的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/913014

相关文章

JVM 的类初始化机制

前言 当你在 Java 程序中new对象时,有没有考虑过 JVM 是如何把静态的字节码(byte code)转化为运行时对象的呢,这个问题看似简单,但清楚的同学相信也不会太多,这篇文章首先介绍 JVM 类初始化的机制,然后给出几个易出错的实例来分析,帮助大家更好理解这个知识点。 JVM 将字节码转化为运行时对象分为三个阶段,分别是:loading 、Linking、initialization

Spring Security 基于表达式的权限控制

前言 spring security 3.0已经可以使用spring el表达式来控制授权,允许在表达式中使用复杂的布尔逻辑来控制访问的权限。 常见的表达式 Spring Security可用表达式对象的基类是SecurityExpressionRoot。 表达式描述hasRole([role])用户拥有制定的角色时返回true (Spring security默认会带有ROLE_前缀),去

浅析Spring Security认证过程

类图 为了方便理解Spring Security认证流程,特意画了如下的类图,包含相关的核心认证类 概述 核心验证器 AuthenticationManager 该对象提供了认证方法的入口,接收一个Authentiaton对象作为参数; public interface AuthenticationManager {Authentication authenticate(Authenti

Spring Security--Architecture Overview

1 核心组件 这一节主要介绍一些在Spring Security中常见且核心的Java类,它们之间的依赖,构建起了整个框架。想要理解整个架构,最起码得对这些类眼熟。 1.1 SecurityContextHolder SecurityContextHolder用于存储安全上下文(security context)的信息。当前操作的用户是谁,该用户是否已经被认证,他拥有哪些角色权限…这些都被保

Spring Security基于数据库验证流程详解

Spring Security 校验流程图 相关解释说明(认真看哦) AbstractAuthenticationProcessingFilter 抽象类 /*** 调用 #requiresAuthentication(HttpServletRequest, HttpServletResponse) 决定是否需要进行验证操作。* 如果需要验证,则会调用 #attemptAuthentica

Spring Security 从入门到进阶系列教程

Spring Security 入门系列 《保护 Web 应用的安全》 《Spring-Security-入门(一):登录与退出》 《Spring-Security-入门(二):基于数据库验证》 《Spring-Security-入门(三):密码加密》 《Spring-Security-入门(四):自定义-Filter》 《Spring-Security-入门(五):在 Sprin

Java架构师知识体认识

源码分析 常用设计模式 Proxy代理模式Factory工厂模式Singleton单例模式Delegate委派模式Strategy策略模式Prototype原型模式Template模板模式 Spring5 beans 接口实例化代理Bean操作 Context Ioc容器设计原理及高级特性Aop设计原理Factorybean与Beanfactory Transaction 声明式事物

NameNode内存生产配置

Hadoop2.x 系列,配置 NameNode 内存 NameNode 内存默认 2000m ,如果服务器内存 4G , NameNode 内存可以配置 3g 。在 hadoop-env.sh 文件中配置如下。 HADOOP_NAMENODE_OPTS=-Xmx3072m Hadoop3.x 系列,配置 Nam

Java进阶13讲__第12讲_1/2

多线程、线程池 1.  线程概念 1.1  什么是线程 1.2  线程的好处 2.   创建线程的三种方式 注意事项 2.1  继承Thread类 2.1.1 认识  2.1.2  编码实现  package cn.hdc.oop10.Thread;import org.slf4j.Logger;import org.slf4j.LoggerFactory

JAVA智听未来一站式有声阅读平台听书系统小程序源码

智听未来,一站式有声阅读平台听书系统 🌟&nbsp;开篇:遇见未来,从“智听”开始 在这个快节奏的时代,你是否渴望在忙碌的间隙,找到一片属于自己的宁静角落?是否梦想着能随时随地,沉浸在知识的海洋,或是故事的奇幻世界里?今天,就让我带你一起探索“智听未来”——这一站式有声阅读平台听书系统,它正悄悄改变着我们的阅读方式,让未来触手可及! 📚&nbsp;第一站:海量资源,应有尽有 走进“智听