引言

我们知HTTP请求是没有状态的,两个请求之间没有直接的关联关系。但大多数情况下, 我们需要保持用户的会话间数据的连续性,这时,为了数据安全起见, 有必要在服务器上临时存储一些上下文数据了。

 

这就是 session 设计的目的。

代码时间

在laravel中可以使用系统提供的Session类方便地操作会话数据,而且其存储介质也是抽象出来的, 可以无缝衔接,只是驱动更换一下罢了。

框架内Session支持的驱动类型如下:

  • file 存储在 storage/framework/sessions 目录
  • cookie 存储在安全加密的 cookie 中
  • database 创建专门的表,存储在数据库内
  • memcached, redis 存储在这些内存数据库内
  • array 每次请求有效,用完即焚,适合测试使用

session 数据以简单的键值方式存储,所以读取某个键名的值,只需

session()->get('user_id');
复制代码

其中函数 session() 是laravel系统提供的助手函数。我们看一下源码的定义:

function session($key = null, $default = null)
{
    if (is_null($key)) {
        return app('session');
    }

    if (is_array($key)) {
        return app('session')->put($key);
    }

    return app('session')->get($key, $default);
}
复制代码

如果键名为空,返回一个 app('session'),使用应用容器实例化一个Session对象。 其中,别名 session 在门面 Illuminate\Support\Facades\Session 内注册的。

因为session是在页面请求之间的保持,所以Request对象也提供了session的方法。 在 FormRequestServiceProvider 内注册请求体时调用初始化方法:

$request = FormRequest::createFrom($app['request'], $request);
复制代码

其中 createFrom 方法做了session数据写入:

if ($session = $from->getSession()) {
    $request->setLaravelSession($session);
}
复制代码

所以有了以上基础,我们就可以这样链式调用获取一个请求所对应的session值:

Route::get('dashboard', function (Request $request) {
    $request->session()->get('user_id');
});
复制代码

或者使用依赖注入的方式,直接从存储中读取数据:

Route::get('dashboard', function (Illuminate\Session\Store $session) {
    return $session->get('user_id');
});
复制代码

为了发挥PHP灵活的特点,助手函数session也提供了完整无二的类似 request cache等等的操作方式读写数据。 比如读取某个键的值:

$value = session()->get('key');
$value = session('key');
复制代码

还有写入值:

session()->put('key', 'value');
session(['key', 'value']);
复制代码

使用技巧

下面说一些在存储session数据上的用法,以及框架提供的类的方法。 比如直接根据键名获取值:

$points = session()->get('points');
复制代码

如果键名不存在则使用默认值:

$points = session()->get('points', 0);
复制代码

来看一下 get 方法的定义,我们会学到更为灵活的用法:

public function get($key, $default = null)
{
    return Arr::get($this->attributes, $key, $default);
}
复制代码

注意在变量类型提示上,$default 是 mixed 泛型。再接着看 Arr::get() 方法的使用:

if (! static::accessible($array)) {
    return value($default);
}
复制代码

如果传入的$this->attributes 是数组,那么使用 value() 助手函数取值。 看一下该助手函数的定义:

function value($value)
{
    return $value instanceof Closure ? $value() : $value;
}
复制代码

我们发现,如果传入的是一个匿名函数,那么直接进行了调用,并返回执行后的值。这样, 我们就可以深度改造一下上面的 session()->get() 方法的第二个 default ,将其传入一个匿名函数, 用于处理获取默认值的逻辑。

看这样一段代码:

$points = session()->get('points', function () {
    return (new PointGetterService)->getPoints();
});
复制代码

是不是瞬间又有新知识了 :-) 有了匿名函数的支持,这个默认值你可以玩出花儿来了~~

写在最后

本文介绍了laravel处理请求间数据的保持方式:session。比介绍了在程序内获取session的方式, 并深入源码,探索了Session::get() 方法的高阶用法。

Happy coding :-)