暂无图片
页面上的实时搜索速度较慢(MySQL InnoDB)
我来答
分享
库海无涯
2023-08-18
页面上的实时搜索速度较慢(MySQL InnoDB)

我的页面上有实时搜索。当向服务器发送请求时,接受过滤和排序参数(开发商、发行商、流派、时间等)。问题在于它的执行需要相当长的时间,并且由于用户输入的每个字符都会触发脚本,因此这不得不引起人们的注意。如果在搜索时省略其他参数并以开发人员为例,您将得到以下查询(执行时间约为 0.14 秒,但在表单中输入时,当然运行速度会更慢):

SELECT games.id AS id FROM games LEFT JOIN games_titles ON games.id=games_titles.game_id WHERE games_titles.lang=1 AND EXISTS(SELECT 1 FROM games_devs WHERE games_devs.game_id=games.id AND games_devs.dev LIKE ‘de%’) ORDER BY games.rating DESC LIMIT 6
解释:

±—±-------------±-------------±-------±-------------------±--------------±--------±-----------------±------±-----------------------------------+
| id | select_type | table | type | possible_keys | key | key_len | ref | rows | Extra |
±—±-------------±-------------±-------±-------------------±--------------±--------±-----------------±------±-----------------------------------+
| 1 | PRIMARY | games | index | PRIMARY | rating_index | 8 | NULL | 1 | Using index |
±—±-------------±-------------±-------±-------------------±--------------±--------±-----------------±------±-----------------------------------+
| 1 | PRIMARY | | eq_ref | distinct_key | distinct_key | 4 | func | 1 | Using where |
±—±-------------±-------------±-------±-------------------±--------------±--------±-----------------±------±-----------------------------------+
| 1 | PRIMARY | games_titles | ref | game_id_index,lang | game_id_index | 4 | test_db.games.id | 1 | Using index condition; Using where |
±—±-------------±-------------±-------±-------------------±--------------±--------±-----------------±------±-----------------------------------+
| 2 | MATERIALIZED | games_devs | range | PRIMARY,dev | dev | 452 | NULL | 49994 | Using where; Using index |
±—±-------------±-------------±-------±-------------------±--------------±--------±-----------------±------±-----------------------------------+
分析:

±—±-----------------------±-------+
| 1 | Starting | 103 µs |
±—±-----------------------±-------+
| 2 | Checking Permissions | 16 µs |
±—±-----------------------±-------+
| 3 | Opening Tables | 32 µs |
±—±-----------------------±-------+
| 4 | After Opening Tables | 11 µs |
±—±-----------------------±-------+
| 5 | System Lock | 11 µs |
±—±-----------------------±-------+
| 6 | Table Lock | 25 µs |
±—±-----------------------±-------+
| 7 | Init | 48 µs |
±—±-----------------------±-------+
| 8 | Optimizing | 50 µs |
±—±-----------------------±-------+
| 9 | Statistics | 145 µs |
±—±-----------------------±-------+
| 10 | Preparing | 92 µs |
±—±-----------------------±-------+
| 11 | Sorting Result | 23 µs |
±—±-----------------------±-------+
| 12 | Executing | 46 µs |
±—±-----------------------±-------+
| 13 | Query End | 14 µs |
±—±-----------------------±-------+
| 14 | Removing Tmp Table | 13 µs |
±—±-----------------------±-------+
| 15 | Query End | 9 µs |
±—±-----------------------±-------+
| 16 | Commit | 11 µs |
±—±-----------------------±-------+
| 17 | Closing Tables | 10 µs |
±—±-----------------------±-------+
| 18 | Unlocking Tables | 9 µs |
±—±-----------------------±-------+
| 19 | Closing Tables | 15 µs |
±—±-----------------------±-------+
| 20 | Starting Cleanup | 9 µs |
±—±-----------------------±-------+
| 21 | Freeing Items | 19 µs |
±—±-----------------------±-------+
| 22 | Updating Status | 73 µs |
±—±-----------------------±-------+
| 23 | Reset For Next Command | 19 µs |
±—±-----------------------±-------+
由于我在网站上有分页,因此我需要获取所有条目的数量,以便可以转到最后一页。但在这种情况下,性能受到的影响更大(在页面上搜索时需要 1.97 秒):

