1

JDOM を使用して XML ファイルを生成する必要がありますが、これはかなり大きくなる可能性があります。すでにメモリ内にあるデータ、主に文字列以外に、JDOMがどれだけの追加メモリスペースを必要とするのか疑問に思っています。テスト用の簡単なプログラムを作成したところ、オーバーヘッドは XML コンテンツの約 2 倍であることがわかりました。

JDOM が追加のメモリを必要とする理由と、それを最適化できる方法があるかどうかを知っている人はいますか? JDOM オブジェクトは、既存の文字列への参照を保持するだけでよいのではないでしょうか?

テストに使用したプログラムは次のとおりです。

public class TestJdomMemoryOverhead {
    private static Runtime runtime = Runtime.getRuntime();

    public static void gc() {
        // Try to give the JVM some hints to run garbage collection
        for (int i = 0; i < 5; i++) {
            runtime.runFinalization();
            runtime.gc();
            Thread.currentThread().yield();
        }
    }

    public static void generateXml(List<String> filenames) throws IOException {
        // generate a simple XML file by these file names. It looks like:
        // <?xml version="1.0" encoding="UTF-8"?>
        // <files>
        // <f n="aaaaaaaaaaaaaaaaaaaaaaaaaaaaaa" />
        // <f n="aaaaaaaaaaaaaaaaaaaaaaaaaaaaaa" />
        // <f n="aaaaaaaaaaaaaaaaaaaaaaaaaaaaaa" />
        // ....
        // ....
        // </files>
        Element filesElem = new Element("files");
        Document doc = new Document(filesElem);
        for (String name : filenames) {
            Element fileElem = new Element("f");
            fileElem.setAttribute("n", name);
            filesElem.addContent(fileElem);
        }
        gc();
        System.out.println("After generating JDOM objects: " + (runtime.totalMemory() - runtime.freeMemory()) + " bytes");
        XMLOutputter outputter = new XMLOutputter(Format.getPrettyFormat());
        BufferedWriter writer = new BufferedWriter(new FileWriter("test.xml", false));
        outputter.output(doc, writer);
        writer.close();
        gc();
        System.out.println("After writing to XML file: " + (runtime.totalMemory() - runtime.freeMemory()) + " bytes");
    }

    public static void main(String[] cmdArgs) throws IOException {
        List<String> filenames = new ArrayList<String>();
        StringBuilder builder = new StringBuilder();
        // 30 unicode chracters, repated 500,000 times. The memory to store
        // these file name strings should be about 30MB.
        builder.append("aaaaaaaaaaaaaaaaaaaaaaaaaaaaaa");
        for (int i = 0; i < 500000; i++) {
            filenames.add(builder.toString());
        }
        gc();
        System.out.println("After generating file names: " + (runtime.totalMemory() - runtime.freeMemory()) + " bytes");
        generateXml(filenames);
        gc();
        System.out.println("Get back to main: " + (runtime.totalMemory() - runtime.freeMemory()) + " bytes");
    }
}

出力は次のとおりです。

After generating file names: 51941096 bytes
After generating JDOM objects: 125766824 bytes
After writing to XML file: 126036768 bytes
Get back to main: 51087440 bytes

ご覧のとおり、JDOM オブジェクトは約 70MB を使用しました。

4

1 に答える 1

1

JDOM が大量のメモリを必要とする理由は、JDOM が主に DOM のようなツリー ベースの API であるためです (ドキュメント ツリーは、使用した方法でメモリ内に作成されます)。しかし、DOM よりもパフォーマンスが優れています。大きな XML ドキュメントを作成している場合は、jdk6 にバンドルされているXMLStreamWriterなどの使用を検討することをお勧めします。

JDOM でできないことについての短い記事を次に示します。

于 2010-08-02T04:22:04.963 に答える