Saltar al contenido

Uso de memoria virtual de Java en Linux, demasiada memoria utilizada

Solución:

Esta ha sido una queja de larga data con Java, pero en gran medida no tiene sentido y, por lo general, se basa en buscar información incorrecta. La fraseología habitual es algo así como “¡Hola mundo en Java ocupa 10 megabytes! ¿Por qué lo necesita?” Bueno, aquí hay una manera de hacer que Hello World en una JVM de 64 bits afirme que ocupa más de 4 gigabytes … al menos por una forma de medición.

java -Xms1024m -Xmx4096m com.example.Hello

Diferentes formas de medir la memoria

En Linux, el comando top le da varios números diferentes de memoria. Esto es lo que dice sobre el ejemplo de Hello World:

  PID USER      PR  NI  VIRT  RES  SHR S %CPU %MEM    TIME+  COMMAND
 2120 kgregory  20   0 4373m  15m 7152 S    0  0.2   0:00.10 java
  • VIRT es el espacio de memoria virtual: la suma de todo en el mapa de memoria virtual (ver más abajo). En gran medida no tiene sentido, excepto cuando no lo es (ver más abajo).
  • RES es el tamaño del conjunto residente: el número de páginas que residen actualmente en la RAM. En casi todos los casos, este es el único número que debe usar cuando diga “demasiado grande”. Pero todavía no es un número muy bueno, especialmente cuando se habla de Java.
  • SHR es la cantidad de memoria residente que se comparte con otros procesos. Para un proceso de Java, esto normalmente se limita a bibliotecas compartidas y archivos JAR asignados en memoria. En este ejemplo, solo tenía un proceso de Java en ejecución, por lo que sospecho que el 7k es el resultado de las bibliotecas utilizadas por el sistema operativo.
  • SWAP no está activado de forma predeterminada y no se muestra aquí. Indica la cantidad de memoria virtual que reside actualmente en el disco, si está o no en el espacio de intercambio. El sistema operativo es muy bueno para mantener páginas activas en la RAM, y las únicas soluciones para el intercambio son (1) comprar más memoria o (2) reducir la cantidad de procesos, por lo que es mejor ignorar este número.

La situación del Administrador de tareas de Windows es un poco más complicada. En Windows XP, hay columnas de “Uso de memoria” y “Tamaño de memoria virtual”, pero la documentación oficial no dice nada sobre lo que significan. Windows Vista y Windows 7 agregan más columnas y, de hecho, están documentadas. De estos, la medición del “Conjunto de trabajo” es la más útil; corresponde aproximadamente a la suma de RES y SHR en Linux.

Comprensión del mapa de memoria virtual

La memoria virtual consumida por un proceso es el total de todo lo que está en el mapa de memoria del proceso. Esto incluye datos (por ejemplo, el montón de Java), pero también todas las bibliotecas compartidas y archivos mapeados en memoria utilizados por el programa. En Linux, puede usar el comando pmap para ver todas las cosas mapeadas en el espacio de proceso (de aquí en adelante solo me referiré a Linux, porque es lo que uso; estoy seguro de que hay herramientas equivalentes para Windows). Aquí hay un extracto del mapa de memoria del programa “Hello World”; el mapa de memoria completo tiene más de 100 líneas y no es inusual tener una lista de mil líneas.

0000000040000000     36K r-x--  /usr/local/java/jdk-1.6-x64/bin/java
0000000040108000      8K rwx--  /usr/local/java/jdk-1.6-x64/bin/java
0000000040eba000    676K rwx--    [ anon ]
00000006fae00000  21248K rwx--    [ anon ]
00000006fc2c0000  62720K rwx--    [ anon ]
0000000700000000 699072K rwx--    [ anon ]
000000072aab0000 2097152K rwx--    [ anon ]
00000007aaab0000 349504K rwx--    [ anon ]
00000007c0000000 1048576K rwx--    [ anon ]
...
00007fa1ed00d000   1652K r-xs-  /usr/local/java/jdk-1.6-x64/jre/lib/rt.jar
...
00007fa1ed1d3000   1024K rwx--    [ anon ]
00007fa1ed2d3000      4K -----    [ anon ]
00007fa1ed2d4000   1024K rwx--    [ anon ]
00007fa1ed3d4000      4K -----    [ anon ]
...
00007fa1f20d3000    164K r-x--  /usr/local/java/jdk-1.6-x64/jre/lib/amd64/libjava.so
00007fa1f20fc000   1020K -----  /usr/local/java/jdk-1.6-x64/jre/lib/amd64/libjava.so
00007fa1f21fb000     28K rwx--  /usr/local/java/jdk-1.6-x64/jre/lib/amd64/libjava.so
...
00007fa1f34aa000   1576K r-x--  /lib/x86_64-linux-gnu/libc-2.13.so
00007fa1f3634000   2044K -----  /lib/x86_64-linux-gnu/libc-2.13.so
00007fa1f3833000     16K r-x--  /lib/x86_64-linux-gnu/libc-2.13.so
00007fa1f3837000      4K rwx--  /lib/x86_64-linux-gnu/libc-2.13.so
...

