0

通常对我来说显而易见的答案是“几个连接”。但情况是这样的:我有一个包含几千个条目的表,tableA。我有另一个包含数千个条目的表,tableB。每个条目都有一个与 tableA 中行的 id 相关的数字。tableB 中将有多行与 tableA 中的单行一起使用。将有第三个表 tableC,其中包含数万个条目。它将有多行对应于 tableB 中的单行。

查询可能看起来像这样:(细节并不重要)

SELECT tableA.*, tableB.*, tableC.* LEFT JOIN tableB ON tableB.idA = tableA.id LEFT JOIN tableC ON tableC.idB = tableB.id WHERE tableA.id = some_number;

这工作得很好而且很花哨,但是由于加入,我得到了大量的重复行。我想要 tableA 的一个结果,tableB 的一些结果和 tableC 的几个结果。

不幸的是,为此我使用了许多嵌套的 PHP 数组和 for 循环。它变得很乱,所以我想知道我是否应该只使用 3 个单独的查询来摆脱所有嵌套的 PHP 循环,还是应该坚持使用 JOINS?

附加信息: “tableA”:工作 ID(主键)名称描述日期时间客户状态

“tableB”:工作 ID(主键)工作(与工作中的行 ID 相关)摘要描述日期

“tableC”:时间 id(主键)工作(与工作中的行 id 相关)开始结束

期望的输出:我不太确定哪种格式最好。我想要一个工作行,有一份工作清单,还有一份工作时间清单。

嵌套循环示例: $work 是一个包含两个值的数组。第一个是作品数组,第二个是时间数组。

foreach ($work[0] as $workk=>$workv) {
    echo("<div><h1>" . $workk . ": " . $workv["summary"] . "</h1><br/>Job: " . $workv["job"] . "</br>Date: " . $workv["date"] . "<br>Description:" . $workv["description"] . "<br/>Times:<br/>");
    foreach ($work[1] as $workid=>$times) {
        if ($workid == $workk) {
            foreach ($times as $time) {
                echo("Start: " . $time[0] . " | End: " . $time[1]);
            }
            break;
        }
    }
}

我在函数中使用此代码获取“工作”数组(其中 $query->result() 是长 JOIN 查询中的一行)

foreach ($query->result() as $row) {
                if (!array_key_exists($row->id, $work)) {
                    $work[$row->id] = array("job"=>$row->job, "summary"=>$row->summary, "description"=>$row->description, "date"=>$row->date);
                }
                $times[$row->id][] = array($row->start, $row->end);
            }
            return array($work, $times);
4

1 回答 1

1

您有两种查询数据库的方法:

  1. 使用两个左连接获取所有数据。
  2. 获取主数据并按需加载详细信息。

*不要分别加载所有三个表,然后尝试在 php 中链接它们

第一种方法一开始可能需要更长的时间来加载,但随着数据的加载,将不再有数据库调用。另一方面,尽管将数据显示为表格(具有重复的主数据)可能很容易,但这通常不是所需的 UI。除非您使用特殊的 UI 组件来自动处理主详细信息,否则您可能难以将表格转换为更合理的形式(在这种情况下是您的循环)。

您的问题实际上是一个 UI 设计问题,即如何显示两个级别的主细节。有许多 UI 模式可以显示主细节。

例如,您只能在下拉列表中加载第一级(表 A)并查询数据库并在下拉更改时显示第二级。或者你可以有一个树状结构,在折叠时查询数据库。

看看这个链接,以获得一些想法。您还可以尝试搜索 php 的主从 UI 组件。

于 2013-04-25T02:23:54.377 回答