MySQL的增删改查

MySQL表的增删改查

注释:在SQL中使用 --(空格) + 注释内容 来写注释

CRUD

crud就是增加(create) 查询(retrieve) 更新(update) 删除(delete)四个单词的缩写

插入

insert into 表名 values (值,值,值.......); // 一次插入一条记录

增加的值的类型要与表中数据的类型匹配

insert into 表名 values(值,值) , (值,值); //一次插入多条记录

一次插入多行操作的效率比一次插入一条,插入多次要更加快

要是输入中文发现报错就说明当前MySQL不支持中文,需要手动修改配置文件来改变字符编码(一劳永逸的办法)

在创建数据库时可以指定字符编码方式

eg: create database customer character set utf8;(每次都要)

insert在指定的列中插入(可以是某一列 也可以是 多个列) 中插入 一个或者多个值

insert into 表名(列名 , 列名) values(值,值);

image-20220731200208968

insert 一次插入多行记录

insert into 表名 values(2,"喜洋洋"),(3,"美羊羊"); //结合上述的demo表

插入日期

首先先创建一个表

create  table  article(
    articleName varchar(20),
    create_date datetime
);

插入数据

insert into  article values("第一",'2019-1-3 10:34'),
    ("第二",'2018-12-5 14:45'),
    ("第三",'2019-3-4 16:34');-- 日期一定要用''引起来

查询

SQL中使用 select 来查询数据

1.全部查找 (查找表中所有的行和列)

select * from 表名 ;

这个命令是十分危险的,select * from 表名,会让服务大规模读取磁盘上的数据,再把数据通过网卡传给客户端

一旦数据太大, 就很有可能把磁盘带宽和网卡带宽直接挤爆,甚至有可能把服务器卡死,无法响应其他的客户端的请求

2.查询某些列的数据

select 列名,列名 from 表名;

相比于上述的全部查询,这里的指定列查询的数据量就会更加小了

查询完显示对应的列之后,展示的知识一个临时表

所有的查询数据都不会改变原本表的数据

3.表达式查询

一边查 一边计算

image-20220731204439305

image-20220731204446718

这里只是临时表展示,并会不改变原本的数据

第三点续: 表达式查询还可以列与列之间进行计算

image-20220731204932121

有上面的图片可以看出来,列名就是临时表的列名

4.给查询结果的列起别名

select 表达式 as 别名 from 表名

image-20220731211418759

5.查询的时候进行去重

在列名之前使用 distinct 去重, 就会对这个列中的数据去重

select distinct 列名 from 表名;

去重也能针对多个列进行去重

但是只有说多个列对应的数据都是一样的,才能实现多个列去重

6.排序(列中数据排序)

升序:select 列名 from 表名 order by 列名;

降序:select 列名 from 表名 order by 列名 desc;

此处的desc是descend, 降低的意思

之前描述表的结构的关键字 desc 是 describe

按语文成绩升序排列

image-20220731215210520

按语文成绩降序排列

image-20220731215331259

排序还能指定多个列进行了优先级排序

指定多个列的时候,那个列靠前,哪个列的优先级就高

升序:select 列名 from 表名 order by 列名,列名;

降序:select 列名 from 表名 order by 列名 desc,列名 desc; (各自都要降序desc)

image-20220731221440512

会先按照math进行升序排列,要是发现有两行的数据是一样的,就会按照chinese比较,按照升序输出

降序:

image-20220731221813686

7.条件查询

select * from 表名 where 条件

条件比较多

image-20220801091634130

image-20220801091640010

条件查询会先执行,所以不能使用别名

and 比 or 的优先级 高,所以一条语句中同时有and 和 or ,优先执行and

要是想要改变优先级,就加括号

判断是不是空的时候,用的是is null 或者 is not null, 不要用 !=

举例:

英语成绩不及格的人

image-20220801091743049

语文成绩好于数学成绩的人

image-20220801091930089

三门成绩之和小于200的人

image-20220801092054581

image-20220801092828055

条件查询是先进行条件判断,再查询数据

所以不认识total别名, 就会报错 属于是先使用再定义(err)