Una explicación rápida del formato: cada fila comienza con la dirección de memoria virtual del segmento. A esto le sigue el tamaño del segmento, los permisos y la fuente del segmento. Este último elemento es un archivo o “anon”, lo que indica un bloque de memoria asignado a través de mmap.

Empezando desde arriba, tenemos

  • El cargador de JVM (es decir, el programa que se ejecuta cuando escribe java). Esto es muy pequeño; todo lo que hace es cargar en las bibliotecas compartidas donde se almacena el código JVM real.
  • Un montón de bloques anónimos que contienen el montón de Java y los datos internos. Esta es una JVM de Sun, por lo que el montón se divide en varias generaciones, cada una de las cuales es su propio bloque de memoria. Tenga en cuenta que la JVM asigna espacio de memoria virtual en función de la -Xmx valor; esto le permite tener un montón contiguo. los -Xms value se usa internamente para decir cuánto del montón está “en uso” cuando se inicia el programa, y ​​para activar la recolección de basura cuando se acerca ese límite.
  • Un archivo JAR mapeado en memoria, en este caso el archivo que contiene las “clases JDK”. Cuando asigna un mapa de memoria a un JAR, puede acceder a los archivos que contiene de manera muy eficiente (en lugar de leerlo desde el principio cada vez). Sun JVM mapeará en memoria todos los JAR en la ruta de clases; si su código de aplicación necesita acceder a un JAR, también puede asignarlo en memoria.
  • Datos por subproceso para dos subprocesos. El bloque 1M es la pila de subprocesos. No tenía una buena explicación para el bloque 4k, pero @ericsoe lo identificó como un “bloque de protección”: no tiene permisos de lectura / escritura, por lo que causará una falla de segmento si se accede, y la JVM lo detecta y traduce a un StackOverFlowError. Para una aplicación real, verá docenas, si no cientos, de estas entradas repetidas a través del mapa de memoria.
  • Una de las bibliotecas compartidas que contiene el código JVM real. Hay varios de estos.
  • La biblioteca compartida para la biblioteca estándar de C. Esta es solo una de las muchas cosas que carga la JVM que no son estrictamente parte de Java.

Las bibliotecas compartidas son particularmente interesantes: cada biblioteca compartida tiene al menos dos segmentos: un segmento de solo lectura que contiene el código de la biblioteca y un segmento de lectura y escritura que contiene datos globales por proceso para la biblioteca (no sé qué segmento sin permisos es; solo lo he visto en x64 Linux). La parte de solo lectura de la biblioteca se puede compartir entre todos los procesos que utilizan la biblioteca; por ejemplo, libc tiene 1,5 M de espacio de memoria virtual que se puede compartir.

¿Cuándo es importante el tamaño de la memoria virtual?

El mapa de memoria virtual contiene muchas cosas. Parte es de solo lectura, parte se comparte y parte se asigna pero nunca se toca (por ejemplo, casi todos los 4Gb de montón en este ejemplo). Pero el sistema operativo es lo suficientemente inteligente como para cargar solo lo que necesita, por lo que el tamaño de la memoria virtual es en gran medida irrelevante.

Donde el tamaño de la memoria virtual es importante es si está ejecutando un sistema operativo de 32 bits, donde solo puede asignar 2 Gb (o, en algunos casos, 3 Gb) de espacio de direcciones de proceso. En ese caso, está tratando con un recurso escaso y es posible que tenga que hacer concesiones, como reducir el tamaño de su pila para asignar en memoria un archivo grande o crear muchos subprocesos.

Pero, dado que las máquinas de 64 bits son omnipresentes, no creo que pase mucho tiempo antes de que el Tamaño de la memoria virtual sea una estadística completamente irrelevante.

¿Cuándo es importante el tamaño del conjunto de residentes?

El tamaño del conjunto residente es la parte del espacio de memoria virtual que se encuentra realmente en la RAM. Si su RSS se convierte en una parte significativa de su memoria física total, podría ser el momento de empezar a preocuparse. Si su RSS crece para ocupar toda su memoria física y su sistema comienza a intercambiarse, ya es hora de comenzar a preocuparse.

