PHP session 默认存于文件系统临时目录,路径由 session.save_path 配置决定,可通过 session_save_path() 查看;可切换为数据库或 Redis 存储,但需正确配置并注册 handler 或启用扩展,旧数据不会自动迁移。
PHP 的 session_start() 默认把会话数据写到临时目录的文件里,不是内存,也不是数据库。这个路径由 session.save_path 配置决定,通常在 php.ini 里设置,比如 /var/lib/php/sessions 或 C:\Windows\Temp。你可以用 session_save_path() 函数运行时查看当前值:
echo session_save_path(); // 输出实际保存路径
注意:Web 服务器(如 Apache/Nginx)必须对这个目录有读写权限,否则会报 Warning: session_start(): Failed to read session data。
PHP 不内置数据库 session 存储,得自己实现 SessionHandlerInterface,然后用 session_set_save_handler() 注册。关键点不是“换配置”,而是“接管整个读写生命周期”。常见错误是只改了 session.save_handler = user 却没注册 handler,结果
session 完全失效。
你需要做三件事:
SessionHandlerInterface,至少包含 read()、write()、destroy()、gc()
session_start() 前调用 session_set_save_handler($handler, true)
id(VARCHAR(128))、data(TEXT)、expires(INT)字段示例表(MySQL):
CREATE TABLE sessions ( id VARCHAR(128) NOT NULL PRIMARY KEY, data TEXT NOT NULL, expires INT NOT NULL );
比起手写数据库 handler,用 Redis 是更轻量、更可靠的替代方案——前提是已安装并启用 redis 或 phpredis 扩展。配置只需两步:
php.ini:session.save_handler = redis,session.save_path = "tcp://127.0.0.1:6379?database=2"
redis.so 已加载(Linux)或 php_redis.dll 已启用(Windows)验证是否生效:var_dump(ini_get('session.save_handler')); 应输出 "redis"。如果还是 "files",说明配置没重载或扩展未启用。重启 PHP-FPM 或 Apache 后再试。
从文件切到数据库或 Redis,PHP 不会帮你把老 session 文件里的数据搬过去。用户原有会话会丢失(表现为登录态突然掉线),因为新 handler 的 read() 找不到对应 id 的记录。
解决办法只有两个:
session.gc_maxlifetime)sess_* 文件内容,提取 key|s:value 结构),再批量写入新存储——但不推荐,格式易错且无加密保障真正要注意的是:无论存哪,session.cookie_httponly、session.cookie_secure 这些安全配置不能因存储方式改变而被忽略。存储位置变了,不代表 session 就安全了。