12

最新の JDK にアップグレードした後、(一部のマシンで) 奇妙なOutOfMemoryException.

この単純なアプリケーションを考えてみましょう:

public class Test
{
    public static void main (String[] args) {
        try {
            java.text.SimpleDateFormat dateFormatter = new java.text.SimpleDateFormat("E dd/MM/yyyy HH:mm");
            System.out.println("formatted date: " + dateFormatter.format(new java.util.Date()));
        } catch (Exception x) {
            System.err.println(x);
            System.exit(1);
        }
    }
}

この小さなプログラムを実行すると、次の例外が発生します ( で実行した場合でも-Xmx2048M -Xms2048):

Exception in thread "main" java.lang.OutOfMemoryError: Java heap space
    at java.util.Currency.readLongArray(Currency.java:657)
    at java.util.Currency.access$100(Currency.java:76)
    at java.util.Currency$1.run(Currency.java:211)
    at java.security.AccessController.doPrivileged(Native Method)
    at java.util.Currency.<clinit>(Currency.java:192)
    at java.text.DecimalFormatSymbols.initialize(DecimalFormatSymbols.java:566)
    at java.text.DecimalFormatSymbols.<init>(DecimalFormatSymbols.java:94)
    at java.text.DecimalFormatSymbols.getInstance(DecimalFormatSymbols.java:157)
    at java.text.NumberFormat.getInstance(NumberFormat.java:767)
    at java.text.NumberFormat.getIntegerInstance(NumberFormat.java:439)
    at java.text.SimpleDateFormat.initialize(SimpleDateFormat.java:664)
    at java.text.SimpleDateFormat.<init>(SimpleDateFormat.java:585)
    at java.text.SimpleDateFormat.<init>(SimpleDateFormat.java:560)
    at Test.main(Test.java:5)

誰かが私にこれを説明してもらえますか?

使用している Java バージョンは次のとおりです。

java version "1.7.0_45"
Java(TM) SE Runtime Environment (build 1.7.0_45-b18)
Java HotSpot(TM) 64-Bit Server VM (build 24.45-b08, mixed mode)

以前のバージョンを使用している場合、問題はありません。

Java -バージョン

java version "1.7.0_25"
Java(TM) SE Runtime Environment (build 1.7.0_25-b16)
Java HotSpot(TM) 64-Bit Server VM (build 23.25-b01, mixed mode)

Javaテスト

formatted date: ma 21/10/2013 10:19

更新:誰もがヒープサイズの増加について言及する前に...私はすでにそれを試しました:

java -Xmx2048M -Xms2048M テスト

Runtime.getRuntime().freeMemory(): 2048120480
Exception in thread "main" java.lang.OutOfMemoryError: Java heap space
    at java.util.Currency.readLongArray(Currency.java:657)
    at java.util.Currency.access$100(Currency.java:76)
    at java.util.Currency$1.run(Currency.java:211)
    at java.security.AccessController.doPrivileged(Native Method)
    at java.util.Currency.<clinit>(Currency.java:192)
    at java.text.DecimalFormatSymbols.initialize(DecimalFormatSymbols.java:566)
    at java.text.DecimalFormatSymbols.<init>(DecimalFormatSymbols.java:94)
    at java.text.DecimalFormatSymbols.getInstance(DecimalFormatSymbols.java:157)
    at java.text.NumberFormat.getInstance(NumberFormat.java:767)
    at java.text.NumberFormat.getIntegerInstance(NumberFormat.java:439)
    at java.text.SimpleDateFormat.initialize(SimpleDateFormat.java:664)
    at java.text.SimpleDateFormat.<init>(SimpleDateFormat.java:585)
    at java.text.SimpleDateFormat.<init>(SimpleDateFormat.java:560)
    at Test.main(Test.java:8)
4

1 に答える 1

10

のコードを掘り下げましたjava.util.Currency。Java JRE の静的クラス初期化子ブロックにあるリソース ファイルを読み込みます。

String homeDir = System.getProperty("java.home");
try {
    String dataFile = homeDir + File.separator +
            "lib" + File.separator + "currency.data";
    DataInputStream dis = new DataInputStream(
        new BufferedInputStream(
        new FileInputStream(dataFile)));
    if (dis.readInt() != MAGIC_NUMBER) {
        throw new InternalError("Currency data is possibly corrupted");
    }
    formatVersion = dis.readInt();
    if (formatVersion != VALID_FORMAT_VERSION) {
        throw new InternalError("Currency data format is incorrect");
    }
    dataVersion = dis.readInt();
    mainTable = readIntArray(dis, A_TO_Z * A_TO_Z);
    int scCount = dis.readInt();
    scCutOverTimes = readLongArray(dis, scCount);

ご覧のとおり、JRE/lib/currency.dataを読み取ります。4 番目の整数にはscCount. この整数は高すぎます。そのファイルが壊れていると思います。置き換えてみてください。

于 2013-10-21T08:52:14.977 に答える