数据库篇-mysql详解( 四 )之范式与数据高级骚操作

标题.png

一 : 主键

primary key,主要的键. 一张表只能有一个字段可以使用对应的键, 用来唯一的约束该字段里面的数据, 不能重复: 这种称之为主键.一张表只能有最多一个主键

操作

  • 增加主键

(1) : 在创建表的时候,直接在字段之后,跟primary key 关键字(主键本身不允许为空)
优点 : 非常的直接
缺点 : 只能使用一个字段作作为主键

create table my_pri1(
name varchar(20) not null comment '姓名',
number char(10) primary key comment '学号: stu + 0000, 不能重复'
)charset utf8;
+--------+-------------+------+-----+---------+-------+
| Field  | Type        | Null | Key | Default | Extra |
+--------+-------------+------+-----+---------+-------+
| name   | varchar(20) | NO   |     | NULL    |       |
| number | char(10)    | NO   | PRI | NULL    |       |
+--------+-------------+------+-----+---------+-------+

(2) : 在创建表的时候,在所有字段之后,使用primary key(主键字段列表)来创建主键(如果有多个字段为主键,可以是复合主键)

-- 复合主键
create table my_pri2(
number char(10) comment '学号: stu + 0000',
course char(10) comment '课程代码: 3901 + 0000',
score tinyint unsigned default 60 comment '成绩',
-- 增加主键限制: 学号和课程号应该是个对应的,具有唯一性
primary key(number,course)
)charset utf8;
+--------+---------------------+------+-----+---------+-------+
| Field  | Type                | Null | Key | Default | Extra |
+--------+---------------------+------+-----+---------+-------+
| number | char(10)            | NO   | PRI | NULL    |       |
| course | char(10)            | NO   | PRI | NULL    |       |
| score  | tinyint(3) unsigned | YES  |     | 60      |       |
+--------+---------------------+------+-----+---------+-------+

(3) : 当表已经创建好之后,额外追加主键,可以通过修改表字段属性,也可以直接追加,Alter table 表名add primary key(字段列表);

注意 : 表中字段对应的数据本身是独立的(不重复)

create table my_pri3(
course char(10) not null comment '课程编号: 3901 + 0000',
name varchar(10) not null comment '课程名字'
);
+--------+-------------+------+-----+---------+-------+
| Field  | Type        | Null | Key | Default | Extra |
+--------+-------------+------+-----+---------+-------+
| course | char(10)    | NO   |     | NULL    |       |
| name   | varchar(10) | NO   |     | NULL    |       |
+--------+-------------+------+-----+---------+-------+
alter table my_pri3 modify course char(10) primary key comment '课程编号: 3901 + 0000';
或者
alter table my_pri3 add primary key(course);(追加主键主要用这个)
+--------+-------------+------+-----+---------+-------+
| Field  | Type        | Null | Key | Default | Extra |
+--------+-------------+------+-----+---------+-------+
| course | char(10)    | NO   | PRI | NULL    |       |
| name   | varchar(10) | NO   |     | NULL    |       |
+--------+-------------+------+-----+---------+-------+
  • 主键约束

主键对应的字段中的数据不允许重复,一旦重复,数据操作失败(增/改)

mysql> select * from my_pri1;
+--------+-----------+
| name   | number    |
+--------+-----------+
| 雪芙   | class0001 |
| 思思   | class0002 |
+--------+-----------+
-- 主键冲突(重复)
insert into my_pri1 values('美美','class0002'); -- 不可以: 主键冲突
ERROR 1062 (23000): Duplicate entry 'class0002' for key 'PRIMARY'
  • 更新与删除主键

没有办法更新主键,主键必须先删除再增加.

mysql> desc my_pri1;
+--------+-------------+------+-----+---------+-------+
| Field  | Type        | Null | Key | Default | Extra |
+--------+-------------+------+-----+---------+-------+
| name   | varchar(20) | NO   |     | NULL    |       |
| number | char(10)    | NO   | PRI | NULL    |       |
+--------+-------------+------+-----+---------+-------+

删除主键
Alter table 表名 drop primary key

alter table my_pri3 drop primary key;
+--------+-------------+------+-----+---------+-------+
| Field  | Type        | Null | Key | Default | Extra |
+--------+-------------+------+-----+---------+-------+
| name   | varchar(20) | NO   |     | NULL    |       |
| number | char(10)    | NO   |     | NULL    |       |
+--------+-------------+------+-----+---------+-------+

二 : 主键分类

