我从以下文档中了解到: http://docs.doctrine-project.org/en/2.1/reference/dql-doctrine-query-language.html#dql-functions 没有ROUND函数但是有一种简单的方法可以在不编写自己的DQL类函数的情况下完成它吗?
修改 如果做平均值并且返回整数,我就不需要完全匹配。
答案 0 :(得分:7)
您需要为此实现custom DQL function。
DoctrineExtensions中有一些例子。
您可以像下面这样实现它:
<?php
namespace MyApp\DQL;
use Doctrine\ORM\Query\AST\Functions\FunctionNode;
use Doctrine\ORM\Query\Lexer;
use Doctrine\ORM\Query\SqlWalker;
class Round extends FunctionNode
{
private $arithmeticExpression;
public function getSql(SqlWalker $sqlWalker)
{
return 'ROUND(' . $sqlWalker->walkSimpleArithmeticExpression(
$this->arithmeticExpression
) . ')';
}
public function parse(\Doctrine\ORM\Query\Parser $parser)
{
$lexer = $parser->getLexer();
$parser->match(Lexer::T_IDENTIFIER);
$parser->match(Lexer::T_OPEN_PARENTHESIS);
$this->arithmeticExpression = $parser->SimpleArithmeticExpression();
$parser->match(Lexer::T_CLOSE_PARENTHESIS);
}
}
然后您可以在引导ORM时在配置中注册它:
$config = new \Doctrine\ORM\Configuration();
$config->addCustomNumericFunction('ROUND', 'MyApp\DQL\Round');
答案 1 :(得分:6)
更清洁的方法是使用略微修改的@Ocramius代码。
将这段代码放在src/YourNamespace/YourMainBundle/DoctrineFunctions/
目录中作为Round.php
文件名:
<?php
namespace YourApp\YourMainBundle\DoctrineFunctions;
use Doctrine\ORM\Query\AST\Functions\FunctionNode,
Doctrine\ORM\Query\Lexer;
class Round extends FunctionNode
{
private $arithmeticExpression;
public function parse(\Doctrine\ORM\Query\Parser $parser)
{
$lexer = $parser->getLexer();
$parser->match(Lexer::T_IDENTIFIER);
$parser->match(Lexer::T_OPEN_PARENTHESIS);
$this->arithmeticExpression = $parser->SimpleArithmeticExpression();
$parser->match(Lexer::T_CLOSE_PARENTHESIS);
}
public function getSql(\Doctrine\ORM\Query\SqlWalker $sqlWalker)
{
return 'ROUND(' . $sqlWalker->walkSimpleArithmeticExpression($this->arithmeticExpression) . ')';
}
}
然后将其放入app/config/config.yml
:
doctrine:
dql:
numeric_functions:
round: YourApp\YourMainBundle\DoctrineFunctions\Round
这将允许您直接在DQL SELECT查询中使用ROUND()
函数;无论是使用QueryBuilder还是直接通过createQuery()
答案 2 :(得分:3)
如果您希望能够指定舍入精度,则可以使用提供的类here。如果您正在使用symfony,请安装捆绑包,因为您还将获得额外的标准mysql函数。
链接资源的代码也可在下面找到:
<?php
namespace Mapado\MysqlDoctrineFunctions\DQL;
use \Doctrine\ORM\Query\AST\Functions\FunctionNode;
use \Doctrine\ORM\Query\Lexer;
/**
* MysqlRound
*
* @uses FunctionNode
* @author Julien DENIAU <julien.deniau@mapado.com>
*/
class MysqlRound extends FunctionNode
{
/**
* simpleArithmeticExpression
*
* @var mixed
* @access public
*/
public $simpleArithmeticExpression;
/**
* roundPrecission
*
* @var mixed
* @access public
*/
public $roundPrecission;
/**
* getSql
*
* @param \Doctrine\ORM\Query\SqlWalker $sqlWalker
* @access public
* @return string
*/
public function getSql(\Doctrine\ORM\Query\SqlWalker $sqlWalker)
{
return 'ROUND(' .
$sqlWalker->walkSimpleArithmeticExpression($this->simpleArithmeticExpression) .','.
$sqlWalker->walkStringPrimary($this->roundPrecission) .
')';
}
/**
* parse
*
* @param \Doctrine\ORM\Query\Parser $parser
* @access public
* @return void
*/
public function parse(\Doctrine\ORM\Query\Parser $parser)
{
$parser->match(Lexer::T_IDENTIFIER);
$parser->match(Lexer::T_OPEN_PARENTHESIS);
$this->simpleArithmeticExpression = $parser->SimpleArithmeticExpression();
$parser->match(Lexer::T_COMMA);
$this->roundPrecission = $parser->ArithmeticExpression();
if ($this->roundPrecission == null) {
$this->roundPrecission = 0;
}
$parser->match(Lexer::T_CLOSE_PARENTHESIS);
}
}