使用连接消除循环中的多个SELECT

时间:2012-03-20 00:11:39

标签: php mysql database-design join

如果愿意的话,我正在尝试通过构建joe's goals克隆来自学PHP / mysql。

基本上每个用户都有多个目标,每天他们都会记录某个事件发生的次数。例如,说我的目标是每天只喝一杯咖啡。如果我今天有3个(哎呀!),今天我会记录3个“支票”。我使用一个名为“支票”的表来保存每天的支票计数。

我有以下表格和示例插页:

CREATE TABLE `users` (
  `user_id` int(5) NOT NULL AUTO_INCREMENT,
  `user_email` varchar(50) NOT NULL,
  `user_name` varchar(25) NOT NULL,
  PRIMARY KEY (`user_id`)
) ENGINE=MyISAM DEFAULT CHARSET=latin1 ;

-- Dumping data for table `users`
INSERT INTO `users` VALUES (1, 'xxx@xxx.com', 'xxx');
INSERT INTO `users` VALUES (2, 'some@guy.com', 'SomeGuy');

CREATE TABLE `goal_settings` (
  `goal_id` int(5) NOT NULL AUTO_INCREMENT,
  `user_id` int(5) NOT NULL,
  `goal_description` varchar(100) NOT NULL,
  PRIMARY KEY (`goal_id`),
  KEY `user_id` (`user_id`)
) ENGINE=MyISAM DEFAULT CHARSET=latin1;

-- Dumping data for table `goal_settings`
INSERT INTO `goal_settings` VALUES (1, 1, 'Run 1k');
INSERT INTO `goal_settings` VALUES (2, 1, 'Read 20 pages');
INSERT INTO `goal_settings` VALUES (3, 2, 'Cups of Coffee');

CREATE TABLE `checks` (
  `check_id` int(40) NOT NULL AUTO_INCREMENT,
  `goal_id` int(5) NOT NULL,
  `check_date` date NOT NULL,
  `check_count` int(3) NOT NULL,
  PRIMARY KEY (`check_id`)
) ENGINE=MyISAM DEFAULT CHARSET=latin1;

-- Dumping data for table `checks`
INSERT INTO `checks` VALUES (6, 1, '2012-03-02', 3);
INSERT INTO `checks` VALUES (2, 1, '2012-03-01', 2);
INSERT INTO `checks` VALUES (3, 2, '2012-03-01', 1);
INSERT INTO `checks` VALUES (5, 1, '2012-02-29', 1);

我想要的输出将goal_ids作为行,将日期范围作为列(如周视图日历)。

goal_id   |  2012-03-01 | 2012-03-02 | 2012-03-03 | ... 2012-03-08 |
--------------------------------------------------------------------
1            2             3           0            ... 0
2            1             0           0            ... 0   

请注意,如果某一天没有针对给定目标的检查,则返回0而不是NULL。

