以 Yii 2.0 基础版为例来介绍 Yii 中是如何处理session的,高级版类似。
Yii 2.0 中对session进行了封装,类文件为
/basic/vendor/yiisoft/yii2/web/Session.php,且已注册为内置组件(session),一般我们会在此基础上对 session 进行二次封装。
1. session 的 curd 操作
在自定义组件目录 /basic/components 下新建一个类文件 SessionHelper.php,来对 session 进行二次封装,代码如下:
session->set($key, $value);
}
/**
* 读取数据
*/
public static function get($key, $defaulValue=NULL)
{
$session = \Yii::$app->session;
return $session->get($key, $defaulValue);
}
/**
* 删除session中指定的键值对
*/
public static function del($key)
{
\Yii::$app->session->remove($key);
}
/**
* 获取sessionId的值
*/
public static function getId()
{
return \Yii::$app->session->getId();
}
}
然后,在控制器层,新增一个控制器 SessController.php,代码如下:
最后,在浏览器地址栏,输入 http://basic.com/sess/get 和 http://basic.com/sess/index,进行测试。
2. sessionFlash 的应用
原理:sessionFlash 只在下一次请求时可以使用,使用一次(读取一次)后就会失效。
应用场景:订单支付(防止表单数据二次提交)
在
SessionHelper.php 类文件中,添加如下方法:
/**
* 设置sessionFlash数据
*/
public static function setFlash($key, $value)
{
\Yii::$app->session->setFlash($key, $value);
}
/**
* 读取sessionFlash数据
*/
public static function getFlash($key, $defaulValue=NULL)
{
return \Yii::$app->session->getFlash($key, $defaulValue);
}
/**
* 检测sessionFlash数据是否存在
*/
public static function hasFlash($key)
{
return \Yii::$app->session->hasFlash($key);
}
在
SessController.php 控制器文件中,添加如下方法:
public function actionSetFlash()
{
// 创建订单成功时,设置flash数据
SessionHelper::setFlash('order_id', '3');
}
public function actionGetFlash()
{
$key = 'order_id';
// 支付订单时,验证订单是否已经支付
if (SessionHelper::hasFlash($key)) { // 未支付
// 实现支付逻辑
echo SessionHelper::getFlash($key);
} else {
echo '请不要重复提交表单!';
}
}
在浏览器地址栏,输入 http://basic.com/sess/set-flash 和 http://basic.com/sess/get-flash 进行测试。
3. session 的共享
session 默认是以磁盘文件的形式保存在web服务器的硬盘上,有些项目,为了实现 session 的共享,即多台web服务器共享 session,可以更改 session 的保存方式,常用的实现方法有两种:
- 将 session 数据保存在 MySQL 数据库中
- 将 session 数据保存在缓存服务器中(redis、memcache等)
这里,我们只介绍将 session 保存在数据库中的方法。
一般来说,把 session 保存在数据库中,需要修改 php.ini 配置文件,但 Yii 2.0 中不需要直接修改 php.ini文件,就可以把 session 保存到数据库中,它对 数据库session 进行了封装,类文件为
\basic\vendor\yiisoft\yii2\web\DbSession.php。
首先,我们需要将 DbSession.php 注册为 session 组件,即在 basic/config/web.php 文件中的 components 中,添加如下的代码:
// 注册session组件(会覆盖默认的session组件),将session保存到数据库中
'session' => [
'class' => 'yii\web\DbSession',
'db' => 'db', // db组件(控制要连接的数据库服务器)的名称
'sessionTable' => 'basic_session', // 数据库中数据表的名称(用于保存session)
],
然后,根据 DbSession.php 文件中创建表的提示,在数据库中创建表 basic_session,sql如下:
CREATE TABLE `basic_session` (
`id` char(40) NOT NULL,
`expire` int(11) DEFAULT NULL,
`data` longblob,
PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8;
这样配置以后,Yii 中通过 session 组件 对 session 执行的操作,就会体现在数据库中的
basic_session 表,而不会再保存到磁盘文件了。
直接在浏览器地址栏,输入 http://basic.com/sess/index,进行测试,查看数据库 basic_session的记录变化。