image-20220801093055021

之前的排序就是属于先执行前面,再进行排序,所有可以使用别名

查询语文成绩大于80,并且数学成绩大于80的人

image-20220801093328041

查询语文成绩大于80 或者 数学成绩大于80

image-20220801093545347

and 和 or 的优先级问题

image-20220801094214753

查询语文成绩在80到90之间的人

image-20220801095003663

模糊搜索 like

image-20220801101950302

查询日期

select * from article where create_date between '2019-01-01 10:30' and '2019-11-10 16:02';

查询文章名为空或者日期为2019年1月1日之后的文章

image-20220801105209616

image-20220802112856860

查询包含的指定的数据 (in)

image-20220802113345537

like 模糊查询 依赖一些通配符来表示要匹配的值的形式

% 代替任意个任意字符(可以是0个字符,也可以是1个字符,还可以是多个字符)

_ 代替一个任意字符

  ![image-20220802113915967](../../../../AppData/Roaming/Typora/typora-user-images/image-20220802113915967.png)

null = null (等于) 结果还是null,也就是false;所以就查询不出来

举例:

image-20220802115614929

遇到这种情况就可以使用<=>来判断

image-20220802115808469

也可以使用is null 和 is not null 来判断null,但是这两种只能判断是否为空的情况 ,<=>适用的范围更广

8.分页查询

有些网站的数据量比较大,一次展示不完,就可以分成多页来展示

select * from 表名 limit N; 查询前 N条 记录

select * from 表名 limit N offset M; 从M条记录 开始,查询N条记录

image-20220802121819428

找出总成绩前三名的同学

image-20220802122247541

找出总成绩前三名的同学(排除猪悟能)

image-20220802122422010

需要注意顺序,代码的执行顺序与写的顺序没有关系

image-20220802122629748

修改

修改就是update

update 表名 set 列名=值,列名= 值 where 条件

将孙悟空的数学成绩改成80分

image-20220802133031623

删除

delete 删除

delete from 表名 where 条件 ;

image-20220802203454195

要是后面没有where 条件,就相当于把表清空了

delete 只是把表中的记录删了

drop就是连表和记录都删掉了,但是这两个的差别不大


在数据库中,select 查询是最复杂的,接下来就要学习一些MySQL的进阶操作

数据库约束

数据库中的数据十分的重要,所以一定要对数据库中的数据进行检查,这就是数据库约束

1.not null 创建数据库的时候,在变量类型后面加上, 确保变量不是null

image-20220802210653546

image-20220802210719061

image-20220802210728474

2.unique 值唯一 ,每次插入的时候都会去遍历查看表中有没有相同的数据,要是有的话,就不能插入

image-20220802211440738

image-20220802211626635

image-20220802211849993

default 值 修改默认值

本来表的默认值是null,可以通过default 来修改默认值

image-20220802220409313

image-20220802220624589

主键 primary key 主键用于身份的标识

主键 primary key 必须是非空的且是唯一的(not null 和 unique 的结合)

image-20220803085641335

在primary key 后面加上auto_increment 就能实现MySQL的数据自增(自增主键机制) , 这样子插入数据的时候就不必手动设置值了,只要将这列设为null即可由数据库自动分配

image-20220803091400957

image-20220803091623979

填成null就会自动分配,要是填成别的数,也能直接插入

image-20220803092006662

自增主键可以理解为是在MySQL中记录了,要是再插入null自增, 当前最大的id是多少,就在最大值后面继续增加

数据库分布式部署: 数据太多,一台服务器放不下,使用多台机器

分库分表:表太大,就可以拆成两个或多个表,分别在不同的主机上,搭建不同的mysql服务器,每个服务器存储其中的一份

对于分库分表的数据来说,从逻辑上还时同一张表,只是存储在不同的机器上,此时,原有的自增主键就失效了,mysql的自增主键保证单机上数据不重复,不能保证分布式部署时不重复

分布式系统生成唯一ID的方法:

生成公式 = (时间戳 + 主机编号 / 机房编号 + 随机因子) => 计算哈希值

