8

我正在使用 CI 的会话与数据库连接。所以我们所有的会话都在我们数据库的这个 ci_sessions 表中,考虑到 session_id 每 5 分钟变化一次,它可以获得很多行。

我们是否需要清空桌子,比如每月/每周一次?

4

6 回答 6

13

虽然@Marc-Audet 说的是真的,但如果你看一下代码,你会发现这是一种非常糟糕的清理会话的方法。

构造_sess_gc函数每次启动时都会调用该函数。因此,基本上每个请求都会自动加载到您的服务器上。

然后,它会生成一个低于 100 的随机数,并查看它是否低于某个值(默认为 5)。如果满足此条件,那么它将删除会话表中last_activity值小于当前时间减去会话到期时间的所有行。

虽然这适用于大多数情况,但技术上可能(如果世界真的是随机的)随机数生成器不会长时间生成低于 5 的数字,在这种情况下,您的会话将不会被清理。

此外,如果您将会话到期时间设置为较长时间(如果设置为 0,CI 会将其设置为 2 年),那么这些行无论如何都不会被删除。如果您的网站足以吸引大量访问者,那么您的 DBA 很快就会指责会话表 :)

它适用于大多数情况 - 但我不会称其为适当的解决方案。他们的会话 id 重新生成确实应该被构建以删除与以前的 id 有关的记录,并且垃圾收集确实不应该留给随机数 - 从理论上讲,可能不会像您希望的那样频繁生成所需的数字.

在我们的例子中,我已经从会话库中删除了会话垃圾收集,并且我每天手动处理一次(使用 cron 作业 .. 和合理的会话到期时间)。这减少了对数据库的不必要点击次数,也不会在数据库中留下大量表。它仍然是一张大桌子,但比以前小了很多。

于 2013-03-12T16:15:11.770 回答
6

鉴于 OP 问题没有 CodeIgniter 2 标记,我将回答如何在 CodeIgniter 3 的数据库不断增长时处理会话清理。

问题:
当您(在 config.php 文件中)设置sess_expiration密钥太高(比如 1 年)和sess_time_to_update密钥太低(比如 5 分钟)时,会话表将随着用户浏览您的网站而不断增长,直到会话行将过期并将被垃圾收集(即 1 年)。

解决方案:
sess_regenerate_destroykey 设置为TRUE(默认设置为FALSE)将在旧会话使用新 id 重新生成时删除旧会话,从而自动清理您的表。

于 2018-02-17T22:14:17.847 回答
5

不,CodeIgniter 会自行清理...

注意
根据CodeIgniter 文档

Session 类具有内置的垃圾收集功能,可以清除过期的会话,因此您无需编写自己的例程来执行此操作。

CodeIgniter 的会话类可能会检查会话表并清理过期条目。但是,文档没有说明何时进行清理。由于 CodeIgniter 没有 cron 作业,因此必须在调用 Session 类时进行清理。我想如果站点永远保持空闲状态,会话表将永远不会被清除。但是,这将是一个不寻常的案例。

于 2013-03-12T12:14:14.953 回答
3

CodeIgniter 实现了SessionHandlerInterface(请参阅自定义驱动程序的文档)。

CodeIgniter 为每个驱动程序(数据库、文件、redis 等)定义了一个名为gc()的垃圾收集器方法,或者您可以为自定义驱动程序定义自定义 gc()。

gc()方法通过 session_set_save_handler() 函数传递给 PHP ,因此PHP在内部根据session.gc_divisorsession.gc_probability设置调用垃圾收集器。

例如,使用以下设置:

session.gc_probability = 1
session.gc_divisor = 100

垃圾收集器进程在每个请求上启动的几率为 1%。

因此,如果设置正确,则无需清理会话表。

于 2016-10-11T20:49:28.003 回答
1

你打电话时:

$this->session->sess_destroy();

它会自行删除数据库中的信息。

于 2013-03-12T12:15:06.680 回答
-2

清理桌子总是很好的做法。否则,如果您查询会话数据以创建报告或其他内容,它将变得缓慢且不可靠。尽管如此,考虑到mysql的性能,是的。

于 2013-03-12T12:15:03.980 回答