MySQL中Join算法实现原理分析

在MySQL 中,只有一种 Join 算法,就是大名鼎鼎的 Nested Loop Join,他没有其他很多数据库所提供的 Hash Join,也没有 Sort Merge Join。顾名思义,Nested Loop Join 实际上就是通过 驱动表的结果集作为循环基础数据,然后一条一条的通过该结果集中的数据作为过滤条件到下一个表中查询数据,然后合并结果。如果还有第三个参与 Join,则再通过前两个表的 Join 结果集作为循环基础数据,再一次通过循环查询条件到第三个表中查询数据,如此往复。

  还是通过示例和图解来说明吧,后面将通过我个人数据库测试环境中的一个 example(自行设计,非MySQL 自己提供) 数据库中的三个表的 Join 查询来进行示例。

  注意:由于这里有些内容需要在MySQL 5.1.18之后的版本中才会体现出来,所以本测试的MySQL 版本为5.1.26

  表结构:

1 sky @localhost : example 11 : 09 : 32 > show create table user_group\G
2
3    *************************** 1 . row ***************************
4
5    Table : user_group
6
7    Create Table : CREATE TABLE `user_group` (
8
9   ` user_id ` int ( 11 ) NOT NULL ,
10
11   `group_id` int ( 11 ) NOT NULL ,
12
13   `user_type` int ( 11 ) NOT NULL ,
14
15   `gmt_create` datetime NOT NULL ,
16
17   `gmt_modified` datetime NOT NULL ,
18
19   `status` varchar ( 16 ) NOT NULL ,
20
21    KEY `idx_user_group_uid` (` user_id `)
22
23   ) ENGINE = MyISAM DEFAULT CHARSET = utf8
24
25    1 row in set ( 0.00 sec)
26
27   sky @localhost : example 11 : 10 : 32 > show create table group_message\G
28
29    *************************** 1 . row ***************************
30
31    Table : group_message
32
33    Create Table : CREATE TABLE `group_message` (
34
35   `id` int ( 11 ) NOT NULL AUTO_INCREMENT,
36
37   `gmt_create` datetime NOT NULL ,
38
39   `gmt_modified` datetime NOT NULL ,
40
41   `group_id` int ( 11 ) NOT NULL ,
42
43   ` user_id ` int ( 11 ) NOT NULL ,
44
45   `author` varchar ( 32 ) NOT NULL ,
46
47   `subject` varchar ( 128 ) NOT NULL ,
48
49    PRIMARY KEY (`id`),
50
51    KEY `idx_group_message_author_subject` (`author`,`subject`( 16 )),
52
53    KEY `idx_group_message_author` (`author`),
54
55    KEY `idx_group_message_gid_uid` (`group_id`,` user_id `)
56
57   ) ENGINE = MyISAM AUTO_INCREMENT = 97 DEFAULT CHARSET = utf8
58
59    1 row in set ( 0.00 sec)
60
61   sky @localhost : example 11 : 10 : 43 > show create table group_message_content\G
62
63    *************************** 1 . row ***************************
64
65    Table : group_message_content
66
67    Create Table : CREATE TABLE `group_message_content` (
68
69   `group_msg_id` int ( 11 ) NOT NULL ,
70
71   `content` text NOT NULL ,
72
73    KEY `group_message_content_msg_id` (`group_msg_id`)
74
75   ) ENGINE = MyISAM DEFAULT CHARSET = utf8
76
77    1 row in set ( 0.00 sec)

  使用Query如下:

1    select m.subject msg_subject, c.content msg_content
2
3    from user_group g,group_message m,group_message_content c
4
5    where g. user_id = 1
6
7    and m.group_id = g.group_id
8
9    and c.group_msg_id = m.id

  看看我们的 Query 的执行计划:

1   sky @localhost : example 11 : 17 : 04 > explain select m.subject msg_subject, c.content msg_content
2
3    -> from user_group g,group_message m,group_message_content c
4
5    -> where g. user_id = 1
6
7    -> and m.group_id = g.group_id
8
9    -> and c.group_msg_id = m.id\G
10
11    *************************** 1 . row ***************************
12
13   id: 1
14
15   select_type: SIMPLE
16
17    table : g
18
19   type: ref
20
21   possible_keys: user_group_gid_ind,user_group_uid_ind,user_group_gid_uid_ind
22
23    key : user_group_uid_ind
24
25   key_len: 4
26
27   ref: const
28
29   rows: 2
30
31   Extra:
32
33    *************************** 2 . row ***************************
34
35   id: 1
36
37   select_type: SIMPLE
38
39    table : m
40
41   type: ref
42
43   possible_keys: PRIMARY ,idx_group_message_gid_uid
44
45    key : idx_group_message_gid_uid
46
47   key_len: 4
48
49   ref: example.g.group_id
50
51   rows: 3
52
53   Extra:
54
55    *************************** 3 . row ***************************
56
57   id: 1
58
59   select_type: SIMPLE
60
61    table : c
62
63   type: ref
64
65   possible_keys: idx_group_message_content_msg_id
66
67    key : idx_group_message_content_msg_id
68
69   key_len: 4
70
71   ref: example.m.id
72
73   rows: 2
74
75   Extra:

  我们可以看出,MySQL Query Optimizer 选择了 user_group 作为驱动表,首先利用我们传入的条件 user_id 通过 该表上面的索引 user_group_uid_ind 来进行 const 条件的索引 ref 查找,然后以 user_group 表中过滤出来的结果集的 group_id 字段作为查询条件,对 group_message 循环查询,然后再通过 user_group 和 group_message 两个表的结果集中的 group_message 的 id 作为条件 与 group_message_content 的 group_msg_id 比较进行循环查询,才得到最终的结果。没啥特别的,后一个引用前一个的结果集作为条件,实现过程可以通过下图表示:


