如果愿意的话,我正在尝试通过构建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处理得比处理好枢轴表?
这让我加入了联盟,这就是我认为我正在寻求帮助的地方。我考虑过每天创建一个新专栏,但我认为这不是理想的。如果有必要,我愿意完全改变架构。
我希望我已经足够清楚了。任何有关正确方向的帮助或指示都将不胜感激。
谢谢!
答案 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>";