理论上无法保证随机因子一定是随机的,但是在实际中随机因子冲突因子冲突的概率是很低的,所以在工程中可以忽略不计

foreign key 外键约束

image-20220803123600058

要使用外键,必须要保证子表引用父表的那一列,一定要是primary key 或者是 unique

班级表(约束别的表) -- 父表

学生表(被约束的表) --子表

当外键约束建立好了之后,要想在学生表中插入记录,classId就一定要在班级表中存在

所以班级表对学生表产生约束

image-20220803124207468

image-20220803124212103

image-20220803124221109

外键约束下,插入数据,会触发查询

往学生表中插入新纪录,就会自动在班级表中查询,看classId是否存在,要是不存在就插入失败,这就是父表对子表的约束

相同的,子表其实对于父表也是由约束的

在子表存在的情况下,要是直接删除父表,就会报错

要是子表中已经引用(使用),父表也是删除不了的

一个实际的问题:(用到了外键 逻辑删除)

image-20220803130746603

卖一件商品在淘宝店铺,订单表就多了一条记录,过了一段时间,商家想要下架该商品,就要把

这个商品的goodsId删除,但是订单表(子表)中依然有goodsId,无法直接删除商品表(父表) 的goodsId , 此时该怎么办?

逻辑删除 : 在电脑中的删除,并不是直接将内容清空,而是将数据标记为"失效",所谓的删除只是对标记为进行了修改

所以,只要在商品表增加一列valid , 要是商品有效,值就是1,无效就是0,这样就可以做到上架下架

数据库的设计

XX项目中,数据库是如何设计的?

这就是在问数据库中有几个表,这些表都是干什么的?每个表中有哪些列,这些列都是干什么的?

数据库设计的主要思路:

  1. 根据需求,找到"实体"
  2. 梳理清楚实体之间的关系

所谓的实体类似于面向对象编程设计,"对象"就是需求中的关键性的名词

实体之间的关系有哪些?

  1. 一对一关系

  2. 一对多关系

  3. 多对多关系

  4. 没有关系

    一般来说,每个实体都会安排一个表

具体实例:

image-20220803224217406

-- 学生与宿舍  m:1
-- 宿舍查房与宿舍 m:1
-- 谁是1谁就会被约束,所以dormitory会在最前面
create table  dormitory(
    id int primary key,
    num varchar(20)
);
create table student(
    id int primary key,
    name varchar(20),
    dorm_id int,
    foreign key(dorm_id) references dormitory(id)-- 上面的宿舍就是被约束的
);
create table exam(
    id int primary key,
    time_date timestamp,--记录考勤的时间
    dorm_id int,
    foreign key(dorm_id) references dormitory(id)
);

image-20220803225131455

-- 用户和车 1:m
-- 车和违章记录  1:m
-- 题目中表示用户和车都被违章记录约束
create table user(
    id int primary key,
    name varchar(20)
);
create table car(
    id int primary key,
    num varchar(20),
    user_id int,
    foreign key(user_id) references user(id)
);
create table info(
    id int primary key,
    name varchar(20),
    user_id int,
    car_id int,
    foreign key(user_id) references user(id),
    foreign key(car_id) references car(id)
);

image-20220803230048523

-- 食堂与仓口 1:m
-- 仓口与收费 1:m
-- 被约束的是1,所以canteen在最前面
create table canteen(
    id int primary key,
    name varchar(20)
);
create table hatch(  -- 食堂仓口表
    id int primary key,
    name varchar(20),
    canteen_id varchar(20),
    foreign key(canteen_id) references canteen(id)
);
create table money(
     price int primary  key,
     name varchar(20),
     hatch_id varchar(20),
     foreign key(hatch_id) references hatch(id)
 );

SQL的新增

插入查询的数据

可以将查询的数据插入到另一个表中

image-20220803154659902

将student的数据插入到student2中

image-20220803155540206

将student中符合条件的数据插入到student2中

聚合查询

查询时带表达式--把行和行放到一起进行计算

聚合查询--SQL中提供了一些函数,通过这些函数,就能进行 行和行的计算