Pero RSS también es engañoso, especialmente en una máquina con poca carga. El sistema operativo no gasta mucho esfuerzo en recuperar las páginas utilizadas por un proceso. Al hacerlo, se obtienen pocos beneficios y existe la posibilidad de que se produzca un error de página costoso si el proceso toca la página en el futuro. Como resultado, la estadística RSS puede incluir muchas páginas que no están en uso activo.

Línea de fondo

A menos que esté intercambiando, no se preocupe demasiado por lo que le dicen las diversas estadísticas de memoria. Con la salvedad de que un RSS en constante crecimiento puede indicar algún tipo de pérdida de memoria.

Con un programa Java, es mucho más importante prestar atención a lo que sucede en el montón. La cantidad total de espacio consumido es importante y hay algunos pasos que puede tomar para reducirlo. Más importante es la cantidad de tiempo que pasa en la recolección de basura y qué partes del montón se recolectan.

Acceder al disco (es decir, una base de datos) es caro y la memoria es barata. Si puede cambiar uno por el otro, hágalo.

Existe un problema conocido con Java y glibc> = 2.10 (incluye Ubuntu> = 10.04, RHEL> = 6).

La cura es establecer este env. variable:

export MALLOC_ARENA_MAX=4

Si está ejecutando Tomcat, puede agregar esto a TOMCAT_HOME/bin/setenv.sh expediente.

Para Docker, agregue esto a Dockerfile

ENV MALLOC_ARENA_MAX=4

Hay un artículo de IBM sobre cómo configurar MALLOC_ARENA_MAX https://www.ibm.com/developerworks/community/blogs/kevgrig/entry/linux_glibc_2_10_rhel_6_malloc_may_show_excessive_virtual_memory_usage?lang=en

Esta publicación de blog dice

Se sabe que la memoria residente se arrastra de manera similar a una pérdida de memoria o una fragmentación de la memoria.

También hay un error JDK abierto JDK-8193521 “glibc desperdicia memoria con la configuración predeterminada”

busque MALLOC_ARENA_MAX en Google o SO para obtener más referencias.

Es posible que desee ajustar también otras opciones de malloc para optimizar la fragmentación baja de la memoria asignada:

# tune glibc memory allocation, optimize for low fragmentation
# limit the number of arenas
export MALLOC_ARENA_MAX=2
# disable dynamic mmap threshold, see M_MMAP_THRESHOLD in "man mallopt"
export MALLOC_MMAP_THRESHOLD_=131072
export MALLOC_TRIM_THRESHOLD_=131072
export MALLOC_TOP_PAD_=131072
export MALLOC_MMAP_MAX_=65536

La cantidad de memoria asignada para el proceso de Java está bastante a la par con lo que esperaría. He tenido problemas similares al ejecutar Java en sistemas integrados / con memoria limitada. Corriendo alguna Las aplicaciones con límites arbitrarios de VM o en sistemas que no tienen cantidades adecuadas de intercambio tienden a romperse. Parece ser la naturaleza de muchas aplicaciones modernas que no están diseñadas para su uso en sistemas con recursos limitados.

Tiene algunas opciones más que puede probar y limitar la huella de memoria de su JVM. Esto podría reducir la huella de memoria virtual:

-XX: ReservedCodeCacheSize = 32m Tamaño de caché de código reservado (en bytes): tamaño máximo de caché de código. [Solaris 64-bit,
amd64, and -server x86: 48m; in
1.5.0_06 and earlier, Solaris 64-bit and and64: 1024m.]

-XX: MaxPermSize = 64m Tamaño de la generación permanente. [5.0 and newer:
64 bit VMs are scaled 30% larger; 1.4
amd64: 96m; 1.3.1 -client: 32m.]

Además, también debe establecer su -Xmx (tamaño máximo de pila) en un valor lo más cercano posible al uso máximo real de memoria de su aplicación. Creo que el comportamiento predeterminado de la JVM sigue siendo doble el tamaño del montón cada vez que lo expande al máximo. Si comienza con un montón de 32M y su aplicación alcanzó un máximo de 65M, entonces el montón acabaría creciendo 32M -> 64M -> 128M.

También puede intentar esto para que la máquina virtual sea menos agresiva a la hora de hacer crecer el montón:

-XX: MinHeapFreeRatio = 40 Porcentaje mínimo de montón libre después de GC para evitar la expansión.

Además, por lo que recuerdo de experimentar con esto hace unos años, la cantidad de bibliotecas nativas cargadas tuvo un gran impacto en la huella mínima. Cargando java.net.Socket agregó más de 15M si recuerdo correctamente (y probablemente no).

¡Haz clic para puntuar esta entrada!
(Votos: 0 Promedio: 0)



Utiliza Nuestro Buscador

Deja una respuesta

Tu dirección de correo electrónico no será publicada. Los campos obligatorios están marcados con *