我能够使用PHP让它工作得很糟糕。截断的代码,但我希望它基本上显示我尝试的内容:[$ goal_ids是一个包含与用户相关的所有目标的数组。 $ num_days是要显示的天数(即列数),$ goal_days是用于保存我们希望获取信息的日期的数组。

$mysqli = new mysqli('xxx','xxx','xxx','goals');
$stmt = $mysqli->stmt_init();
$stmt = $mysqli->prepare("SELECT checks.check_count AS check_count 
FROM `checks` WHERE goal_id = ? AND check_date = ?");

for($i=0; $i<=$goal_count - 1; $i++){  
  echo '<tr id="'.$goalid.'">';
      for($j=0; $j <=$num_days; $j++){
          $checkdate = $goal_days[$j];
          $goalid = (integer) $goal_ids[$i];
            if (!$stmt->bind_param("ii", $goalid, $checkdate)) {
              echo "Binding parameters failed: (" . $stmt->errno . ") " . $stmt->error;
            }
            if (!$stmt->execute()) {
              echo "Execute failed: (" . $stmt->errno . ") " . $stmt->error;
              }
              $stmt->bind_result($check_count);
                if($stmt->fetch()){
              echo "<td>".$check_count."</td>";
             }
             else{
              echo '<td>0</td>';
             }
          }
   echo "</tr>";
} 
echo "</table>";
$stmt->close();

这显然效率低下,因为对于m个目标和n天,它会生成m x n个选择语句。

从阅读开始,我似乎基本上都在尝试制作一个数据透视表,但我也知道它们效率也很低,而且我猜我正在做的事情是由PHP处理得比处理好枢轴表?

这让我加入了联盟,这就是我认为我正在寻求帮助的地方。我考虑过每天创建一个新专栏,但我认为这不是理想的。如果有必要,我愿意完全改变架构。

我希望我已经足够清楚了。任何有关正确方向的帮助或指示都将不胜感激。

谢谢!

2 个答案:

答案 0 :(得分:1)

考虑添加一个天数表(或在运行时创建临时表),只保留您需要的连续日期或日期。然后,您可以使用单个查询获得一个很好的检查计数列表:

SELECT g.goal_id, d.day, COALESCE(c.check_count,0) as check_count 
FROM
  goal_settings g 
JOIN 
  days d
LEFT JOIN 
  checks c 
ON c.goal_id = g.goal_id AND c.check_date = d.day
WHERE 
  g.user_id = 1 
  AND d.day BETWEEN '2012-03-01' AND '2012-03-03'
ORDER BY g.goal_id, d.day

产生如下行集:

goal_id  | day        | check_count
    1    | 2012-03-01 |      2
    1    | 2012-03-02 |      3
    1    | 2012-03-03 |      0
    2    | 2012-03-01 |      1
    2    | 2012-03-02 |      0
    2    | 2012-03-03 |      0

然后使用php在循环中获取这些行以构建一个漂亮的html表 - 如果更改了goal_id,则打印新行,依此类推。

答案 1 :(得分:1)

如果我理解你的问题,那么我建议你应该进行常规选择,其中goal_id和check_date的每个组合将在结果集中获得记录,然后在客户端,你将为每个check_date创建一个列通过拥有一个数组数组并在其中插入checkcount来说。

这应该至少比使用m x n select语句更快。

为了提高效率,你的sql可以通过goal_id和check_date对它进行排序,这将导致记录被组合在一起。

以下是sql语句的示例:

SELECT check_date, goal_id, check_count FROM checks ORDER BY goal_id, check_date 

这是PHP示例代码,假设您有一个数组数组“$ array_of_arrays”(初始化为零以避免null问题),外键是goal_id,内键是check_date:

while ($row = mysqli_fetch_result($result)){
     $row_goal_id = $row["goal_id"];
     $row_check_date = $row["check_date"];
     $array_of_arrays[$row_goal_id][$row_check_date] = $row["check_count"];
}

然后您可以使用数组数组来执行您喜欢的操作,例如,如果您希望输出为HTML表示例,则使用&lt; / td&gt;&lt; td&gt;加入内部数组。和外部数组&lt; / td&gt;&lt; / tr&gt;&lt; tr&gt;&lt; td&gt;。

如何创建和初始化“$ array_of_arrays”数组的示例如下(假设您有一个包含所有目标的数组$目标和包含所有日期的数组$日期,如果您不知道那么事先你可以通过做一个SELECT DISTINCT)

从检查表中获取它们
 $array_of_arrays = array();   
 foreach ($goals as $key=>$value){
         $array_of_arrays[$value] = array();
         foreach ($checks as $key1=>$value1){
                  $array_of_arrays[$value][$value1] = 0;
         }
   }

可以使用类似的方法生成最终的HTML表格,如下所示:

$final_array = array();
foreach ($array_of_arrays as $key=>$value){ 
    $final_array[$key] = implode("</td><td>", $value);
}
$final_str = implode("</td></tr><tr><td>", $final_array);
$table_str = "<table><tr><td>" . $final_str . "</td></tr></table>";