下面的我们调整一下 group_message_content 去掉上面的 idx_group_message_content_msg_id 这个索引,然后再看看会是什么效果:

1   sky @localhost : example 11 : 25 : 36 > drop index idx_group_message_content_msg_id on group_message_content;
2
3   Query OK, 96 rows affected ( 0.11 sec)
4
5   sky @localhost : example 10 : 21 : 06 > explain
6
7    -> select m.subject msg_subject, c.content msg_content
8
9    -> from user_group g,group_message m,group_message_content c
10
11    -> where g. user_id = 1
12
13    -> and m.group_id = g.group_id
14
15    -> and c.group_msg_id = m.id\G
16
17    *************************** 1 . row ***************************
18
19   id: 1
20
21   select_type: SIMPLE
22
23    table : g
24
25   type: ref
26
27   possible_keys: idx_user_group_uid
28
29    key : idx_user_group_uid
30
31   key_len: 4
32
33   ref: const
34
35   rows: 2
36
37   Extra:
38
39    *************************** 2 . row ***************************
40
41   id: 1
42
43   select_type: SIMPLE
44
45    table : m
46
47   type: ref
48
49   possible_keys: PRIMARY ,idx_group_message_gid_uid
50
51    key : idx_group_message_gid_uid
52
53   key_len: 4
54
55   ref: example.g.group_id
56
57   rows: 3
58
59   Extra:
60
61    *************************** 3 . row ***************************
62
63   id: 1
64
65   select_type: SIMPLE
66
67    table : c
68
69   type: ALL
70
71   possible_keys: NULL
72
73    key : NULL
74
75   key_len: NULL
76
77   ref: NULL
78
79   rows: 96
80
81   Extra: Using where ; Using join buffer

  我们看到不仅仅 group_message_content 表的访问从 ref 变成了 ALL,此外,在最后一行的 Extra信息从没有任何内容变成为 Using where; Using join buffer,也就是说,对于从 ref 变成 ALL 很容易理解,没有可以使用的索引的索引了嘛,当然得进行全表扫描了,Using where 也是因为变成全表扫描之后,我们需要取得的 content 字段只能通过对表中的数据进行 where 过滤才能取得,但是后面出现的 Using join buffer 是一个啥呢?

  我们知道,MySQL 中有一个供我们设置的参数 join_buffer_size ,这里实际上就是使用到了通过该参数所设置的 Buffer 区域。那为啥之前的执行计划中没有用到呢?

  实际上,Join Buffer 只有当我们的 Join 类型为 ALL(如示例中),index,rang 或者是 index_merge 的时候 才能够使用,所以,在我们去掉 group_message_content 表的 group_msg_id 字段的索引之前,由于 Join 是 ref 类型的,所以我们的执行计划中并没有看到有使用 Join Buffer。

  当我们使用了 Join Buffer 之后,我们可以通过下面的这张图片来表示 Join 完成过程:


  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
基于Tkinter和MySQL开发登录界面系统的算法实现和分析可以概括如下: 1. 连接到MySQL数据库:使用Python的MySQL驱动程序连接到MySQL数据库,可以使用pymysql库等来实现。 2. 处理登录事件:在登录按钮的事件处理程序,获取用户输入的用户名和密码,并将其与MySQL数据库的数据进行比较。可以使用SQL语句查询数据库是否存在相应的用户名和密码,如果存在,则登录成功,否则登录失败。 3. 显示结果:根据查询结果显示相应的消息框,提示用户登录是否成功。 4. 关闭数据库连接:在程序结束时,关闭MySQL数据库连接。 下面是一个基于Tkinter和MySQL的登录界面系统算法实现的示例代码: ```python import pymysql import tkinter as tk from tkinter import messagebox # 连接到MySQL数据库 conn = pymysql.connect(host='localhost', user='root', password='', db='test') cursor = conn.cursor() # 创建登录界面 root = tk.Tk() root.title('登录界面') root.geometry('300x200') username_label = tk.Label(root, text='用户名:') username_label.pack() username_entry = tk.Entry(root) username_entry.pack() password_label = tk.Label(root, text='密码:') password_label.pack() password_entry = tk.Entry(root, show='*') password_entry.pack() # 处理登录按钮的事件 def login(): username = username_entry.get() password = password_entry.get() sql = "SELECT * FROM users WHERE username='%s' AND password='%s'" % (username, password) cursor.execute(sql) result = cursor.fetchone() if result: messagebox.showinfo('登录成功', '欢迎您,%s!' % username) else: messagebox.showerror('登录失败', '用户名或密码错误!') # 创建登录按钮 login_button = tk.Button(root, text='登录', command=login) login_button.pack() # 运行主循环 root.mainloop() # 关闭MySQL数据库连接 cursor.close() conn.close() ``` 此示例代码,程序首先连接到MySQL数据库,然后创建登录界面,包括用户名和密码输入框、登录按钮等组件。在登录按钮的事件处理程序,程序使用SQL语句查询数据库是否存在相应的用户名和密码,如果存在,则显示“登录成功”消息框,否则显示“登录失败”消息框。最后,程序关闭MySQL数据库连接。 需要注意的是,在实现时需要注意安全问题,比如输入数据的验证和过滤,以防止SQL注入等安全问题。此外,还需要对密码进行加密,以保护用户隐私和安全。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值