LEFT JOIN,如果存在第二个表的结果,则添加到 PHP 中的数组

Posted

技术标签:

【中文标题】LEFT JOIN,如果存在第二个表的结果,则添加到 PHP 中的数组【英文标题】:LEFT JOIN, add to array in PHP if results from second table exist 【发布时间】:2016-08-15 13:02:23 【问题描述】:

我正在尝试通过显示某些术语的定义(如果存在定义)来扩展我正在构建的字典的功能。

我从两个表中取数据如下:

    $query = $db->query("SELECT * FROM ".DICTIONARY_TABLE." " .
            "LEFT JOIN ".DICTIONARY_DEFINITIONS." ON ".DICTIONARY_TABLE.".id = ".DICTIONARY_DEFINITIONS.".term_id ".
            "WHERE ".DICTIONARY_TABLE.".".$source." LIKE '%".$keyword."%' ".
            "AND ".DICTIONARY_TABLE.".theme_id = ".$selected_theme_id." ".
            "ORDER BY ".DICTIONARY_TABLE.".id");

之后,在while循环中,我首先获取主题名称(从另一个表中),然后将查询结果添加到数组中:

        while($row = $query->fetch(PDO::FETCH_ASSOC)) 
    // get theme name
    $theme_name = "theme_".$lang;
    $theme_query= $db->query("SELECT theme_id,".$theme_name." FROM ".DICTIONARY_THEMES." WHERE theme_id = ".$theme_id."");
    $theme_row  = $theme_query->fetch(PDO::FETCH_ASSOC);
    $theme      = $theme_row[$theme_name];

    // add all results to an array
    $results[] = array(
        'english'           => $row['english'],
        'bulgarian'         => $row['bulgarian'],
        'english_abbr'      => $row['english_abbr'],
        'bulgarian_abbr'    => $row['bulgarian_abbr'],
        'theme'             => $theme
    );

之后,我尝试检查 LEFT JOIN 是否确实从定义表中返回了任何结果,如果是,也将这些结果添加到数组中 - 但这是我失败的地方...

    // check if definition exists for this term
    if(isset($row['bulgarian_definition'])) 
        array_push($results['bulgarian_definition'], $row['bulgarian_definition']);
    
    if(isset($row['english_definition'])) 
        array_push($results['english_definition'], $row['english_definition']);
    

我已经尝试了所有可以找到的方法,首先检查变量是否已定义,然后将它们推送到$results 数组。没有任何效果。

我似乎无法成功找出english_definition 和/或bulgarian_definition 的设置。当我在 phpMyAdmin 中运行查询本身时,它工作得很好。

我能想到的唯一“解决方案”是放弃为定义创建一个单独的表并扩展主表的想法,但我知道这不是一个好方法。 任何关于我做错了什么的见解将不胜感激。谢谢!

编辑:我改变了元素添加到数组的方式:

    // check if definition exists for this term
    if(isset($row['bulgarian_definition'])) 
        $results['bulgarian_definition'] = $row['bulgarian_definition'];
    
    if(isset($row['english_definition'])) 
        $results['english_definition'] = $row['english_definition'];
    

这就是诀窍。当我将$results 数组转储到while 循环之外时,两个定义都已添加。

但是,我现在得到了大量的 Warning: Illegal string offset 'theme' in... 和 'english' 和 'bulgarian' - 这发生在下面,当我在 foreach 循环中运行 $results 数组开始打印它们时:

    foreach($results as $result) 

    if($theme != $result['theme']) 
        $theme = $result['theme'];
        $search_results .= "<h3>" . $result['theme'] . "</h3>";
    

    if($source == "english") 
        foreach ($keywords as $keyword) 
            $result['english'] = preg_replace("|($keyword)|Ui", "<span style=\"color:#780223\">" . $keyword . "</span>", $result['english']);
        

还不知道为什么会发生这种情况,会继续寻找。

第二次编辑:决定将这两个定义直接放在 $results 数组中,如下所示:

    while($row = $query->fetch(PDO::FETCH_ASSOC)) 
    // get theme name
    $theme_name = "theme_".$lang;
    $theme_query= $db->query("SELECT theme_id,".$theme_name." FROM ".DICTIONARY_THEMES." WHERE theme_id = ".$theme_id."");
    $theme_row  = $theme_query->fetch(PDO::FETCH_ASSOC);
    $theme      = $theme_row[$theme_name];

    // add all results to an array
    $results[] = array(
        'english'           => $row['english'],
        'bulgarian'         => $row['bulgarian'],
        'english_abbr'      => $row['english_abbr'],
        'bulgarian_abbr'    => $row['bulgarian_abbr'],
        'theme'             => $theme,
        'bulgarian_definition'  => $row['bulgarian_definition'],
        'english_definition'    => $row['english_definition']
    );  
