gpt4 book ai didi

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

转载 作者:可可西里 更新时间:2023-11-01 08:36:47 25 4
gpt4 key购买 nike

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

基本上每个用户都有多个目标,他们每天记录某个事件发生了多少次。例如,假设我的目标是每天只喝 1 杯咖啡。如果我今天有 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 作为行和日期范围作为列(如周 View 日历)。

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 处理比通过做一个更好数据透视表?

剩下的就是连接,我认为我正在寻求帮助。我考虑过每天创建一个新专栏,但我认为这并不理想。如有必要,我愿意完全改变架构。

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

谢谢!

最佳答案

如果我理解正确你的问题那么我建议你应该做一个常规选择,其中 goal_id 和 check_date 的每个组合都会在结果集中获得一条记录,然后在客户端你将为每个 check_date 创建一个列比如说有一个数组数组并在其中插入校验计数。

这至少应该比 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”(初始化为零以避免空问题),外键是 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 表格示例然后加入内部数组 和外部数组 .

如何创建和初始化“$array_of_arrays”数组的示例如下(假设您有一个包含所有目标的数组 $goals 和一个包含所有日期的数组 $dates,如果您不知道那么提前你可以通过执行 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>";

关于php - 使用连接消除循环中的多个 SELECT,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/9779792/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com