Laravel redis 中 select 操作影响其他连接
在 laravel 框架中使用 Redis 时,如果通过 Redis::connection(‘config1’) 获取的 Redis 连接,执行 select 操作后,可能会影响之前连接数据库的数据。
问题分析
通过查看 IlluminateSupportFacadesRedis 源码,可知 getFacadeaccessor 方法返回了 redis。使用 dd(get_class(app(‘redis‘))) 可以看到,redis 是由 IlluminateRedisRedisManager 实现的。
IlluminateRedisRedisManager 的 connection 方法对于相同名称的连接,只会在首次解析后保存起来。后续需要时直接返回,即同一名称每次调用 connection 都返回相同 Redis 实例。这导致多个连接指向同一 Redis 实例,从而导致 select 操作影响其他连接。
解决方案
如果需要多个独立的连接,可以使用以下方案:
- 克隆连接:但经过测试发现,对 Redis 连接克隆后,实际使用时仍是同一连接,不能满足需求。
- 使用 resolve 方法创建新连接:$b = app()->make(‘config1’) 这样创建的 $b 将是一个新连接,不会受 select 操作影响。
需要注意的是,Laravel Redis 连接如果启用 ‘cluster’ 配置项且节点数目大于 1,则不适用此解决方案,因为此时 Laravel 自动实现的集群连接机制会导致不同 connection 名称的实例也会指向同一底层连接池。