我有两个表,一个消息表和一个日志表。
我想返回消息表中已记录到用户且消息#大于传递参数的所有行。
我拥有的是:
SELECT * FROM message WHERE id IN ( SELECT msg_id AS id FROM msg_log WHERE user_num='1000' AND msg_id >='1660192')
此查询不返回任何内容,不会产生任何错误,只会超时...无论是从PHP脚本还是phpMyAdmin运行都无关紧要。事实上,如果从phpMyAdmin运行它甚至不显示查询正在运行,但它不允许您执行另一个查询。
如果我突破IN
选择
SELECT msg_id AS id FROM msg_log WHERE user_num='1000' AND msg_id >='1660192')
这将返回0条记录(应该如此)
如何解决此问题?
答案 0 :(得分:2)
试试这个:
SELECT message.* FROM message
INNER JOIN msg_log
ON message.id = msg_log.msg_id
AND msg_log.user_num = '1000'
AND msg_logmsg_id >= '1660192'
INNER JOIN
语法应该比IN
子句更快
当然,你可以加快查询
INDEX
或PRIMARY_KEY
INDEX
,可能与 message.id 有FOREIGN_KEY
答案 1 :(得分:0)
你是否尝试过没有php的mysql监视器执行此查询?您可能有大型表和mysql需要更多时间来执行查询,而不允许使用php脚本。你可以尝试:
1)在msg_log表中的user_num上添加索引,这可以加快查询速度,
2)在php_max_execution_time
中设置.htaccess
以允许php执行更长的脚本
答案 2 :(得分:0)
由于MySQL的特殊性,内连接比IN更快。 MySQL所做的是为它遇到的每一行运行子查询 。这对于数据库来说是不寻常的行为,但它是记录在案的。
联接可以解决此问题。但是,正确的等效语法需要不同的:
SELECT m.*
FROM message m join
(select distinct msg_id
from msg_log
WHERE user_num='1000' AND msg_id >='1660192'
) ml
on ml.msg_id = m.id
如果您想避免每条消息都有多行,那么distinct就很重要了。
您也可以使用EXISTS子句修复它:
SELECT *
FROM message m
WHERE EXISTS (SELECT 1
FROM msg_log ml
WHERE user_num='1000' AND msg_id >='1660192' and m.id = ml.msg_id
)
如果你喜欢连接上的相关子查询。