java中模拟c中对sizeof的实现思路:利用java中GC内存回收前后的heap size差别,得出每个object的大小。
这是一个程序,java中没有现成的sizeof的实现,原因主要是java中的基本数据类型的大小都是固定的,所以看上去没有必要用sizeof这个关键字。实现的想法是这样的:java.lang.Runtime类中有一些简单的能涉及到内存管理的函数:Every Java application has a single instance of class Runtime that allows the application to interface with the environment in which the application is running. The current runtime can be obtained from the getRuntime method. long freeMemory()
Returns the amount of free memory in the Java Virtual Machine.
void gc()
Runs the garbage collector.
static Runtime getRuntime()
Returns the runtime object associated with the current Java application.
long maxMemory()
Returns the maximum amount of memory that the Java virtual machine will attempt to use.
void runFinalization()
Runs the finalization methods of any objects pending finalization.
使用这些简单的内存访问,可以得到内存的一些情况,我们通过建立一个大的某个类的数组,来查看内存用了多少,进而可以求得类的大小。程序原码:/* * 创建日期 2007-4-19*/ public class Sizeof { public static void main(String[] args) throws Exception { // Warm up all classes/methods we will use runGC(); usedMemory(); // Array to keep strong references to allocated objects final int count = 100000; Object[] objects = new Object[count]; long heap1 = 0; // Allocate count+1 objects, discard the first one for (int i = -1; i < count; ++i) { Object object = null; /* Instantiate your data here and assign it to object*/ //object = new Object (); //object = new Integer (i); // ???? //object = new Long (i); //???? //object = new String (); //object = new Byte( (byte) 0); //???? //object = new Float( 0.2f); //???? //object = new Double( 0); //???? //object = new HelloWorld(); if (i >= 0) objects[i] = object; else { object = null; // Discard the warm up object runGC(); heap1 = usedMemory(); // Take a before heap snapshot } } runGC(); long heap2 = usedMemory(); // Take an after heap snapshot: final int size = (int) Math.round(((double) (heap2 - heap1)) / count); System.out.println("'before' heap: " + heap1 + ", 'after' heap: " + heap2); System.out .println("heap delta: " + (heap2 - heap1) + ", {" + objects[0].getClass() + "} size = " + size + " bytes"); for (int i = 0; i < count; ++i) objects[i] = null; objects = null; } private static void runGC() throws Exception { // It helps to call Runtime.gc() // using several method calls: for (int r = 0; r < 4; ++r) _runGC(); } private static void _runGC() throws Exception { long usedMem1 = usedMemory(), usedMem2 = Long.MAX_VALUE; for (int i = 0; (usedMem1 < usedMem2) && (i < 500); ++i) { s_runtime.runFinalization(); s_runtime.gc(); Thread.yield(); usedMem2 = usedMem1; usedMem1 = usedMemory(); } } private static long usedMemory() { return s_runtime.totalMemory() - s_runtime.freeMemory(); } private static final Runtime s_runtime = Runtime.getRuntime(); } // End of class 需要注意的是上面的一句: Object[] objects = new Object[count];只是分配了数组空间,没有分配对象的空间。数组中只有引用而已。 结论:下代码测试基本对象时,得出的结果象下面:
Object类对象是 8字节
Integer类对象是 16字节
Long类对象是 16字节
Byte类对象是 16字节
怎么会这样呢???不解。 查看网上的帖子:解释如下: 这个例子写的很好,正好说明了java中基本类型封装对象所占内存的大小.
1.简单的Object对象要占用8个字节的内存空间,因为每个实例都至少必须包含一些最基本操作,比如:wait()/notify(),equals(), hashCode()等
2.使用Integer对象占用了16个字节,而int占用4个字节,说了封装了之后内存消耗大了4倍
3.那么Long看起来比Integer对象应该使用更多空间,结果Long所占的空间也是16个字节.
那么就正好说明了JVM的对于基本类型封装对象的内存分配的规则是如下:
Object所占内存(8个字节)+最大基本类型(long)所占内存(8个字节) = 16字节.
JVM强制使用8个字节作为边界.
所以所有基本类型封装对象所占内存的大小都是16字节.但是还是有区别,比如:Integer对象虽然占用了16个字节的内存,但是只是利用了 Object所占内存(8个字节)+int所占内存(4个字节) = 12字节.还有4个字节根本没有被使用.呵呵,仔细分析了一晚,还是有很多收获的
分享到:
相关推荐
3. `java-sizeof-0.0.4-sources.jar`:包含了源代码,开发者可以查看源码来理解工具的工作原理,学习如何实现类似的功能,或者在遇到问题时进行调试。 使用"java-sizeof-0.0.4",开发者可以进行以下操作: - 分析...
本篇文章将探讨如何通过Java来实现这样的功能,以及相关知识点。 首先,Java虚拟机(JVM)在运行时为每个对象分配内存,包括实例变量、类信息、对其他对象的引用等。但是,Java API本身并不提供直接获取对象大小的...
releaseNotes.txt文件记录了SizeOf.jar的版本更新历史,包括新功能、改进和已知问题的修复。通过查看这个文件,用户可以了解每个版本的变更,选择最适合他们需求的版本。 总的来说,SizeOf.jar是一个实用的开发工具...
`SizeOf`工具可能通过直接访问JVM的内部信息来实现这些计算,这可能涉及到使用反射(Reflection)或者特定于JVM的API,如Sun的`sun.misc.Unsafe`类,虽然这并不是标准的Java API,但在某些情况下能提供更底层的内存...
通过这些源代码,开发者可以在两个不同的编程环境中实现相同的安全功能,从而保证数据在传输或存储过程中的安全性。 总的来说,理解并正确使用DES加解密在Java和Delphi中的实现,对于跨平台开发和数据安全具有重要...
在C、C++和JAVA这三种编程语言中,链表的实现方式各有特点,下面将详细介绍它们。 首先,让我们从C语言开始。C语言中的链表实现主要依赖于结构体和指针。一个基本的链表节点结构体通常包含两个部分:数据存储区域和...
此外,Java虚拟机(JVM)是实现跨平台的关键,Java支持多线程编程,提供了异常处理机制来管理程序运行时的错误,且Java具有自动内存管理功能,如垃圾回收(Garbage Collection),使得程序员无需手动管理内存。...
在本主题中,我们将深入探讨如何在Android应用中实现C语言调用Java方法。 首先,我们需要理解JNI的基本结构。在Android项目中,JNI相关的代码通常被放在`src/main/jni`目录下,这里的源代码文件通常以`.c`或`.cpp`...
由于Java本身没有提供直接修改环境变量的API,因此需要借助JNI来实现这一功能。下面将详细介绍具体的步骤: ### 步骤一:创建Java类 首先,创建一个Java类`ChangeEnv.java`,用于定义一个本地接口,该接口将被用于...
`sizeof`不是Java的关键字,而是C/C++中的一个运算符,Java中类似的功能是`SizeOf`类或者`Arrays#size()`方法(D)。 7. `super()`在Java中用于调用父类的构造函数。在子类的构造方法中,`super()`语句用于调用父类...
- Java中没有`sizeof`操作符来获取变量或数据类型的大小,因为这些信息在编译时就已经确定,并不需要在运行时动态获取。 - 变量的声明和初始化需要遵循一定的规则,例如静态变量(`static`)在类加载时就会被初始...
- **解析**:`sizeof`不是Java语言中的关键字。 #### 题目16: 类的构造与使用 - **知识点**:Java类的构造函数和方法调用 - **解析**:题目中的代码示例未完整给出,但从已给出的部分可以看出,`Parser`类继承自`...
Java是由语法规则和类库两部分构成的,语法规则定义了编写程序的规则,而类库则提供了丰富的预定义类,使得开发者能够方便地实现各种功能,如网络连接和数据库操作。Java与C/C++有密切联系,很多语法结构与C++类似,...
知识点:**3.5 Java语言流程控制语判语句的功能及使用** `while(true)`和`while(!false)`都会导致无限循环。`while(true)`条件永远为真,而`!false`等于`true`,因此这两个条件都不会结束循环。 ### 7. 会产生编译...
+ STAT_SIZE_PER_FRAM * sizeof(_UL); ``` 2. 注释 良好的注释能帮助其他开发者快速理解代码功能和逻辑,规范要求: - 注释要简洁明了,描述代码的主要作用,避免过于冗余。 - 函数、方法、类上方应有简要说明其...
4. **与C/C++的区别**:Java不支持面向过程编程、全程变量、goto语句、宏定义、全局函数、结构、联合和指针等,而是通过引用模型实现类似指针的功能,通过类实现结构类型。 5. **继承与接口**:Java不支持多重继承...
18. **多线程**:Java实现多线程有两种方式,一是继承`Thread`类,二是实现`Runnable`接口,并将其实例传递给`Thread`对象。 19. **布局管理器**:Java GUI中常见的布局管理器有FlowLayout(流式布局)、GridLayout...