Por defecto PHP escribe los datos de la sesión en un fichero en disco. La función session_start()
bloquea el fichero de la sesión de forma que nadie más puede leerlo. Ese bloqueo durará hasta que se acaba la ejecución del script o hasta que se libere manualmente usando alguna de las siguientes funciones: session_abort()
, session_commit()
, session_destroy()
o session_write_close()
.
El problema
La mayoría de los programadores inician la sesión al principio del script, pero la dejan abierta hasta el final. Eso en entornos con múltiples peticiones sobre la misma sesión (por ejemplo con mucho AJAX) es posible que acabe causando un cuello de botella en momentos putuales. En el momento que una petición tarde demasiado en responder (debido por ejemplo a una query complicada) las demás peticiones no podrán acceder al fichero de sesión que estará bloqueado con el resultado de un tiempo de respuesta significativamente elevado. Pero sólo en momentos puntuales.
La solución
Lo más fácil es liberar el fichero de sesión justo después del session_start()
. Los datos de sesión seguirán disponibles para lectura a través de la variable superglobal $_SESSION
, pero no se podrá encribir nada en la sesión.
<?php // Start the session session_start(); // Session is open to read/write ops $_SESSION['latestRequestTime'] = time(); // Close the session session_write_close(); // Now session is read-only $user = $_SESSION['userNAme'];
Alternativas
Si no es posible cerrar la sesión justo después de abrirla (por ejemplo porque haya que escribir datos después de obtenerlos y/o procesarlos) hay algunas alternativas que pueden utilizarse: Guardar la sesión en una base de datos MySQL, ya que el motor de base de datos no bloquea la sesión, o bien guardar la sesión en algún sistema de caché como Redis, Memcache, etc.
Enlaces
- PHP Session Locks – How to Prevent Blocking Requests [EN]
- session_start() takes VERY LONG TIME (stackoverflow) [EN]
- session_start seems to be very slow (but only sometimes) (stackoverflow) [EN]
- Saving PHP's Session data to a database [EN]
- Storing Sessions in a Database [EN]
- How to save PHP Sessions to a database [EN]
- Saving PHP Sessions in Redis [EN]
- Manejo de sesiones (sitio web oficial) [ES]