2012-02-07 4 views
8

¿Hay alguna manera elegante de activar automáticamente las advertencias de memoria en mi aplicación Java cuando la memoria libre alcanza un cierto umbral?¿Cuál es una excelente manera de advertir automáticamente a mi aplicación Java cuando la memoria está baja?

Ten en cuenta que esta es una pregunta al estilo Jeopardy a la que ya tengo una respuesta, solo quería publicarla aquí para que el mundo la descubra porque la solución me ayudó un montón.

+0

Vea también otras preguntas/respuestas que mencionan el 'MemoryMXBean', como http://stackoverflow.com/ questions/433406/how-to-get-the-max-sizes-of-the-heap-and-permgen-from-the-jvm – DNA

Respuesta

8

Aquí hay una gran pequeña clase escrita por Heinz Kabutz que funciona perfectamente para mí "fuera de la caja". Encontrado en una cuestión de edad "especialistas en Java": http://www.javaspecialists.eu/archive/Issue092.html

import java.lang.management.ManagementFactory; 
import java.lang.management.MemoryMXBean; 
import java.lang.management.MemoryNotificationInfo; 
import java.lang.management.MemoryPoolMXBean; 
import java.lang.management.MemoryType; 
import java.util.ArrayList; 
import java.util.Collection; 

import javax.management.Notification; 
import javax.management.NotificationEmitter; 
import javax.management.NotificationListener; 

/** 
* This memory warning system will call the listener when we exceed the 
* percentage of available memory specified. There should only be one instance 
* of this object created, since the usage threshold can only be set to one 
* number. 
* 
* (adapted from http://www.javaspecialists.eu/archive/Issue092.html) 
*/ 

public class MemoryWarningSystem { 

    public interface Listener { 

     void memoryUsageLow(long usedMemory, long maxMemory); 
    } 

    private final Collection<Listener> listeners = new ArrayList<Listener>(); 

    private static final MemoryPoolMXBean tenuredGenPool = findTenuredGenPool(); 

    public MemoryWarningSystem() { 
     MemoryMXBean mbean = ManagementFactory.getMemoryMXBean(); 
     NotificationEmitter emitter = (NotificationEmitter) mbean; 
     emitter.addNotificationListener(new NotificationListener() { 
      @Override 
      public void handleNotification(Notification n, Object hb) { 
       if (n.getType().equals(
         MemoryNotificationInfo.MEMORY_THRESHOLD_EXCEEDED)) { 
        long maxMemory = tenuredGenPool.getUsage().getMax(); 
        long usedMemory = tenuredGenPool.getUsage().getUsed(); 
        for (Listener listener : listeners) { 
         listener.memoryUsageLow(usedMemory, maxMemory); 
        } 
       } 
      } 
     }, null, null); 
    } 

    public boolean addListener(Listener listener) { 
     return listeners.add(listener); 
    } 

    public boolean removeListener(Listener listener) { 
     return listeners.remove(listener); 
    } 

    public void setPercentageUsageThreshold(double percentage) { 
     if (percentage <= 0.0 || percentage > 1.0) { 
      throw new IllegalArgumentException("Percentage not in range"); 
     } 
     long maxMemory = tenuredGenPool.getUsage().getMax(); 
     long warningThreshold = (long) (maxMemory * percentage); 
     tenuredGenPool.setUsageThreshold(warningThreshold); 
    } 

    /** 
    * Tenured Space Pool can be determined by it being of type HEAP and by it 
    * being possible to set the usage threshold. 
    */ 
    private static MemoryPoolMXBean findTenuredGenPool() { 
     for (MemoryPoolMXBean pool : ManagementFactory.getMemoryPoolMXBeans()) { 
      // I don't know whether this approach is better, or whether 
      // we should rather check for the pool name "Tenured Gen"? 
      if (pool.getType() == MemoryType.HEAP 
        && pool.isUsageThresholdSupported()) { 
       return pool; 
      } 
     } 
     throw new IllegalStateException("Could not find tenured space"); 
    } 
} 

de uso:

MemoryWarningSystem system = new MemoryWarningSystem(); 
    system.setPercentageUsageThreshold(0.8d); 
    system.addListener(new Listener() { 
     @Override 
     public void memoryUsageLow(long usedMemory, long maxMemory) { 
      System.out.println("low: "+usedMemory+"/"+maxMemory); 
     } 
    }); 
+1

Esto se disparará incluso si hubiera mucho espacio libre después de un GC. Sería más útil tener un hilo de fondo solo verificar inmediatamente después de un GC. –

+1

Otro problema que esto puede tener es que el espacio fijo puede cambiar de tamaño. Esto significa que el warningThreshold puede volverse incorrecto con el tiempo. Si su espacio fijo está muy fragmentado, puede parecer que tiene mucho espacio libre que no se puede usar, es decir, puede obtener un OOME cuando parezca que solo está lleno al 50%. ;) –

Cuestiones relacionadas