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

时间:2016-04-22 04:08:52

标签: php mysql arrays left-join

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

我从以下两个表中获取数据:

    $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...和“英语”以及“保加利亚语” - 当我在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']);
        }

不知道为什么会这样,会继续寻找。

SECOND EDIT:决定将两个定义直接放在$ 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' => null,如果存在定义,它就在那里。到目前为止一切都很好。

新问题是我无法再按主题对结果进行分组 - 显示最后找到的结果的主题。这完全是另一个问题。真正令我烦恼的是,在我添加定义之前,一切都运行得很好。您可以使用工作网站here

问题解决了!

  1. 决定不将值推送到数组(如上所示)。
  2. 摆脱了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");
    
  3. 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
    

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

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

1 个答案:

答案 0 :(得分:0)

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