SELECT games.id AS id FROM games LEFT JOIN games_titles ON games.id=games_titles.game_id WHERE games_titles.lang=1 AND EXISTS(SELECT 1 FROM games_devs WHERE games_devs.game_id=games.id AND games_devs.dev LIKE ‘de%’) ORDER BY games.rating
解释:

±—±-------------±-------------±-------±-------------------±--------------±--------±-----------------±------±-----------------------------------+
| id | select_type | table | type | possible_keys | key | key_len | ref | rows | Extra |
±—±-------------±-------------±-------±-------------------±--------------±--------±-----------------±------±-----------------------------------+
| 1 | PRIMARY | games | index | PRIMARY | rating_index | 8 | NULL | 99889 | Using index |
±—±-------------±-------------±-------±-------------------±--------------±--------±-----------------±------±-----------------------------------+
| 1 | PRIMARY | | eq_ref | distinct_key | distinct_key | 4 | func | 1 | Using where |
±—±-------------±-------------±-------±-------------------±--------------±--------±-----------------±------±-----------------------------------+
| 1 | PRIMARY | games_titles | ref | game_id_index,lang | game_id_index | 4 | test_db.games.id | 1 | Using index condition; Using where |
±—±-------------±-------------±-------±-------------------±--------------±--------±-----------------±------±-----------------------------------+
| 2 | MATERIALIZED | games_devs | range | PRIMARY,dev | dev | 452 | NULL | 49994 | Using where; Using index |
±—±-------------±-------------±-------±-------------------±--------------±--------±-----------------±------±-----------------------------------+
分析:

±—±-----------------------±-------+
| 1 | Starting | 150 µs |
±—±-----------------------±-------+
| 2 | Checking Permissions | 19 µs |
±—±-----------------------±-------+
| 3 | Opening Tables | 30 µs |
±—±-----------------------±-------+
| 4 | After Opening Tables | 12 µs |
±—±-----------------------±-------+
| 5 | System Lock | 11 µs |
±—±-----------------------±-------+
| 6 | Table Lock | 13 µs |
±—±-----------------------±-------+
| 7 | Init | 57 µs |
±—±-----------------------±-------+
| 8 | Optimizing | 49 µs |
±—±-----------------------±-------+
| 9 | Statistics | 142 µs |
±—±-----------------------±-------+
| 10 | Preparing | 93 µs |
±—±-----------------------±-------+
| 11 | Sorting Result | 22 µs |
±—±-----------------------±-------+
| 12 | Executing | 21 µs |
±—±-----------------------±-------+
| 13 | Sending Data | 182 ms |
±—±-----------------------±-------+
| 14 | End Of Update Loop | 28 µs |
±—±-----------------------±-------+
| 15 | Removing Tmp Table | 790 µs |
±—±-----------------------±-------+
| 16 | End Of Update Loop | 15 µs |
±—±-----------------------±-------+
| 17 | Query End | 9 µs |
±—±-----------------------±-------+
| 18 | Commit | 11 µs |
±—±-----------------------±-------+
| 19 | Closing Tables | 10 µs |
±—±-----------------------±-------+
| 20 | Unlocking Tables | 9 µs |
±—±-----------------------±-------+
| 21 | Closing Tables | 17 µs |
±—±-----------------------±-------+
| 22 | Starting Cleanup | 9 µs |
±—±-----------------------±-------+
| 23 | Freeing Items | 19 µs |
±—±-----------------------±-------+
| 24 | Updating Status | 75 µs |
±—±-----------------------±-------+
| 25 | Reset For Next Command | 18 µs |
±—±-----------------------±-------+
CREATE TABLE games (
id int(10) unsigned NOT NULL AUTO_INCREMENT,
rating double NOT NULL DEFAULT 0,
date date DEFAULT NULL,
date_type int(1) NOT NULL DEFAULT 1,
img varchar(500) DEFAULT NULL,
img_type int(10) NOT NULL,
url varchar(100) NOT NULL,
PRIMARY KEY (id),
KEY rating_index (rating) USING BTREE,
KEY date_index (date) USING BTREE
) ENGINE=InnoDB AUTO_INCREMENT=326678 DEFAULT CHARSET=utf8 COLLATE=utf8_general_ci