// end while

这现在可以正常工作了。当我转储数组时,如果不存在定义,我有'english_definition' =&gt; null,如果存在定义,它就在那里。到目前为止一切顺利。

新问题是我不能再按主题对结果进行分组 - 显示最后找到的结果的主题。这完全是一个不同的问题。真正让我恼火的是,在我添加定义之前,一切正常。您可以在工作网站here。

问题解决了!

    决定不向数组推送值(如上所示)。

    删除了while 循环中获取主题名称的额外查询,将其移至执行搜索的查询。因此查询本身是:

                    $query = $db->query("SELECT * FROM ".DICTIONARY_TABLE." " .
            "JOIN ".DICTIONARY_THEMES." ON ".DICTIONARY_TABLE.".theme_id = ".DICTIONARY_THEMES.".theme_id ".
            "LEFT JOIN ".DICTIONARY_DEFINITIONS." ON ".DICTIONARY_TABLE.".id = ".DICTIONARY_DEFINITIONS.".term_id ".
            "WHERE ".DICTIONARY_TABLE.".".$source." LIKE '%".$keyword."%' ".
            "ORDER BY ".DICTIONARY_TABLE.".theme_id, ".DICTIONARY_TABLE.".id");
    

while 循环是:

    while($row = $query->fetch(PDO::FETCH_ASSOC)) 

    $theme_name = "theme_".$lang;

    // add all results to an array
    $results[] = array(
        'english'           => $row['english'],
        'bulgarian'         => $row['bulgarian'],
        'english_abbr'      => $row['english_abbr'],
        'bulgarian_abbr'    => $row['bulgarian_abbr'],
        'theme'             => $row[$theme_name],
        'bulgarian_definition'  => $row['bulgarian_definition'],
        'english_definition'    => $row['english_definition']
    );  
// end while

上面的网站链接现在将加载升级后的搜索功能,其中将显示定义(如果存在)。对于任何好奇的人来说,一个示例词是“蠕虫”。

事实证明,有时解决问题所需要的只是向人们展示它然后开始思考它,因为有时解决方案就在您面前。感谢所有参与的人!

【问题讨论】:

请告诉我们您在 $row 数组中填写的代码 您是否执行过print_r($results) 以查看array_push() 是否有效,但只是将它们添加为新的数组元素,而不是将它们添加到您在循环中创建的$result[] 中。跨度> 是的,我做了一个print_r($results)array_push 没有添加新元素——因为english_definitionbulgarian_definition 都是NULL,我想。 $results[] = $row['english_definition'] 也没有。 【参考方案1】:

已编辑: 删除了以前的答案,因为它不正确。一旦我找到问题的解决方案,将更新此答案。没有删除这个答案,因为我没有找到这样做的选项。虽然有时占用空间并感觉自己很重要很有趣:),但不要讨厌玩家,讨厌游戏。


【讨论】:

查看$theme_query的查询,它是SELECT theme_id,".$theme_name." FROM ...。所以这些列不在$theme_row 中,而是在$row 中,这是外部循环变量。 @Sean 你是对的。我评论了他的问题以显示他填写 $row 数组的代码,现在考虑新信息,我将编辑我的答案。 在您的第二部分 - 如果您在 while($row = $query-&gt;fetch(PDO::FETCH_ASSOC)) 的每个循环上添加数组,$results[] = array( 是正确的。如果返回超过 1 行,则需要为 $result[],与 $result 一样,仅在每次循环迭代时它会覆盖先前的值,并且只会保存最后一个循环值。 @Sean 因此,如果我有一个名为 $result 的数组已经包含信息,并且我想向其中添加行,那么我可以使用 $result[] = array(...) 并且这会给我相同的结果为array_push($result, array(...))? 是的。我更喜欢 $result[] 而不是 array_push(),但他们基本上做同样的事情 - ***.com/questions/559844/…

以上是关于LEFT JOIN,如果存在第二个表的结果,则添加到 PHP 中的数组的主要内容,如果未能解决你的问题,请参考以下文章

SQL LEFT JOIN 从第二个表中排除两条记录

oracle SQL left join()或full out join()根据键排除记录

MySQL LEFT JOIN 在第二个表中具有可选值

LEFT JOIN 两个 MySql 表,而第二个表按 desc 排序并限制为 1

多表查询-inner join left join right joinfull join

sql语句中join、left join 、right join有啥区别?