session自动退出登录问题

本文详细探讨了PHP中Session的工作原理及其可能导致的问题,特别是Session在特定条件下的失效问题。通过调整配置参数,如session.gc_maxlifetime等,可以有效延长或缩短Session的有效期。

先说说现象:今天登录9街后台的时候,很奇怪一登录就退出,后台代码一直没有改动过,查了一下原因,找到了根本原因,后面会贴上找到的相关文档。解决办法就是在产生的session的页面session_star();加上以下三句代码:

PHP Code 复制内容到剪贴板
  1. ini_set("session.gc_divisor", 1);  
  2. ini_set("session.gc_maxlifetime", 5);  
  3. ini_set("session.cookie_lifetime", 10);  

以下是转摘网上文档

首先 gc是什么?

gc, 是garbage collection 的简称.这个进程一般都跟着每起一个SESSION而开始运行的.gc目的是为了在session文件过期以后自动销毁删除这些文件.

1、session_gc_probaility
PHP默认不是每个SESSION启动都会启动一个GC来跟踪。这个参数是控制gc跟session启动概率。默认 1。值越大,概率越大。

2、session.gc_divisor
功能同上。 默认100。值越小,概率越大。

3、session.gc_maxlifetime 
超过设定时间,gc就认为是垃圾文件。

总结
session_gc_probaility和session.gc_divisor是一对控制gc启动的概率的两个参数。前者是分子,后者是分母。默认是1/100。 1%的几率。 也就是说100个请求中只有一个gc会伴随100个中的某个请求而启动。
session处理是所有的Web应用都必须面对的问题。PHP中对session有效期的处理,和其他的解决方案有着很大的不同,这是和PHP的工作机制相关的。

   在传统的client/server应用中,对于session失效的情况,可以交给网络协议自己来处理。无论是client端主动关闭连接,还是因为网络异常而导致的连接中断,server端都能够得到通知,触发连接中断的事件。只要编程响应这一事件,执行指定的操作即可。
    但对于web应用来说,情况却完全不一样。HTTP协议本身是无状态的,也就是说,每当client/server完成一次请求/响应的过程后,连接就会被断开。在断开连接以后,server并不知道client是否继续“在线”,还会继续发送下一次请求。
    换句话说,无论client端的用户已经关闭了浏览器窗口,还是用户仅仅在阅读当前网页并准备在下一秒钟继续浏览,或者用户因为Windows崩溃/停电/硬盘坏掉/网线被拔/地球爆炸而彻底无法再发送下一个请求,server都一无所知。(在HTTP 1.1中,浏览器可以通过keep-alive参数,来通知server不要在响应请求后主动断开连接,从而实现物理上的长连接。但是,这只是为了提高网络传输的性能而采取的措施,HTTP 在逻辑上仍然是无状态的。)因此,只能通过某种模拟的方式来判断当前session是否有效。如果某个session在超过一段时间后没有对server 端发出请求,server都会判断用户已经“离线”,当前session失效,并触发连接中断的事件。要做到这一点,server需要运行一个后台线程,定时扫描所有的session信息,判断session是否已经超时。
    PHP处理session的原理也不例外,但是在具体的实现方式上,却与众不同。这是因为,由于PHP的工作机制,它并没有一个后台线程,来定时地扫描session信息并判断其是否失效。它的解决之道是,当一个有效请求发生时,PHP会根据某个概率,来决定是否调用一个GC(Garbage Collector)。
    GC的工作,就是扫描所有的session信息,用当前时间减去session的最后修改时间(modified date),同配置参数(configuration option)session.gc_maxlifetime的值进行比较,如果生存时间已经超过gc_maxlifetime,就把该session删除。
    这是很容易理解的,因为如果每次请求都要调用GC代码,那么PHP的效率就会低得令人吃不消了。这个概率取决于配置参数 session.gc_probability/session.gc_divisor的值(可以通过php.ini或者ini_set()函数来修改)。
    默认情况下,session.gc_probability = 1,session.gc_divisor=100,也就是说有1%的可能性会启动GC。这三个参 数,session.gc_maxlifetime/session.gc_probability/session.gc_divisor都可以通过 php.ini或者ini_set()函数来修改。但要记得,如果使用ini_set()函数的话,必须在每一个页面的开始处都调用 ini_set()。
这又导致了另外一个问题,gc_maxlifetime只能保证session生存的最短时间,并不能够保存在超过这一时间之后session信息立即会得到删除。因为GC是按概率启动的,可能在某一个长时间内都没有被启动,那么大量的session在超过 gc_maxlifetime以后仍然会有效。当然,发生这种情况的概率很小,但是如果你的应用对 session的失效期要求很精确的话,这会导致很严重的问题。解决这个问题的一个方法是,把 session.gc_probability/session.gc_divisor的机率提高,如果提到100%,就会彻底解决这个问题,但显然会对性能造成严重的影响。另一个方法是放弃PHP的GC,自己在代码中判断当前session的生存时间,如果超出了 gc_maxlifetime,就清空当前session。
    PHP中的session有效期默认是1440秒(24分钟),也就是说,客户端超过24分钟没有刷新,当前session就会失效。要修改这个默认值,正确的解决办法是修改配置参数session.gc_maxlifetime。
