Doctrine2:PostgreSQL 的 YEAR、MONTH、DAY 或 DATE_FORMAT

3

由于Doctrine2不支持DAY()MONTH()YEAR()DATE_FORMAT()函数,那么在使用PostgreSQL数据库时,如何在查询构建器中使用其中一个这些函数呢?

我找到了几篇教程,但它们都适用于MySQL,而不是PostgreSQL。


你能否将时间戳提取到 PHP 对象中,并使用 PHP 的(虽然较差)时间戳操作工具来实现你的目标? - Jasen
当然你可以用PHP来实现。但在这种情况下,你不能使用->groupBy('year')(请参考下面的示例)。 - Thomas Landauer
1个回答

6

由于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()
;

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接