多表查询 : 连接查询-子查询

MySQL基础操作链接 ; 工具: SQLyog

MySQL语法顺序:

  1. select[distinct]  
  2. from  
  3. join(left join/right join)  
  4. on  
  5. where  
  6. group by  
  7. having   
  8. union  
  9. order by  
  10. limit  

1. 表结构

      用户,角色,权限三张表(主表)及三者之间的关系通过两张 “第三张外键表”维护。“外键表”中的两个字段分别使用外键指向主表的主键。(一个用户可以有多个角色,一个角色可以有多个权限;正常来看是是一对多的关系,但是反过来 某个权限可以有多个角色拥有。 所以三者关系必须理解为多对多的关系,所以需要 “第三张外键表”去维护两张表之间的关系,同时保证实体表与实体表之间互相独立)。

表名:


用户信息表 user

 

用户角色表 user_role

字段

类型

描述

字段

类型

描述

id

int(11) NOT NULL

主键,自增

uid

int(11) NOT NULL

外键:user id

name

varchar(255) NOT NULL

名称

rid

int(11) NOT NULL

外键:role id

password

varchar(255) NOT NULL

密码

 

 

 

age

int(11) NOT NULL

年龄

 

 

 

gender

char(1) NOT NULL

性别

 

 

 

角色信息表 role

 

角色权限表 role_permisson

字段

类型

描述

 

字段

类型

描述

id

int(11) NOT NULL

主键,自增

rid

int(11) NOT NULL

外键:role id

rolename

varchar(255) NOT NULL

名称

pid

int(11) NOT NULL

外键:permission id

权限信息表 permission

 

 

 

字段

类型

描述

 

 

 

id

int(11) NOT NULL

 

<p style="margin-left:0pt;">主键,自增</p>
		</td>
		<td style="width:104px;"> </td>
		<td style="width:144px;"> </td>
		<td style="width:143px;"> </td>
	</tr><tr><td style="width:141.95pt;">
		<p style="margin-left:0pt;">name</p>
		</td>
		<td style="width:141.95pt;">
		<p style="margin-left:0pt;">varchar(255) NOT NULL</p>
		</td>
		<td style="width:141.9pt;">
		<p style="margin-left:0pt;">权限</p>
		</td>
		<td style="width:104px;"> </td>
		<td style="width:144px;"> </td>
		<td style="width:143px;"> </td>
	</tr></tbody></table></div><hr><h2><a name="t3"></a><a name="t3"></a><strong><strong><strong>具体建表语句:</strong></strong></strong></h2>


创建用户表:

创建角色表:

 创建权限表:

创建用户角色表:

创建角色权限表:

 

2. 查询操作

1. (三表)哪些角色具有‘查询’的权限?

  • 普通查询
  • 子查询:  

注:通过子查询在role_permission表中查询出的rid有多个值(1,2,3),则需要在括号前面添加any字段。Any关键字 表示 where r.id = rp.rid1 or r.id = rp.rid2 or ... 。(扩展:all关键字用and替换or;some关键字和any关键字相同)。

 子查询-内连接:

mysql 多列排名 mysql多表排序_外键

  • 内连接Join...on...     

mysql 多列排名 mysql多表排序_字段_02

 

2.(五表)查询用户“管理员A” 有哪些角色和哪些权限?(分组排序)

  • 排序:先按 角色id 升序,然后在按 权限id升序

查询后的数据:

mysql 多列排名 mysql多表排序_mysql 多列排名_03

 

  • 分组:在上述的基础上:

 1. 查询 管理员A拥有的权限?; 在 order by 的前面加上 group by p.name

mysql 多列排名 mysql多表排序_主键_04

 

 

3.综合-统计

  1. 查询出 至少有2中权限的角色?
  • 首先:查询出角色和权限信息;(拥有2种权限以上的只有 管理员和教师)

mysql 多列排名 mysql多表排序_字段_05

  • 其次 使用 count 函数统计出权限大于2的角色信息

 查询 角色信息,按照角色名称或id进行分组查询。条件可以选择统计权限的个数count(p.name),也可以统计角色的个数,方法不是唯一。

mysql 多列排名 mysql多表排序_主键_06

 

 

 

 

 

 

注: 去重可以使用distinct 和group by 。 显然关联查询中不能使用distinct。原因:distinct位置固定只能跟在select 后面如(select distinct name from...),且只适用于查询某个字段;若需要查询多个字段(如 select distinct name ,age from ...),则mysql会过滤(name字段并且age字段)也相同的数据。

 

后续...

 

 

 

 

附:查看mysql查询效率--explain的用法

explain语句用于查看一条SQL语句的查询执行计划,直接把explain放到要执行的SQL语句的前面即可。

例:explain select * from ...

explain extended和explain的输出结果一样,只是用explain extended语句后可以通过show warnings查看一条SQL语句的反编译的结果,让我们知道我们输入的一条SQL语句真正是怎么执行的。 

对输入结果简单解释一下:

  • select_type:表示select类型,常见的取值有SIMPLE(不使用表连接或子查询)、PRIMARY(主查询,即外层的查询)、UNION(UNION中的或者后面的查询语句)、SUBQUERY(子查询中的第一个select)等。
  • table:输出结果集的表。
  • type:表示表的连接类型,性能由好到差的连接类型为
  1. system(表中仅有一行,即常量表)、
  2. const(单表中最多有一个匹配行,例如PRIMARY KEY或者UNIQUE INDEX)、
  3. eq_ref(对于前面的每一行,在此表中只查询一条记录,简单来说,就是多表连接中使用PRIMARYKEY或者UNIQUE INDEX)、
  4. ref(与eq_ref类似,区别在于不使用PRIMARYKEY或者UNIQUE INDEX,而是使用普通的索引)、
  5. ref_of_null(与ref类似,区别在于条件中包含对NULL的查询)
  6. index_merge(索引合并化)、
  7. unique_subquery(in的后面是一个查询主键字段的子查询)、
  8. index_subquery(与unique_subquery类似,区别在于in的后面是查询非唯一索引字段的子查询)、
  9. range(单表中的范围查询)、
  10. index(对于前面的每一行都通过查询索引来得到数据)、
  11. all(对于前面的每一行的都通过全表扫描来获得数据)。

  结果值从好到坏依次是:system > const > eq_ref > ref > fulltext > ref_or_null >index_merge > unique_subquery > index_subquery > range > index > ALL

一般来说,得保证查询至少达到range级别,最好能达到ref。

  • possible_keys:表示查询时,可能使用到的索引。
  • key:表示实际使用的索引
  • key_len:索引字段的长度
  • rows:扫描行的数量
  • extra:执行情况的说明和描述。
  •  


多表查询 : 连接查询-子查询