image-20220803195859460

image-20220803195912152

分组查询

group by 列名; -- 按照列名分组

根据查询结构,进行分组

根据列名,将值相同的记录分成一组,然后就可以用聚合函数进行数据的聚合了

建立一个表

image-20220805123822301

image-20220805123905460按照职位进行分组,针对分组后,求每一组关于工资的平均值

image-20220805124042535

按照职位进行分组,分组后,求每组工资的最大值

image-20220805124427766

image-20220805124649293

这样写展示的是每一组的第一个记录,其实是没有任何价值的

分组查询有两种情况

  1. 先条件筛选,再分组( where 条件 group by 列名)
  2. 先分组,后条件筛选 (group by 列名 having 条件)

举例:

1.计算各个岗位的平均薪资(除去张三)

这就是先除去张三,在进行分组计算

image-20220805130226942

2.计算平均薪资大于10000的岗位

先进行分组,再筛选薪资大于10000的岗位

image-20220805130327051

3.求除了张三之外,每个岗位的平均薪资,并且保留平均薪资大于10000的岗位

image-20220805130642011

联合查询/多表查询(重点)

要想理解多表查询,首先要知道笛卡尔积

笛卡尔积就是两个表的记录的排列组合

笛卡尔积的列数就是两张表的列数之和

笛卡尔积的行数就是两张表的行数之积

使用联合查询/多表查询的时候,要是表比较大,计算笛卡尔积的时候,就会十分低效,甚至成为"危险操作"

笛卡尔积是一个简单粗暴的操作,只是进行简单的所有的排列组合,所以这么多的记录是有些是合理的,有些是不合理的,所以要寻找两张表的共同点(连接条件),进行筛选,将不合理的记录删除掉

联合查询/多表查询 = 笛卡尔积 + 连接条件 + 其他条件 (根据具体的需求)

联合查询/多表查询的两种写法

select *(列名) from 表名,表名 where 连接条件; -- 内连接

select *(列名) from 表名 join 表名 on 连接条件; -- 外连接

笛卡尔积:image-20220805212949054

连接条件

注意:两个表的列名是可以相同的,多表查询的时候,要是两个表的列是相同的,就用表名.列名加以区分

image-20220805213446525

join on写法

image-20220805214311882

举例:

一共有学生表 课程表 成绩表 班级表 4张表

image-20220805215809965

查询许仙同学的成绩

分析:许仙在学生表中,成绩在成绩表中

可以借助多表查询

  1. 借助笛卡尔积,将score 和 student 进行排列组合
  2. 给笛卡尔积加上连接条件,筛掉不合理的记录
  3. 根据"许仙"这个名字进行筛选
  4. 选择要显示的列

image-20220805220611556

image-20220805221032459

image-20220805221054094

image-20220805221139296

刚开始接触的时候,还是要一点一点进行操作,慢慢写,方便理解也方便排查错误

使用join on写法

image-20220805221710422

查询所有同学的总成绩,以及个人信息

这里要求总成绩,此处的表是按照行来排列的,所以不能使用表达式累加来算,只能根据聚合函数sum

进行计算

还是按照上面的思路一步一步进行操作

  1. 笛卡尔积(成绩表 学生表)
  2. 连接条件(精简数据)
  3. 进行分组
  4. 计算总成绩(聚合函数--行与行的操作)

image-20220805222656529

image-20220805223223097

image-20220805223240329

image-20220805223335135

join on写法:

image-20220805223737364

查询出同学的成绩,带有课程名的那种

同学名字在学生表

课程名字在课程表

分数在分数表

  1. 先算笛卡尔积(三表连接--其实算的速度不慢,只是打印的速度慢)

  2. 寻找连接条件

  3. 选择要展示的列即可

    image-20220805224230059

image-20220805224517885

image-20220805224915033

join on 写法(两个表两个表地连接)

image-20220805225453924

内连接与外连接

select 列名 from 表名,表名 where 连接条件; 内连接

select 列名 from 表名 join 表名 on 连接条件; 使用 join 可以作为内连接也可以作为外连接,写作 left join on 就是左外连接 right join on就是有右外连接

