gpt4 book ai didi

mysql - 将 MySQL 转换为 Doctrine 查询生成器。 IF 和 CONCAT 的问题。或者选择子查询的另一种方法

转载 作者:IT老高 更新时间:2023-10-28 23:46:26 26 4
gpt4 key购买 nike

我的类别有一个表,每个类别都有 id、name 和 parent_id。

Select IF(a.parent_id IS NULL, a.name, CONCAT((SELECT b.name FROM category b WHERE b.id = a.parent_id), " / ", a.name) ) as n, a.id, a.parent_id
FROM category a
ORDER BY n

我想将它转换为我的 Doctrine2 查询生成器

    $em = $this->getDoctrine()->getEntityManager();
$qb = $em->createQueryBuilder();
$q = $qb
->select("c.id")
->addSelect(
"IF(c.parent_id IS NULL, c.name, CONCAT((" .
$em->createQueryBuilder()
->select("t.name")
->from("MyBundle:Category", "t")
->getQuery()->getDQL() .
"), \" / \", c.name) )"
)
->from("MyBundle:Category", "c");
echo $q->getQuery()->getSQL();
exit;

类似的东西,但我不能使用 IFCONCAT

最佳答案

好的,我找到了解决方案。

您可以使用CASE 代替IF。检查一下,但是当我使用 CASE 时,我不能 CONCAT 我的字段:

$em = $this->getDoctrine()->getEntityManager();
$qb = $em->createQueryBuilder();
$q = $qb
->select("c.id")
->addSelect("CASE WHEN (c.parent IS NULL) THEN c.name ELSE 'something' END")
->from("MyBundle:Category", "c")
->leftJoin("c.parent", "t");

echo $q->getQuery()->getSQL();

另一种解决方案是创建您自己的 DQL 函数,例如 IF 并像这样使用它:

$em = $this->getDoctrine()->getEntityManager();
$qb = $em->createQueryBuilder();
$q = $qb
->select("c.id")
->addSelect("IF(c.parent IS NULL, c.name, CONCAT(CONCAT(t.name, ' / '), c.name))")
->from("MyBundle:Category", "c")
->leftJoin("c.parent", "t");

echo $q->getQuery()->getSQL();

要创建此 IF,您可以转到此链接并了解:http://docs.doctrine-project.org/projects/doctrine-orm/en/latest/reference/dql-doctrine-query-language.html#adding-your-own-functions-to-the-dql-language

我将在这里发布我的这个 IF 类(class)和 config.yml 以帮助其他人。这是 IfFunction 类(我从 https://github.com/beberlei/DoctrineExtensions/blob/master/src/Query/Mysql/IfElse.php 得到的):

<?php
namespace MyName\MiscBundle\Doctrine\ORM\Query\AST\Functions;

use Doctrine\ORM\Query\AST\Functions\FunctionNode;
use Doctrine\ORM\Query\Lexer;

/**
* Usage: IF(expr1, expr2, expr3)
*
* If expr1 is TRUE (expr1 <> 0 and expr1 <> NULL) then IF() returns expr2;
* otherwise it returns expr3. IF() returns a numeric or string value,
* depending on the context in which it is used.
*
* @author Andrew Mackrodt <andrew@ajmm.org>
* @version 2011.06.19
*/
class IfFunction extends FunctionNode
{
private $expr = array();

public function parse(\Doctrine\ORM\Query\Parser $parser)
{
$parser->match(Lexer::T_IDENTIFIER);
$parser->match(Lexer::T_OPEN_PARENTHESIS);
$this->expr[] = $parser->ConditionalExpression();

for ($i = 0; $i < 2; $i++)
{
$parser->match(Lexer::T_COMMA);
$this->expr[] = $parser->ArithmeticExpression();
}

$parser->match(Lexer::T_CLOSE_PARENTHESIS);
}

public function getSql(\Doctrine\ORM\Query\SqlWalker $sqlWalker)
{
return sprintf('IF(%s, %s, %s)',
$sqlWalker->walkConditionalExpression($this->expr[0]),
$sqlWalker->walkArithmeticPrimary($this->expr[1]),
$sqlWalker->walkArithmeticPrimary($this->expr[2]));
}
}

之后,您需要像这样更新您的 config.yml(只添加了最后 3 行):

doctrine:
dbal:
driver: "%database_driver%"
host: "%database_host%"
port: "%database_port%"
dbname: "%database_name%"
user: "%database_user%"
password: "%database_password%"
charset: UTF8

orm:
auto_generate_proxy_classes: "%kernel.debug%"
auto_mapping: true
dql: #ADDED THIS LINE
string_functions: #ADDED THIS LINE
IF: MyName\MiscBundle\Doctrine\ORM\Query\AST\Functions\IfFunction #ADDED THIS LINE

谢谢

关于mysql - 将 MySQL 转换为 Doctrine 查询生成器。 IF 和 CONCAT 的问题。或者选择子查询的另一种方法,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/14098219/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com