[英]MYSQL Slow query, how to optimize?
我有一个很大的问题,我对SQL不是很好...我在mysql中有一个数据库,当我执行此查询时,我的响应时间大约为0.2秒,因此当我将其称为用户列表时(在servlet中)响应时间大约几秒钟。
SELECT visible,nlikes,nomecognome,profile_img,users_face.id
FROM users_face
LEFT OUTER JOIN `likes_face`
on (users_face.fb_id = likes_face.fb_id)
WHERE users_face.fb_id =? and users_face.token_valid=1
ORDER BY date DESC limit 1
有没有办法优化此代码或任何好的资源来研究查询的优化?
ArrayList<SocialMan> mans = new ArrayList<>();
PreparedStatement ps;
int nlikes, userid;
String nomeCogn, prof;
boolean visible;
FacebookClient facebookClient = new DefaultFacebookClient(token, Version.VERSION_2_6);
com.restfb.Connection<User> myFriends;
myFriends = facebookClient.fetchConnection("me/friends",User.class, Parameter.with("limit", 999));
for (User u : myFriends.getData()) {
nlikes = -1;
userid = -1;
nomeCogn = "ERROR";
prof = "ERROR";
visible = false;
try {
ps = con.prepareStatement("SELECT visible,nlikes,nomecognome,profile_img,users_face.id FROM users_face LEFT OUTER JOIN `likes_face` on (users_face.fb_id = likes_face.fb_id) WHERE users_face.fb_id =? and users_face.token_valid=1 ORDER BY date DESC limit 1");
ps.setString(1, "" + u.getId());
ResultSet rs = ps.executeQuery();
while (rs.next()) {
nlikes = rs.getInt("nlikes");
userid = rs.getInt("id");
nomeCogn = rs.getString("nomecognome");
prof = rs.getString("profile_img");
visible = rs.getBoolean("visible");
}
} catch (SQLException ex) {
Logger.getLogger(FaceLikes.class.getName()).log(Level.SEVERE, null, ex);
}
// System.out.println("NOMECOGNOME: "+nomeCogn);
if (userid != -1 && visible) {
mans.add(new SocialMan(nomeCogn, userid, prof, nlikes));
}
}
nlikes = -1;
userid = -1;
nomeCogn = "ERROR";
prof = "ERROR";
CREATE TABLE IF NOT EXISTS `users_face` (
`id` int(11) NOT NULL AUTO_INCREMENT,
`fb_id` varchar(45) NOT NULL,
`fb_token` varchar(300) NOT NULL,
`nomecognome` varchar(100) NOT NULL,
`data_iscrizione` timestamp NULL DEFAULT CURRENT_TIMESTAMP,
`profile_img` varchar(255) NOT NULL,
`visible` int(11) NOT NULL DEFAULT '1',
`TOKEN` varchar(255) NOT NULL,
`locale` varchar(255) NOT NULL,
`token_valid` tinyint(1) NOT NULL,
PRIMARY KEY (`id`,`fb_id`),
UNIQUE KEY `fb_id` (`fb_id`),
UNIQUE KEY `id` (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8 AUTO_INCREMENT=173 ;
CREATE TABLE IF NOT EXISTS `likes_face` (
`id` int(11) NOT NULL AUTO_INCREMENT,
`date` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP,
`nlikes` int(11) NOT NULL,
`fb_id` varchar(255) NOT NULL,
PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8 AUTO_INCREMENT=1182636 ;
(“在开始采取行动之前阅读整个答案。”)
这根本不符合逻辑:
PRIMARY KEY (`id`,`fb_id`),
UNIQUE KEY `fb_id` (`fb_id`),
UNIQUE KEY `id` (`id`)
暂时改为
PRIMARY KEY(fb_id),
INDEX(id)
假设您通常通过fb_id查找记录。
如果你说“用户列表”。 你是说WHERE fb_id IN ( ... )
? 可能不是因为你有LIMIT 1
。 或者你的意思是你反复调用那个SELECT
?
插科打诨! likes_face
有一个你没有使用的id,加上你正在使用但没有编入索引的 fb_id
。 对两个表使用相同的 id:要么删除AUTO_INCREMENT
并更改likes_face
代码逻辑,要么删除id
并使fb_id
为两个表的PK。
这些表是1:1吗?
出了点问题。 为什么likes_face have
AUTO_INCREMENT = 1182636 , while there seem to be only 173 users? If the tables are 1:1, are you using
, while there seem to be only 173 users? If the tables are 1:1, are you using
REPLACE`吗? 你将用尽ids!
重新开始......
如果表是1:1:
id
在两个表上都有PRIMARY KEY(fb_id)
, CHARACTER SET ascii
如果Facebook将id限制为ascii, 如果likes_face
是每天喜欢的数量:
id
, CHARACTER SET ascii
如果Facebook将id限制为ascii, likes_face
PRIMARY KEY(fb_id, date)
。 如果表是1:许多以其他方式,我无法理解架构。
这就是我能想到的:
ALTER TABLE users_face ADD INDEX (fb_id, token_valid);
ALTER TABLE likes_face ADD INDEX (fb_id, nlikes, date);
您无法优化排序,因为date
列位于非主表中。
如果要进一步优化,接下来要做的是将nlikes
和date
列移动到users_face表中,然后按以下顺序在该表上创建索引: (fb_id, token_valid, date, nlikes)
。
然后,您可以跳过查询中的连接,排序顺序将是索引的顺序,因此将进行优化。
在那之后(或者可能在非规范化之前 ),你应该考虑缓存数据。
您可以使用MYSQL语句的EXPLAIN。 使用EXPLAIN优化查询
有人建议创建索引。 创建索引是一个不错的选择,但如果要经常修改该列,请避免使用索引。
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.