image-20220806092252316

这样子的表是有相互照应体现的,在左表中的记录,在右表中的有体现

此时内连接与外连接是一样的

image-20220806092502233

左表中的3 王五 , 在右表中没有体现,右表中的4 在左表中也没有体现 -- 外连接

image-20220806093057201

image-20220806093104951

上面就是内连接,将两张表中有对应关系的记录整合在一起

image-20220806093247509

左连接会尽量将左侧表的数据完整地展示出来,右表无法体现就直接置为空

同理,右连接会尽量将右表的数据完整的展示出来,左表无法体现就直接置为空

image-20220806094702997

子查询

子查询其实就是套娃,利用上一步的结果作为下一步的条件

注意:上一步的结果必须只有一条记录

要是上一步的结果不止一条,就用in

举例:

寻找和"不想毕业"同班的同学

  1. 首先先确定"不想毕业"的班级号
  2. 利用上一步的班级号来查询

image-20220806104153724

image-20220806104314517

image-20220806125627735

合并查询

合并查询就是将查询结果哈合并在一起

union 合并(要是有重复的数据,会进行去重)

union all 合并(不去重,直接合并)

查询名字为"英文"的课程或者id<3

image-20220806130754148

当然,这样子也可以直接就用 or ,使用union只是多了一种选择

小练习:

image-20220806213645828

image-20220806213657184

查询男女员工的平均工资
select sex,avg(salary) from emp group by sex;

image-20220806213722006

 查询各部门的总薪水
 select depart,sum(salary) from emp group by depart;

image-20220806213751987

 查询总薪水排名第二的部门
 select depart,sum(salary) from emp  group by depart order by sum(salary) desc limit 1 offset 1;
 -- 先按照职位分组,然后按照薪资降序排列,最后只要排名第二的行

image-20220806214156479

第4题

查询姓名重复的员工信息
select depart,avg(salary) from emp  where   salary>10000 and  sex="男" group by depart;

image-20220806232734862

第5题

查询各部门薪水大于10000的男性员工的平均薪水
select depart,avg(salary) from emp  where   salary>10000 and  sex="男" group by depart;

image-20220806220641335

image-20220808104512550

select gpa from user_profile  where university = "复旦大学" order by gpa desc limit 1;-- 一定要先是条件  排序 分页这样的顺序
全部评论

相关推荐

在前端开发中,可以使用多种方式来测试前端模块的功能和性能。以下是几个常用的前端模块测试方法:https://www.nowcoder.com/issue/tutorial?zhuanlanId=Mg58Em&amp;uuid=ba06d8fbb87f45f7bf340c85dc4f0cc1单元测试:单元测试是针对代码的最小单元进行的测试,可以保证模块的某个具体功能的正常运作。在前端开发中,可以使用各种单元测试框架,如Jest、Mocha、Karma等来编写和运行单元测试。集成测试:集成测试是测试多个模块组合后的整体功能,或者测试模块与外部依赖(如接口、数据库等)的集成情况。可以使用工具如Selenium、Cypress等进行模拟用户操作和功能测试,并通过配置测试套件来测试整个应用的各个部分。性能测试:性能测试是为了验证模块在不同条件下的性能表现和稳定性。可以使用工具如WebPageTest、Lighthouse、JMeter等来测试网页的加载速度、渲染性能、资源使用情况等。UI测试:UI测试是为了验证模块的界面交互和可用性。可以使用工具如Puppeteer、Playwright、TestCafe等进行自动化测试,模拟用户操作和检查界面元素的状态和行为。在进行前端模块测试时,可以借助于这些工具和框架来编写测试代码和运行测试,以保证前端模块的质量和功能正常性。根据项目的需求,可以选择合适的测试方法和工具来进行测试。另外,还可以结合持续集成和持续部署的流程,将测试自动化,并与代码库的提交和构建过程集成,以保证项目在不断迭代中的质量和稳定性。
点赞 评论 收藏
分享
评论
2
15
分享
牛客网
牛客企业服务