CREATE TABLE games_titles (
game_id int(50) NOT NULL,
title varchar(150) DEFAULT NULL,
lang int(10) NOT NULL DEFAULT 1,
KEY game_id_index (game_id) USING BTREE,
KEY lang (lang,title)
) ENGINE=InnoDB DEFAULT CHARSET=utf8 COLLATE=utf8_general_ci

CREATE TABLE games_devs (
id int(50) NOT NULL AUTO_INCREMENT,
game_id int(50) NOT NULL,
dev varchar(150) NOT NULL,
PRIMARY KEY (id),
KEY game_id_index (game_id) USING BTREE
) ENGINE=InnoDB AUTO_INCREMENT=126699 DEFAULT CHARSET=utf8 COLLATE=utf8_general_ci
该请求还会按语言进行检查,因为该网站是多语言的。但我决定省略这些细节,因为这里的主要负载仍然来自搜索本身。我还没有将索引放在“dev”字段上,其余的一切都已设置完毕。您可以看到上面表格的结构。

您能告诉我如何提高搜索性能或者您可以提供什么替代 LIKE 运算符吗?我听说很多人不建议使用它。现在在“dev”表中,条目的形式为“dev_1”、“dev_2”等。如果在形式中输入字母“p”,则脚本将被触发约10毫秒,其中字母“d”当然,绕过所有选项需要更长的时间。当然,我知道在实际制作中写作会有更多变化,但这里最好的解决方案是什么?分页时,每增加一页,搜索时间就会增加(逻辑上,因为MySQL算法需要生成一个新的数据集,但是从初始位置做一个缩进),恐怕会花很长时间,

我来答
添加附件
收藏
分享
问题补充
1条回答
默认
最新
redgame
把sql改写下,子查询提上去
暂无图片 评论
暂无图片 有用 0
打赏 0
暂无图片
回答交流
Markdown


请输入正文
提交
相关推荐
MySQL的存储过程与函数的联系与区别是怎样的?
回答 1
已采纳
存储过程是用户定义的一系列SQL语句的集合,用户可以调用存储过程,而函数通常是数据库已定义的方法,它接收参数并返回某种类型的值并且不涉及特定用户表。函数是为其他程序服务的,需要在其他语句中调用函数才可
MySQL CPU占用率100%
回答 5
已采纳
楼上得迷路了把???
华为云裸金属 部署11g RAC,共享盘为一块独立的高性能磁盘,划分三个分区用于ASM(OCR/DATA/FRA),请问是否可行?
回答 1
为什么不弄3块盘呢?如果磁盘不够用的话,我觉得adg也是个不错的选择
同城机房A到B的同步如何做?
回答 1
如果你是需要两个机房之间的数据相互同步的话,就A和B机房的主做个双向同步
关系型数据库,如MySQL,Oracle,SQL server等等,未来发展还有哪些功能是值得期待的?
回答 3
融合NOSQL,MLAI
有什么mysql合适的增量数据同步工具
回答 1
已采纳
XtraBackup开源免费的;
怎么样使用提示 让MYSQL 8.0 使用嵌套循环?
回答 2
每个版本的hint不一样8.0.18,8.0.19,及以后版本。在这块mysql改动较大。8.0.19及以后,可以用/NOBNL()/就可以实现/NOHASHJOIN/这个hint只有在8.0.18版
某库夜里发生了锁超时,开发需要定位相关的sql语句,mysql中如下处理这种case?
回答 2
获取超时的历史语句比较简单!1、开启performanceschema的eventsstatements开关UPDATEperformanceschema.setupconsumersSETENABL
rpm包安装的mysql,安装目录在哪里
回答 2
已采纳
(CentOSRPM安装MySQL)find/namemysqlprint/etc/logrotate.d/mysql/etc/rc.d/init.d/mysql/var/lib/mysql/var/
怎么查 MySQL 某个字段的创建时间的语句?
回答 2
没有办法,mysql本身是没有统计字段创建、修改时间的。只有通过日志或自己写触发器或中间件实现
问题信息
请登录之后查看
邀请回答
暂无人订阅该标签,敬请期待~~