在实际创建表的过程中, 很少使用真实业务数据作为主键字段(业务主键,如学号,课程号); 大部分的时候是使用逻辑性的字段(字段没有业务含义,值是什么都没有关系), 将这种字段主键称之为逻辑主键

Create table my_student(
Id int primary key auto_increment comment ‘逻辑主键: 自增长’,​-- 逻辑主键
Number char(10) not null  comment ‘学号’,
Name varchar(10) not null
)

三 : 自动增长

自增长: 当对应的字段,不给值,或者说给默认值,或者给NULL的时候, 会自动的被系统触发, 系统会从当前字段中已有的最大值再进行+1操作,得到一个新的在不同的字段.
自增长通常是跟主键搭配

( 1 )新增自增长

auto_increment
① : 任何一个字段要做自增长必须前提是本身是一个索引(key一栏有值)

mysql> create table my_auto(
    -> id int auto_increment comment '自动增长',
    -> name varchar(10) not null
    -> )charset utf8;
ERROR 1075 (42000): Incorrect table definition; there can be only one auto column and it must be defined as a key

② : 自动增长字段必须是数字(整形)

mysql> create table my_auto(
    -> id varchar(1) primary key auto_increment comment '自动增长',
    -> name varchar(10) not null
    -> )charset utf8;
ERROR 1063 (42000): Incorrect column specifier for column 'id'

③ : 一张表最多只能有一个自增长

-- 自增长
mysql> create table my_auto(
    -> id int primary key auto_increment comment '自动增长',
    -> name varchar(10) not null
    -> )charset utf8;
Query OK, 0 rows affected (0.03 sec)
+-------+-------------+------+-----+---------+----------------+
| Field | Type        | Null | Key | Default | Extra          |
+-------+-------------+------+-----+---------+----------------+
| id    | int(11)     | NO   | PRI | NULL    | auto_increment |
| name  | varchar(10) | NO   |     | NULL    |                |
+-------+-------------+------+-----+---------+----------------+

( 2 )自增长使用

当自增长被给定的值为NULL或者默认值的时候会触发自动增长.

insert into my_auto(name) values('雪芙');
insert into my_auto values(null,'思思');
insert into my_auto values(default,'彩彩');
+----+--------+
| id | name   |
+----+--------+
|  1 | 雪芙   |
|  2 | 思思   |
|  3 | 彩彩   |
+----+--------+

自增长如果对应的字段输入了值,那么自增长失效,但是下一次还是能够正确的自增长(从最大值+1)

insert into my_auto(name) values('雪芙');
insert into my_auto values(null,'思思');
insert into my_auto values(default,'彩彩');
+----+--------+
| id | name   |
+----+--------+
|  1 | 雪芙   |
|  2 | 思思   |
|  3 | 彩彩   |
|  6 | 美美   |
|  7 | 想想   |
+----+--------+

如何确定下一次是什么自增长,可以通过查看表创建语句看到