我曾经在网上搜索过这个问题的解决方式,找到的结果千奇百怪。
    有的说要设置“session_life_time”,据我知所,PHP中没有这个参数。有的说要调用session_set_cookie_params,或者设置 session.cookie_lifetime,这仅仅用于设置client端cookie的生存时间,换言之,只当client端cookie的生存时间小于server端的session生存期时,修改这个值才有效,并且最长不能超过server端的session生存期,原因很简单,当 server端的session已经失效时,client端cookie的生存时间再长也是没有意义的。还有的说要调用 session_cache_expire,这个参数用于通知浏览器和proxy,当前页面的内容应该被缓存多长时间,和session的生存期并没有直接关系。
    听起来,这种解决方案很完美。但是,当你在实际中尝试修改session.gc_maxlifetime的值的时候,你很可能会发现,这个参数基本不起作用,session有效期仍然保持24分钟的默认值。甚至可能出现,在开发环境下工作正常,在服务器上却无效!
为了彻底解决这个问题,需要对PHP的工作细节进行进一步的分析。
在默认情况下,PHP 中的session信息会以文本文件的形式,被保存在系统的临时文件目录中。这个路径由配置参数session.save_path指定。在Linux下,这一路径通常为\tmp,在 Windows下通常为C:\Windows\Temp。当服务器上有多个PHP应用时,它们会把自己的session文件都保存在同一个目录中(因为它们使用同一个session.save_path参数)。同样地,这些PHP应用也会按一定机率启动GC,扫描所有的session文件。
问题在于,GC在工作时,并不会区分不同站点的session。举例言之,站点A的gc_maxlifetime设置为2小时,站点B的 gc_maxlifetime设置为默认的24分钟。当站点B的GC启动时,它会扫描公用的临时文件目录,把所有超过24分钟的session文件全部删除掉,而不管它们来自于站点A或B。这样,站点A的gc_maxlifetime设置就形同虚设了。
    找到问题所在,解决起来就很简单了。在页面的开始处调用session_save_path()函数,它能够修改session.save_path参 数,把保存session的目录指向一个专用的目录,例如\tmp\myapp\。这样,gc_maxlifetime参数就工作正常了。
    使用公用的session.save_path还会导致安全性问题,因为这意味着,同一台服务器上的其它PHP程序也可以读取你的站点的session文 件,这可能被用于黑客攻击。另一个问题是效率:在一个繁忙的站点中,可能存在成千上万个session文件,而把许多不同网站的session文件都放在 同一个目录下,无论是对单个文件的读写,还是遍历所有文件进行GC,都无疑会导致性能的降低。因此,如果你的PHP应用和别的PHP应用运行在同一台服务 器上的话,强烈建议你使用自己的 session.save_path。
    严格地来说,这算是PHP的一个bug。当PHP在进行GC时,它应该区别来自不同站点的session文件,并应用不同的gc_maxlifetime值。目前,最新的PHP 5.2.X仍然存在这个问题。
    上文说到,在一个繁忙的站点中,可能存在成千上万个session文件,即使区分了不同站点的session.save_path目录,单个站点的session文件数目仍然可能导致效率问题。

