使用仪器来确定对象的大小

时间:2016-09-14 20:03:17

标签: java

我使用了文章http://www.devinline.com/2016/05/java-instrumentation-fundamental-part-1.html?m=1

我需要获得查询结果的大小。

但是要求

long sizeOfObject = InstrumentationAgent.findSizeOfObject(myvar);

返回错误

  

代理未启动。

我有抛出Exception类的主要方法 你能给出正确语法的建议吗?

修订: 代理代码:

package org.h2.command;
import java.lang.instrument.Instrumentation;
import java.lang.instrument.UnmodifiableClassException;

public class InstrumentationAgent {
    /*
     * System classloader after loading the Agent Class, invokes the premain
     * (premain is roughly equal to main method for normal Java classes)
     */
    private static volatile Instrumentation instrumentation;

    public static void premain(String agentArgs, Instrumentation instObj) {
        // instObj is handle passed by JVM
        instrumentation = instObj;
    }

    public static void agentmain(String agentArgs, Instrumentation instObj)
        throws ClassNotFoundException, UnmodifiableClassException {
    }

    public static long findSizeOfObject(Object obj) {
        // use instrumentation to find size of object obj
        if (instrumentation == null) {
            throw new IllegalStateException("Agent not initted");
        } else {
            return instrumentation.getObjectSize(obj);
        }
    }
}

我的调用:

import java.io.BufferedWriter;
import java.io.FileWriter;
import java.io.IOException;
import java.lang.instrument.Instrumentation;
import org.h2.command.InstrumentationAgent;
import static java.lang.System.out;  

public class CacheOptimize {
    public long Size;

    public static void main(String[] args) throws Exception {
        Class.forName("org.h2.Driver");
        Connection conn = DriverManager.getConnection("jdbc:h2:file:D:/server/h2/exp1.h2.db", "sa", "sa");
        Statement stat = conn.createStatement();

        ResultSet rs;
        rs = stat.executeQuery("select * from TAbles");
        Size = InstrumentationAgent.findSizeOfObject(rs);   
    }
    stat.close();
    conn.close();
}

1 个答案:

答案 0 :(得分:1)

您忘记添加META-INF/MANIFEST.MF条目

Premain-Class: org.h2.command.InstrumentationAgent

或在没有-javaagent:path/to/agent.jar的情况下运行您的应用程序。

Here您可以找到有关如何使用代理启动应用程序的完整工作示例。

您还可以在official javadoc中找到有关清单条目和正在运行的代理的更多信息。

注意

您似乎正在尝试获取ResultSet返回的数据大小,而不是ResultSet对象本身消耗的内存量。问题是

size = InstrumentationAgent.findSizeOfObject(rs);

将不是最佳方法,因为ResultSet仅将游标维护到数据库行,并且不存储所有结果。但是,您可以使用它获取所有数据,并使用findSizeOfObject汇总大小。但你应该知道的最后一件事是Instrumentation#getObjectSize可能会返回不准确的结果

  

返回指定对象占用的存储量的特定于实现的近似值。结果可能包括对象的一些或全部开销,因此对于实现中的比较而不是实现之间的比较是有用的。在单次调用JVM期间,估计值可能会发生变化。