sql >> データベース >  >> RDS >> Mysql

MySQLをDoctrineQueryBuilderに変換します。 IFおよびCONCATに関する問題。または選択時のサブクエリの別のアプローチ

    解決策を見つけました。

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

    別の解決策は、IFのような独自のDQL関数を作成することです。 次のように使用します:

    $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のクラスと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 <[email protected]>
     * @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
    

    ありがとう



    1. ツリー構造用に最適化されたSQL

    2. MySQLで末尾の空白を削除する方法

    3. SQLで設定されたルールに基づいて列の行をフィルタリングする

    4. MySQLで重複する値を見つける