在系统使用过程中,如果用户操作时自动跳转到登录界面,这通常与会话管理、Token 过期或 Cookie 失效等相关。以下是问题的排查与解决方案: ### 问题排查 1. **检查 Token 有效性** 系统通常使用 Token(如 JWT)进行身份验证和会话管理。如果 Token 过期或被清除,系统会跳转到登录页。可以通过浏览器开发者工具检查 `Cookie` 或 `LocalStorage` 中是否存在有效的 Token。如果 Token 不存在或已过期,则需要重新登录。 2. **检查 Token 存储方式** 如果 Token 存储在 Cookie 中,需确认 Cookie 是否设置了正确的 `Domain` 和 `Path` 属性。如果设置错误,可能导致 Token 无法正确读取。此外,检查是否启用了 `HttpOnly` 或 `Secure` 标志,这些标志可能会影响前端对 Cookie 的访问。 3. **检查 Token 刷新机制** 如果系统支持 Token 刷新机制,需确认刷新逻辑是否正常。例如,当 Token 即将过期时,系统应自动调用刷新接口获取新的 Token。如果刷新失败,可能会导致用户被强制登出。 4. **检查后端会话管理逻辑** 后端可能设置了较短的会话超时时间,或者在某些操作后主动清除了会话信息。检查后端代码或日志,确认是否存在强制登出的逻辑,例如调用了清除 Token 的接口。 5. **检查跨域问题** 如果前端与后端存在跨域请求,需确认是否正确配置了跨域头(如 `Access-Control-Allow-Credentials`)。如果跨域请求未携带 Cookie,可能导致 Token 无法传递。 6. **检查浏览器安全策略** 某些浏览器(如 Safari)对第三方 Cookie 有严格的限制,可能导致 Token 无法正常存储或读取。检查浏览器控制台是否有相关警告或错误信息。 ### 解决方案 1. **延长 Token 有效期** 如果 Token 过期时间较短,可以适当延长 Token 的有效期,以减少用户频繁登录的情况。同时,确保 Token 刷新机制正常工作,以便在 Token 过期前自动刷新。 2. **优化 Cookie 设置** 确保 Cookie 的 `Domain` 和 `Path` 设置正确,以便在不同页面或子域名下都能正确读取。例如,如果系统部署在 `example.com`,可以将 `Domain` 设置为 `.example.com`,以确保子域名也能访问 Cookie。 3. **实现 Token 刷新机制** 在前端定期调用 Token 刷新接口,确保 Token 始终有效。可以结合 `setInterval` 定时刷新,或者在每次请求失败时尝试刷新 Token。 ```javascript // 示例:使用 Axios 拦截器实现 Token 刷新 import axios from 'axios'; const apiClient = axios.create({ baseURL: 'https://api.example.com', withCredentials: true, }); let isRefreshing = false; let refreshSubscribers = []; const onRefreshed = (token) => { refreshSubscribers.forEach((callback) => callback(token)); refreshSubscribers = []; }; const addRefreshSubscriber = (callback) => { refreshSubscribers.push(callback); }; apiClient.interceptors.response.use( (response) => response, async (error) => { const { config, response } = error; const originalRequest = config; if (response && response.status === 401 && !originalRequest._retry) { originalRequest._retry = true; if (!isRefreshing) { isRefreshing = true; try { const refreshTokenResponse = await axios.post('/refresh-token'); const newToken = refreshTokenResponse.data.token; isRefreshing = false; onRefreshed(newToken); return apiClient(originalRequest); } catch (refreshError) { isRefreshing = false; // 处理刷新失败的情况,例如跳转到登录页 window.location.href = '/login'; } } else { return new Promise((resolve) => { addRefreshSubscriber((token) => { originalRequest.headers['Authorization'] = `Bearer ${token}`; resolve(apiClient(originalRequest)); }); }); } } return Promise.reject(error); } ); ``` 4. **检查后端会话配置** 如果后端使用了会话管理(如基于 Cookie 的会话),需检查会话超时时间配置。可以在后端配置文件中调整 `session.timeout` 参数,延长会话有效期。 5. **处理跨域问题** 如果存在跨域请求,需在后端配置中启用 `Access-Control-Allow-Credentials`,并在前端请求中设置 `withCredentials: true`,以确保 Cookie 能够正确传递。 6. **兼容浏览器安全策略** 针对 Safari 等浏览器对第三方 Cookie 的限制,可以考虑使用 `SameSite=None; Secure` 的 Cookie 属性,并确保使用 HTTPS 协议。 ```http Set-Cookie: token=abc123; Path=/; Domain=.example.com; Secure; HttpOnly; SameSite=None ``` 7. **清理浏览器缓存与 Cookie** 如果用户长时间未清理浏览器缓存,可能导致 Token 或 Cookie 异常。建议用户定期清理浏览器缓存与 Cookie,以避免因旧数据导致的登录问题。 8. **排查 LDAP 集成问题** 如果系统集成了 LDAP 认证,需检查 LDAP 配置是否正确,特别是域名链接是否拼接正确。如果 LDAP 登录页无法正常加载,可能导致用户无法重新登录。 ```python # 示例:检查 LDAP 登录链接拼接逻辑 def generate_ldap_login_url(domain): ldap_base_url = "https://ldap.example.com/login" return f"{ldap_base_url}?domain={domain}" ``` 9. **检查环境变量与配置文件** 如果系统依赖环境变量或配置文件,需确认相关配置是否正确。例如,Token 的密钥、域名、超时时间等参数是否配置正确。 ```bash # 示例:环境变量配置示例 TOKEN_SECRET=your-secret-key TOKEN_EXPIRATION=3600 # 单位:秒 DOMAIN=example.com ``` 10. **排查最近修改的内容** 如果上述方法均无法解决问题,可以回忆最近是否修改了系统文件、安装了新组件或更新了依赖库。逐一排查这些更改,确认是否影响了登录流程。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值