mysql> show create table my_auto;
+---------+---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------+
| Table   | Create Table                                                                                                                                                                                      |
+---------+---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------+
| my_auto | CREATE TABLE `my_auto` (
  `id` int(11) NOT NULL AUTO_INCREMENT COMMENT '自动增长',
  `name` varchar(10) NOT NULL,
  PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=8 DEFAULT CHARSET=utf8     |
+---------+---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------+

( 3 )修改自增长

自增长如果是涉及到字段改变,必须先删除自增长,后增加(一张表只能有一个自增长)
修改当前自增长已经存在的值 : 修改只能比当前已有的自增长的最大值大,不能小(小了不生效)
Alter table 表名 auto_increment = 值;
向下修改(小),但是不生效

alter table my_auto auto_increment = 4; -- 向下修改(小)

向上修改(大),生效

alter table my_auto auto_increment = 10; -- 向上修改 
---------------------------------------------------------+
| my_auto | CREATE TABLE `my_auto` (
  `id` int(11) NOT NULL AUTO_INCREMENT COMMENT '自动增长',
  `name` varchar(10) NOT NULL,
  PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=10 DEFAULT CHARSET=utf8     |
+---------+----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------+

( 4 )修改自增长步长

为什么自增长是从1开始,为什么每次都是自增1呢.
所有系统的变现(如字符集,校对集)都是由系统内部的变量进行控制的.
查看自增长对应的变量 : show variables like 'auto_increment%';

-- 查看自增长变量
show variables like 'auto_increment%';

auto_increment_incremen : 步长
auto_increment_offset : 起始值

+--------------------------+-------+
| Variable_name            | Value |
+--------------------------+-------+
| auto_increment_increment | 1     |
| auto_increment_offset    | 1     |
+--------------------------+-------+

可以修改变量实现不同的效果,修改是对整个数据修改,而不是单张表;
set auto_increment_increment = 5; 一次自增5

set auto_increment_increment = 5;
+--------------------------+-------+
| Variable_name            | Value |
+--------------------------+-------+
| auto_increment_increment | 5     |
| auto_increment_offset    | 1     |
+--------------------------+-------+
+----+--------+
| id | name   |
+----+--------+
|  1 | 雪芙   |
|  2 | 思思   |
|  3 | 彩彩   |
|  6 | 美美   |
|  7 | 想想   |
| 11 | 劲劲   |
| 16 | 辰辰   |
+----+--------+

( 5 )删除自增长

自增长是字段的一个属性,可以通过modify来进行修改(保证字段没有auto_increment即可)
Alter table 表名 modify 字段类型;

错误 : 主键理论是单独存在,有主键的时候,千万不要再加主键

alter table my_auto modify id int primary key; 
ERROR 1068 (42000): Multiple primary key defined

正确 :

+-------+-------------+------+-----+---------+----------------+
| Field | Type        | Null | Key | Default | Extra          |
+-------+-------------+------+-----+---------+----------------+
| id    | int(11)     | NO   | PRI | NULL    | auto_increment |
| name  | varchar(10) | NO   |     | NULL    |                |
+-------+-------------+------+-----+---------+----------------+
alter table my_auto modify id int;
+-------+-------------+------+-----+---------+-------+
| Field | Type        | Null | Key | Default | Extra |
+-------+-------------+------+-----+---------+-------+
| id    | int(11)     | NO   | PRI | NULL    |       |
| name  | varchar(10) | NO   |     | NULL    |       |
+-------+-------------+------+-----+---------+-------+

四 : 唯一键

一张表往往有很多字段需要具有唯一性,数据不能重复: 但是一张表中只能有一个主键: 唯一键(unique key)就可以解决表中有多个字段需要唯一性约束的问题.
唯一键的本质与主键差不多: 唯一键默认的允许自动为空,而且可以多个为空(空字段不参与唯一性比较)

( 1 )增加唯一键

  • 在创建表的时候,字段之后直接跟unique/unique key
create table my_unique1(
number char(10) unique comment '学号: 唯一,允许为空',
name varchar(20) not null
)charset utf8;
+--------+-------------+------+-----+---------+-------+
| Field  | Type        | Null | Key | Default | Extra |
+--------+-------------+------+-----+---------+-------+
| number | char(10)    | YES  | UNI | NULL    |       |
| name   | varchar(20) | NO   |     | NULL    |       |
+--------+-------------+------+-----+---------+-------+
  • 在所有的字段之后增加unique key (字段列表); 复合唯一键
create table my_unique2(
number char(10) not null comment '学号',
name varchar(20) not null,
-- 增加唯一键
unique key(number)
)charset utf8;

之所以会显示PRI : 刚好是一个不为空的唯一键(主键性质一样) : 原因是因为该表没有主键

+--------+-------------+------+-----+---------+-------+
| Field  | Type        | Null | Key | Default | Extra |
+--------+-------------+------+-----+---------+-------+
| number | char(10)    | NO   | PRI | NULL    |       |
| name   | varchar(20) | NO   |     | NULL    |       |
+--------+-------------+------+-----+---------+-------+
  • 在创建表之后增加为一键
+--------+-------------+------+-----+---------+----------------+
| Field  | Type        | Null | Key | Default | Extra          |
+--------+-------------+------+-----+---------+----------------+
| id     | int(11)     | NO   | PRI | NULL    | auto_increment |
| number | char(10)    | NO   |     | NULL    |                |
| name   | varchar(20) | NO   |     | NULL    |                |
+--------+-------------+------+-----+---------+----------------+
-- 追加唯一键
alter table my_unique3 add unique key(number);
+--------+-------------+------+-----+---------+----------------+
| Field  | Type        | Null | Key | Default | Extra          |
+--------+-------------+------+-----+---------+----------------+
| id     | int(11)     | NO   | PRI | NULL    | auto_increment |
| number | char(10)    | NO   | UNI | NULL    |                |
| name   | varchar(20) | NO   |     | NULL    |                |
+--------+-------------+------+-----+---------+----------------+

( 2 )唯一键约束

唯一键与主键本质相同 : 唯一的区别就是唯一键默认允许为空,而且多个为空.

insert into my_unique1 values(null,'美美'),('class0001','彩彩'),(null,'娜娜');
+-----------+--------+
| number    | name   |
+-----------+--------+
| NULL      | 美美   |
| class0001 | 彩彩   |
| NULL      | 娜娜   |
+-----------+--------+

添加重复元素报错

insert into my_unique1 values('class0001','思思');
ERROR 1062 (23000): Duplicate entry 'class0001' for key 'number'

: 如果唯一键也不允许为空那么唯一键与主键的约束作用是一致的.

( 3 )更新/删除唯一键

更新唯一键 : 先删除后增(唯一键可以有多个 : 可以不删除)
删除唯一键
Alter table 表名 drop unique key; -- 错误: 唯一键有多个
Alter table 表名 drop index 索引名字; -- 唯一键默认的使用字段名作为索引名字

alter table my_unique3 drop index number;
+--------+-------------+------+-----+---------+----------------+
| Field  | Type        | Null | Key | Default | Extra          |
+--------+-------------+------+-----+---------+----------------+
| id     | int(11)     | NO   | PRI | NULL    | auto_increment |
| number | char(10)    | NO   |     | NULL    |                |
| name   | varchar(20) | NO   |     | NULL    |                |
+--------+-------------+------+-----+---------+----------------+

五 : 索引

几乎所有的索引都是建立在字段之上.
索引 : 系统根据某种算法,将已有的数据(未来可能新增的数据)单独建立一个文件,文件能够实现快速匹配数据,并且能够快速的找到对应表中的记录.
索引的意义
1.提升查询数据的效率
2.约束数据的有效性( 唯一性等)
增加索引的前提条件: 索引本身会产生索引文件(有时候有可能比数据文件还大) ,会非常耗费磁盘空间.
如果某个字段需要作为查询的条件经常使用, 那么可以使用索引(一定会想办法增加);
如果某个字段需要进行数据的有效性约束, 也可能使用索引(主键,唯一键)

Mysql中提供了多种索引

  • 主键索引: primary key
  • 唯一索引: unique key
  • 全文索引: fulltext index
  • 普通索引: index

全文索引: 针对文章内部的关键字进行索引
全文索引最大的问题: 在于如何确定关键字
英文很容易: 英文单词与单词之间有空格
中文很难: 没有空格, 而且中文可以各种随意组合(分词: sphinx)

六 : 范式

Normal Format : 是一种离散数学中的知识, 是为了解决一种数据的存储与优化的问题: 保存数据的存储之后, 凡是能够通过关系寻找出来的数据,坚决不再重复存储: 终极目标是为了减少数据的冗余.

范式: 是一种分层结构的规范, 分为六层: 每一次层都比上一层更加严格: 若要满足下一层范式,前提是满足上一层范式.

六层范式 : 1NF,2NF,3NF...6NF, 1NF是最底层,要求最低;6NF最高层,最严格.

Mysql属于关系型数据库: 有空间浪费: 也是致力于节省存储空间: 与范式所有解决的问题不谋而合: 在设计数据库的时候, 会利用到范式来指导设计.
但是数据库不单是要解决空间问题,要保证效率问题: 范式只为解决空间问题, 所以数据库的设计又不可能完全按照范式的要求实现: 一般情况下,只有前三种范式需要满足.

范式在数据库的设计当中是有指导意义: 但是不是强制规范.

( 1 ) 1NF

第一范式 : 在设计表存储数据的时候, 如果表中设计的字段存储的数据,在取出来使用之前还需要额外的处理(拆分),那么说表的设计不满足第一范式: 第一范式要求字段的数据具有原子性: 不可再分.
讲师代课表

代课表

上表设计不存在问题,但是如果需求是将数据查出来之后,要求显示一个老师从什么时候开始上课,到什么时候节课: 需要将代课时间进行拆分: 不符合1NF, 数据不具有原子性, 可以再拆分.

解决方案 : 将代课时间拆分成两个字段就解决问题
解决

( 2 )2NF

第二范式 : 在数据表设计的过程中,如果有复合主键(多字段主键), 且表中有字段并不是由整个主键来确定, 而是依赖主键中的某个字段(主键的部分): 存在字段依赖主键的部分的问题, 称之为部分依赖: 第二范式就是要解决表设计不允许出现部分依赖.

讲师带课表
讲师代课表

以上表中: 因为讲师没有办法作为独立主键, 需要结合班级才能作为主键(复合主键: 一个老师在一个班永远只带一个阶段的课): 代课时间,开始和结束字段都与当前的代课主键(讲师和班级): 但是性别并不依赖班级, 教室不依赖讲师: 性别只依赖讲师, 教室只依赖班级: 出现了性别和教室依赖主键中的一部分: 部分依赖.不符合第二范式.

解决 :

解决方案1: 可以将性别与讲师单独成表, 班级与教室也单独成表.
解决方案2: 取消复合主键, 使用逻辑主键

( 3 )3NF

要满足第三范式,必须满足第二范式.

第三范式: 理论上讲,应该一张表中的所有字段都应该直接依赖主键(逻辑主键: 代表的是业务主键), 如果表设计中存在一个字段, 并不直接依赖主键,而是通过某个非主键字段依赖,最终实现依赖主键: 把这种不是直接依赖主键,而是依赖非主键字段的依赖关系称之为传递依赖. 第三范式就是要解决传递依赖的问题.

讲师带课表

以上设计方案中: 性别依赖讲师存在, 讲师依赖主键; 教室依赖班级,班级依赖主键: 性别和教室都存在传递依赖.

解决方案 :

将存在传递依赖的字段,以及依赖的字段本身单独取出,形成一个单独的表, 然后在需要对应的信息的时候, 使用对应的实体表的主键加进来.

讲师代课表

讲师表

班级表

从表面来看讲师表班级表也会有传递依赖,但是逻辑主键只是业务主键的一个替代,永远不要看逻辑主键,要看真正所代表的主键
所以 讲师表: ID = 讲师​, 班级表中: ID = 班级

( 4 )逆规范化

有时候, 在设计表的时候,如果一张表中有几个字段是需要从另外的表中去获取信息. 理论上讲, 的确可以获取到想要的数据, 但是就是效率低一点. 会刻意的在某些表中,不去保存另外表的主键(逻辑主键), 而是直接保存想要的数据信息: 这样一来,在查询数据的时候, 一张表可以直接提供数据, 而不需要多表查询(效率低), 但是会导致数据冗余增加.

讲师代课信息表


逆规规范化 : 磁盘利用率与效率的对抗

七 : 数据高级操作

( 1 )主键冲突

在数据插入的时候, 假设主键对应的值已经存在: 插入一定会失败!
当主键存在冲突的时候(Duplicate key),可以选择性的进行处理: 更新和替换

+---------+------+
| name    | room |
+---------+------+
| PHP0810 | B203 |
| PHP0710 | B203 |
+---------+------+

插入重复数据

insert into my_class values('PHP0810','B205');
ERROR 1062 (23000): Duplicate entry 'PHP0810' for key 'PRIMARY'
解决主键冲突 :
-- 主键冲突: 更新
insert into my_class values('PHP0810','B205')
-- 冲突处理
on duplicate key update
-- 更新教室
room = 'B205';
+---------+------+
| name    | room |
+---------+------+
| PHP0710 | B203 |
| PHP0810 | B205 |

( 2 )主键冲突 : 替换

Replace into 表名 [(字段列表:包含主键)] values(值列表);

replace into my_class values('PHP0710','A203');
+---------+------+
| name    | room |
+---------+------+
| PHP0710 | A203 |
| PHP0810 | B205 |
+---------+------+

( 3 )蠕虫复制

从已有的数据中去获取数据,然后将数据又进行新增操作: 数据成倍的增加.

表创建高级操作: 从已有表创建新表(复制表结构)
Create table 表名 like 数据库.表名;

+--------+--------+
| name   | money  |
+--------+--------+
| 思思   | 123.23 |
| 美美   | 234.55 |
mysql> create table my_copy like my_teacher;
Query OK, 0 rows affected (0.04 sec)
+-------+---------------+------+-----+---------+-------+
| Field | Type          | Null | Key | Default | Extra |
+-------+---------------+------+-----+---------+-------+
| name  | varchar(20)   | NO   |     | NULL    |       |
| money | decimal(10,2) | NO   |     | NULL    |       |
+-------+---------------+------+-----+---------+-------+

蠕虫复制: 先查出数据, 然后将查出的数据新增一遍
Insert into 表名[(字段列表)] select 字段列表/* from 数据表名;

mysql> insert into my_copy select * from my_copy;
Query OK, 2 rows affected (0.00 sec)
Records: 2  Duplicates: 0  Warnings: 0

mysql> insert into my_copy select * from my_copy;
Query OK, 4 rows affected (0.00 sec)
Records: 4  Duplicates: 0  Warnings: 0

mysql> insert into my_copy select * from my_copy;
Query OK, 8 rows affected (0.01 sec)
Records: 8  Duplicates: 0  Warnings: 0
+--------+--------+
| name   | money  |
+--------+--------+
| 思思   | 123.23 |
| 美美   | 234.55 |
| 思思   | 123.23 |
| 美美   | 234.55 |
| 思思   | 123.23 |
| 美美   | 234.55 |
| 思思   | 123.23 |
| 美美   | 234.55 |
| 思思   | 123.23 |
| 美美   | 234.55 |
| 思思   | 123.23 |
| 美美   | 234.55 |
| 思思   | 123.23 |
| 美美   | 234.55 |
| 思思   | 123.23 |
| 美美   | 234.55 |
+--------+--------+

蠕虫复制的意义

  • 从已有表拷贝数据到新表中
  • 可以迅速的让表中的数据膨胀到一定的数量级: 测试表的压力以及效率

( 4 )更新数据

Update 表名 set 字段 = 值 [where条件];

高级新增语法
Update 表名 set 字段 = 值 [where条件] [limit 更新数量];

mysql> update my_copy set name = '可可' where name = '美美' limit 3;
Query OK, 3 rows affected (0.00 sec)
+--------+--------+
| name   | money  |
+--------+--------+
| 思思   | 123.23 |
| 可可   | 234.55 |
| 思思   | 123.23 |
| 可可   | 234.55 |
| 思思   | 123.23 |
| 可可   | 234.55 |
| 思思   | 123.23 |
| 美美   | 234.55 |
| 思思   | 123.23 |
| 美美   | 234.55 |
| 思思   | 123.23 |
| 美美   | 234.55 |
| 思思   | 123.23 |
| 美美   | 234.55 |
| 思思   | 123.23 |
| 美美   | 234.55 |
+--------+--------+

( 5 )删除数据

与更新类似: 可以通过limit来限制数量
Delete from 表名 [where条件] [limit 数量];

mysql> delete from my_copy where name = '思思' limit 5;
Query OK, 5 rows affected (0.00 sec)
+--------+--------+
| name   | money  |
+--------+--------+
| 可可   | 234.55 |
| 可可   | 234.55 |
| 可可   | 234.55 |
| 美美   | 234.55 |
| 美美   | 234.55 |
| 思思   | 123.23 |
| 美美   | 234.55 |
| 思思   | 123.23 |
| 美美   | 234.55 |
| 思思   | 123.23 |
| 美美   | 234.55 |
+--------+--------+

删除: 如果表中存在主键自增长,那么当删除之后, 自增长不会还原

数据的删除是不会改变表结构, 只能删除表后重建表

Truncate 表名;​-- 先删除改变,后新增改变

+----+--------+
| id | name   |
+----+--------+
|  1 | 雪芙   |
|  2 | 思思   |
|  3 | 彩彩   |
|  6 | 美美   |
|  7 | 想想   |
| 11 | 劲劲   |
| 16 | 辰辰   |
+----+--------+
-- 清空表: 重置自增长
truncate my_auto;
+---------+--------------------------------------------------------------------------------------------------------------------------------------------+
| Table   | Create Table                                                                                                                               |
+---------+--------------------------------------------------------------------------------------------------------------------------------------------+
| my_auto | CREATE TABLE `my_auto` (
  `id` int(11) NOT NULL,
  `name` varchar(10) NOT NULL,
  PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8 |
+---------+--------------------------------------------------------------------------------------------------------------------------------------------+

( 6 )查询数据

基本语法
Select 字段列表/* from 表名 [where条件];

完整语法
Select [select选项] 字段列表[字段别名]/* from 数据源 [where条件子句] [group by子句] [having子句] [order by子句] [limit 子句];

Select选项: select对查出来的结果的处理方式
All: 默认的,保留所有的结果
Distinct: 去重, 查出来的结果,将重复给去除(所有字段都相同)

select distinct * from  my_auto;
字段别名

字段别名: 当数据进行查询出来的时候, 有时候名字并一定就满足需求(多表查询的时候, 会有同名字段). 需要对字段名进行重命名: 别名

语法
字段名 [as] 别名;

数据源

数据源: 数据的来源, 关系型数据库的来源都是数据表: 本质上只要保证数据类似二维表,最终都可以作为数据源.

数据源分为多种: 单表数据源, 多表数据源, 查询语句

  • 单表数据源: select * from 表名;

  • 多表数据源: select* from 表名1,表名2...;
    从一张表中取出一条记录,去另外一张表中匹配所有记录,而且全部保留:(记录数和字段数),将这种结果成为: 笛卡尔积(交叉连接): 笛卡尔积没什么卵用, 所以应该尽量避免.

  • 子查询: 数据的来源是一条查询语句(查询语句的结果是二维表)
    Select * from (select 语句) as 表名;

( 7 )Where子句

Where子句: 用来判断数据,筛选数据.
Where子句返回结果: 0或者1, 0代表false,1代表true.

判断条件:

比较运算符:>,<, >=,<= ,!= ,<>,=, like, between and,in/not in
逻辑运算符: &&(and), ||(or), !(not)

Where原理:

where是唯一一个直接从磁盘获取数据的时候就开始判断的条件: 从磁盘取出一条记录, 开始进行where判断: 判断的结果如果成立保存到内存;如果失败直接放弃.

( 8 )Group by子句

Group by:分组的意思, 根据某个字段进行分组(相同的放一组,不同的分到不同的组)

+----+-----------+--------+------+--------+
| id | number    | name   | sex  | height |
+----+-----------+--------+------+--------+
|  1 | class0001 | 思思   | 男   |    184 |
|  2 | class0002 | 可可   | 男   |    182 |
|  3 | class0003 | 娜娜   | 女   |    156 |
|  4 | class0004 | 美美   | 男   |    172 |
|  5 | class0005 | 彩彩   | 男   |    166 |
+----+-----------+--------+------+--------+

基本语法 : gruop by 字段名;

select * from my_student group by sex;

分组的意思: 是为了统计数据(按组统计: 按分组字段进行数据统计)
SQL提供了一系列统计函数

  • Count(): 统计分组后的记录数: 每一组有多少记录
  • Max():​统计每组中最大的值
  • Min(): 统计最小值
  • Avg(): 统计平均值
  • Sum(): 统计和
mysql> select sex,count(*),max(height),min(height) from my_student group by sex;
+------+----------+-------------+-------------+
| sex  | count(*) | max(height) | min(height) |
+------+----------+-------------+-------------+
| 女   |        1 |         156 |         156 |
| 男   |        4 |         184 |         166 |
+------+----------+-------------+-------------+

Count函数: 里面可以使用两种参数: *代表统计记录,字段名代表统计对应的字段(NULL不统计)

Group by 字段 [asc|desc];​-- 对分组的结果然后合并之后的整个结果进行排序
按照男,女 校对集顺序

多字段分组 : 先根据一个字段进行分组,然后对分组后的结果再次按照其他字段进行分组

alter table my_student add c_id int;
Query OK, 0 rows affected (0.06 sec)
Records: 0  Duplicates: 0  Warnings: 0

mysql> update my_student set c_id = ceil(random()*3);
+----+-----------+--------+------+--------+------+
| id | number    | name   | sex  | height | c_id |
+----+-----------+--------+------+--------+------+
|  1 | class0001 | 思思   | 男   |    184 |    3 |
|  2 | class0002 | 可可   | 男   |    182 |    1 |
|  3 | class0003 | 娜娜   | 女   |    156 |    1 |
|  4 | class0004 | 美美   | 男   |    172 |    3 |
|  5 | class0005 | 彩彩   | 男   |    166 |    3 |
+----+-----------+--------+------+--------+------+

多字段分组,先按班级份,再按性别分

mysql> select c_id,sex,count(*),group_concat(name) from my_student group by c_id,sex; -- 多字段排序

group_concat: 可以对分组的结果中的某个字段进行字符串连接(保留该组所有的某个字段):

+------+------+----------+----------------------+
| c_id | sex  | count(*) | group_concat(name)   |
+------+------+----------+----------------------+
|    1 | 女   |        1 | 娜娜                 |
|    1 | 男   |        1 | 可可                 |
|    3 | 男   |        3 | 思思,美美,彩彩       |
+------+------+----------+----------------------+

回溯统计 : with rollup: 任何一个分组后都会有一个小组, 最后都需要向上级分组进行汇报统计: 根据当前分组的字段. 这就是回溯统计: 回溯统计的时候会将分组字段置空.

select c_id,count(*) from my_student group by c_id with rollup;
+------+----------+
| c_id | count(*) |
+------+----------+
|    1 |        2 |
|    3 |        3 |
| NULL |        5 |
+------+----------+

多字段回溯 : 考虑第一层分组会有此回溯: 第二次分组要看第一次分组的组数, 组数是多少,回溯就是多少,然后加上第一层回溯即可.

+------+------+----------+------------------------------------+
| c_id | sex  | count(*) | group_concat(name)                 |
+------+------+----------+------------------------------------+
|    1 | 女   |        1 | 娜娜                               |
|    1 | 男   |        1 | 可可                               |
|    1 | NULL |        2 | 娜娜,可可                          |
|    3 | 男   |        3 | 思思,美美,彩彩                     |
|    3 | NULL |        3 | 思思,美美,彩彩                     |
| NULL | NULL |        5 | 娜娜,可可,思思,美美,彩彩           |
+------+------+----------+------------------------------------+

( 9 )Having子句

Having子句: 与where子句一样: 进行条件判断的.
Where是针对磁盘数据进行判断: 进入到内存之后,会进行分组操作: 分组结果就需要having来处理.
Having能做where能做的几乎所有事情, 但是where却不能做having能做的很多事情.

① : 分组统计的结果或者说统计函数都只有having能够使用.

select c_id,count(*) from my_student group by c_id having count(*) >= 2;
+------+----------+
| c_id | count(*) |
+------+----------+
|    1 |        2 |
|    3 |        3 |
+------+----------+

② Having能够使用字段别名: where不能: where是从磁盘取数据,而名字只可能是字段名: 别名是在字段进入到内存后才会产生.

select c_id,count(*) as total from my_student group by c_id having total >= 2;
+------+-------+
| c_id | total |
+------+-------+
|    1 |     2 |
|    3 |     3 |
+------+-------+

报错

select c_id,count(*) as total from my_student where total >= 2 group by c_id ;
ERROR 1054 (42S22): Unknown column 'total' in 'where clause'

( 10 )Order by 子句

Order by: 排序, 根据某个字段进行升序或者降序排序, 依赖校对集.
Order by 字段名 [asc|desc]; -- asc是升序(默认的),desc是降序

select * from my_student order by c_id;
+----+-----------+--------+------+--------+------+
| id | number    | name   | sex  | height | c_id |
+----+-----------+--------+------+--------+------+
|  2 | class0002 | 可可   | 男   |    182 |    1 |
|  3 | class0003 | 娜娜   | 女   |    156 |    1 |
|  1 | class0001 | 思思   | 男   |    184 |    3 |
|  4 | class0004 | 美美   | 男   |    172 |    3 |
|  5 | class0005 | 彩彩   | 男   |    166 |    3 |
+----+-----------+--------+------+--------+------+

排序可以进行多字段排序: 先根据某个字段进行排序, 然后排序好的内部,再按照某个数据进行再次排序:

-- 多字段排序: 先班级排序,后性别排序
select * from my_student order by c_id, sex desc;
+----+-----------+--------+------+--------+------+
| id | number    | name   | sex  | height | c_id |
+----+-----------+--------+------+--------+------+
|  2 | class0002 | 可可   | 男   |    182 |    1 |
|  3 | class0003 | 娜娜   | 女   |    156 |    1 |
|  1 | class0001 | 思思   | 男   |    184 |    3 |
|  4 | class0004 | 美美   | 男   |    172 |    3 |
|  5 | class0005 | 彩彩   | 男   |    166 |    3 |
+----+-----------+--------+------+--------+------+

( 11 )Limit子句

Limit子句是一种限制结果的语句: 限制数量.
Limit有两种使用方式

  • 只用来限制长度(数据量): limit 数据量
select * from my_student limit 2;
+----+-----------+--------+------+--------+------+
| id | number    | name   | sex  | height | c_id |
+----+-----------+--------+------+--------+------+
|  1 | class0001 | 思思   | 男   |    184 |    3 |
|  2 | class0002 | 可可   | 男   |    182 |    1 |
+----+-----------+--------+------+--------+------+
  • 限制起始位置,限制数量: limit 起始位置,长度
select * from my_student limit 0,2; -- 记录数是从0开始编号
+----+-----------+--------+------+--------+------+
| id | number    | name   | sex  | height | c_id |
+----+-----------+--------+------+--------+------+
|  1 | class0001 | 思思   | 男   |    184 |    3 |
|  2 | class0002 | 可可   | 男   |    182 |    1 |
+----+-----------+--------+------+--------+------+

Limit主要用来实现数据的分页: 为用户节省时间,提交服务器的响应效率, 减少资源的浪费.
对于用户来讲: 可以点击的分页按钮: 1,2,3,4
对于服务器来讲: 根据用户选择的页码来获取不同的数据: limit offset,length;

Length: 每页显示的数据量: 基本不变
Offset: offset = (页码 - 1) * 每页显示量

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,001评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,210评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,874评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,001评论 1 291
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,022评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,005评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,929评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,742评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,193评论 1 309
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,427评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,583评论 1 346
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,305评论 5 342
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,911评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,564评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,731评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,581评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,478评论 2 352

推荐阅读更多精彩内容