postgresql - Doctrine2:PostgreSQL 的 YEAR、MONTH、DAY 或 DATE_FORMAT
问题描述
由于 Doctrine2 中既没有DAY()
, MONTH()
, YEAR()
, 也没有DATE_FORMAT()
,当使用 PostgreSQL 数据库时,如何在 Query Builder 中使用这些函数之一?
我找到了几个教程,但它们都适用于 MySQL,没有适用于 PostgreSQL。
解决方案
由于数据库供应商之间的 SQL 语法不同,因此不可能(或至少不是那么容易)创建独立于供应商的解决方案。所以这是 PostgreSQL 的一种方法。
我们要使用的 SQL 函数是to_char()
,请参阅https://www.postgresql.org/docs/9.6/static/functions-formatting.html
首先我们需要创建一个自定义 DQL 函数,参见https://www.doctrine-project.org/projects/doctrine-orm/en/2.6/cookbook/dql-user-defined-functions.html
namespace App\DQL;
use Doctrine\ORM\Query\AST\Functions\FunctionNode;
use Doctrine\ORM\Query\Parser;
use Doctrine\ORM\Query\Lexer;
use Doctrine\ORM\Query\SqlWalker;
class ToChar extends FunctionNode
{
public $timestamp = null;
public $pattern = null;
// This tells Doctrine's Lexer how to parse the expression:
public function parse(Parser $parser)
{
$parser->match(Lexer::T_IDENTIFIER);
$parser->match(Lexer::T_OPEN_PARENTHESIS);
$this->timestamp = $parser->ArithmeticPrimary();
$parser->match(Lexer::T_COMMA);
$this->pattern = $parser->ArithmeticPrimary(); // I'm not sure about `ArithmeticPrimary()` but it works. Post a comment, if you know more details!
$parser->match(Lexer::T_CLOSE_PARENTHESIS);
}
// This tells Doctrine how to create SQL from the expression - namely by (basically) keeping it as is:
public function getSql(SqlWalker $sqlWalker)
{
return 'to_char('.$this->timestamp->dispatch($sqlWalker) . ', ' . $this->pattern->dispatch($sqlWalker) . ')';
}
}
然后我们在 Symfony 4 中注册它,参见https://symfony.com/doc/current/doctrine/custom_dql_functions.html
# config/packages/doctrine.yaml
doctrine:
orm:
dql:
string_functions:
to_char: App\DQL\ToChar
现在我们可以在任何存储库中使用它:
return $this->createQueryBuilder('a')
->select("to_char(a.timestamp, 'YYYY') AS year")
->groupBy('year')
->orderBy('year', 'ASC')
->getQuery()
->getResult()
;
推荐阅读
- javascript - 防止 NavLink 中另一个元素的 NavLink 行为
- vba - 邮件发送后如何执行代码?
- sql - PrestoDB 中给定用户每天获取 1 行数据
- sqlite - SQLite - 一起使用 WHERE 和 ORDER BY 时提高查询性能
- reactjs - 尝试重复组件时出现打字稿错误
- javascript - 如何在 vue-good-table 中显示数组类型字段的下拉列表
- fortran - Fortran 输出文件将数据写入列而不是行
- android - 如何覆盖 Android Button 中的材料设计 statepressed
- r - 在 anaconda 中更新 R
- javascript - 实施 Checkmarx 建议的点击劫持修复引入了高严重性客户端 DOM XSS 漏洞