代码之家  ›  专栏  ›  技术社区  ›  uckelman

如何让psoldgen扩展以容纳非常大的对象?

  •  3
  • uckelman  · 技术社区  · 14 年前

    我有一个程序需要为一个非常大的对象分配堆空间,但我发现 OutOfMemoryException 当有比大型对象所需的更多的空闲堆时。

    这个测试程序演示了问题:

    public class HeapTest {
      public static void main(String[] args) {
        final int size = Integer.parseInt(args[0]);
        System.out.println(size >> 20);
        final byte[] buf = new byte[size];
        System.out.println(buf.length);
      }
    }
    

    如果我运行它 java -Xmx40M -XX:+PrintGCDetails HeapTest 28000000 ,我得到以下输出:

    26
    [GC [PSYoungGen: 317K->176K(9216K)] 317K->176K(36544K), 0.0007430 secs] [Times: user=0.01 sys=0.00, real=0.00 secs] 
    [GC [PSYoungGen: 176K->144K(9216K)] 176K->144K(36544K), 0.0004650 secs] [Times: user=0.00 sys=0.00, real=0.00 secs] 
    [Full GC [PSYoungGen: 144K->0K(9216K)] [PSOldGen: 0K->115K(9216K)] 144K->115K(18432K) [PSPermGen: 2710K->2710K(21248K)], 0.0053960 secs] [Times: user=0.00 sys=0.00, real=0.01 secs] 
    [GC [PSYoungGen: 0K->0K(9216K)] 115K->115K(36544K), 0.0002010 secs] [Times: user=0.00 sys=0.00, real=0.00 secs] 
    [Full GC [PSYoungGen: 0K->0K(9216K)] [PSOldGen: 115K->112K(15104K)] 115K->112K(24320K) [PSPermGen: 2710K->2708K(21248K)], 0.0078150 secs] [Times: user=0.01 sys=0.00, real=0.00 secs] 
    Exception in thread "main" java.lang.OutOfMemoryError: Java heap space
        at HeapTest.main(HeapTest.java:5)
    Heap
     PSYoungGen      total 9216K, used 476K [0x00000000ff2b0000, 0x0000000100000000, 0x0000000100000000)
      eden space 7936K, 6% used [0x00000000ff2b0000,0x00000000ff327190,0x00000000ffa70000)
      from space 1280K, 0% used [0x00000000ffa70000,0x00000000ffa70000,0x00000000ffbb0000)
      to   space 1280K, 0% used [0x00000000ffec0000,0x00000000ffec0000,0x0000000100000000)
     PSOldGen        total 27328K, used 112K [0x00000000fd800000, 0x00000000ff2b0000, 0x00000000ff2b0000)
      object space 27328K, 0% used [0x00000000fd800000,0x00000000fd81c378,0x00000000ff2b0000)
     PSPermGen       total 21248K, used 2815K [0x00000000f3000000, 0x00000000f44c0000, 0x00000000fd800000)
      object space 21248K, 13% used [0x00000000f3000000,0x00000000f32bff48,0x00000000f44c0000)
    

    我试图创建的数组应该是27344K左右,略大于psoldgen对象空间的大小。但是,总的未使用空间约为54000K,几乎是阵列所需空间的两倍。由于我的(real)程序在数组上工作到位,它几乎不使用任何超出数组所需内存的内存,因此分配两倍我需要的内存,只使用其中的一半似乎是浪费。

    有没有一种方法可以说服JVM让老一代的增长超过它在默认情况下的能力?

    1 回复  |  直到 9 年前
        1
  •  3
  •   Neeme Praks Tim Pote    14 年前

    NewRatio NewSize MaxNewSize SurvivorRatio

    另外,如果你担心 永久性一代 使用太多的内存,您可以用 MaxPermSize 命令行选项。

    Tuning Garbage Collection with the 5.0 Java[tm] Virtual Machine 更多细节,尤其是关于 generations .

    例如,设置最大值 年轻一代 尺寸至 10M 允许示例程序成功运行:

    java -Xmx40M -XX:+PrintGCDetails -XX:MaxNewSize=10M HeapTest 28000000
    

    在我的计算机上生成以下输出:

    26
    28000000
    Heap
     par new generation   total 9216K, used 1499K [106810000, 107210000, 107210000)
      eden space 8192K,  18% used [106810000, 106986ed0, 107010000)
      from space 1024K,   0% used [107010000, 107010000, 107110000)
      to   space 1024K,   0% used [107110000, 107110000, 107210000)
     concurrent mark-sweep generation total 30720K, used 27343K [107210000, 109010000, 109010000)
     concurrent-mark-sweep perm gen total 21248K, used 4501K [109010000, 10a4d0000, 10e410000)