- 在【config/database.php】将 “trigger_sql” 设置为true,会记录SQL日志 。但是无论是否请求中出错,都会记录。
// 监听SQL
'trigger_sql' => true,
- 在日志配置【config/log.php】里添加这个日志通道,“error” 的报错给 “file”;“sql” 的报错给 “sql”;这样是为了,报错时关掉 “sql” 日志而不关掉 “error” 日志。
// 日志类型记录的通道 ['error'=>'email',...]
'type_channel' => ['error' => 'file', 'sql' => 'sql'],
- 新增一个日志通道,是专门给sql的。(不知道实时记录和非实时记录的区别,有懂得大佬请赐教)
// 其它日志通道配置
'sql' => [
// 日志记录方式
'type' => 'File',
// 日志保存目录
'path' => '',
// 单文件日志写入
'single' => false,
// 独立日志级别
'apart_level' => ['sql'],
// 最大日志文件数量
'max_files' => 0,
// 使用JSON格式记录
'json' => false,
// 日志处理
'processor' => null,
// 关闭通道日志写入
'close' => false,
// 日志输出格式化
'format' => '[%s][%s] %s',
// 是否实时写入
'realtime_write' => true,
],
- 最后在【app\ExceptionHandle.php】文件中加入 “事务回滚” 和 “关闭日志” ,这样就OK了,系统报错时会调用这个 “render()” 方法,然后就事务回滚,关闭 “sql” 日志。
public function render($request, Throwable $e): Response
{
event('dbRollback'); // 加这个,只要报错就回滚
Log::close('sql'); // 加这个,没报错sql日志正常记录,报错sql日志不会记录
// 添加自定义异常处理机制
if ($e instanceof MyException) {
return MyException::response($request, $e);
} else {
// 其他错误交给系统处理
return parent::render($request, $e);
}
}
event(‘dbRollback’)是自己写的回滚时间监听。
在【app\event\Transaction.php】中自定义一个事件监听,只要在需要的地方使用event(‘dbStart’)就能开启事务,其他同样如此。
<?php
namespace app\event;
use think\Event;
use think\facade\Db;
class Transaction
{
private $start = false;
public function DbStart()
{
if (!$this->start) {
$this->start = true;
Db::startTrans();
}
}
public function DbCommit()
{
if ($this->start) {
Db::commit();
}
}
public function DbRollback()
{
if ($this->start) {
Db::rollback();
}
}
public function subscribe(Event $event)
{
$event->listen('dbStart', [$this, 'DbStart']);
$event->listen('dbCommit', [$this, 'DbCommit']);
$event->listen('dbRollback', [$this, 'DbRollback']);
}
}