文章目录
unit 01-MySQL
数据库概述
什么是数据库?
所谓的数据库就是指存储和管理数据的仓库
扩展内容1:数据库有哪些分类?(了解)
早期: 层次式数据库、网络型数据库
现在:关系型数据库、非关系型数据库
什么是关系型数据库?
底层以二维表的形式保存数据的库就是关系型数据库
stu-学生表
学生编号 | 姓名 | 年龄 |
---|---|---|
1001 | 刘沛霞 | 35 |
1002 | 陈子枢 | 18 |
扩展内容2:常见的关系型数据库有哪些?(了解)
-
Sql Server:微软提供,收费,适用于一些中型或大型的项目中,在java中的使用占比不高(.NET中使用的较多)
-
Oracle:甲骨文公司提供,收费,适用于一些大型或者超大型的项目中,在java中的使用占比非常高
-
mysql:瑞典r公司提供,免费开源,适用于一些小型或者中型的项目中,在Java中的使用占比较高(小巧轻量)
mariadb其实就是MySQL的一个分支,用法和MySQL完全一样。 -
DB2:IBM公司提供,收费,在一些银行、金融等行业中使用较多。在java中的使用占比也不高。
-
Sqlite:迷你数据库,嵌入式设备中(安卓、苹果手机、pad)
数据库相关概念
1、什么是数据库服务器
数据库服务器就是一个软件(比如mysql软件)将数据库软件安装在电脑上,当前电脑就是一个数据库服务器。就可以对外提供存取数据的服务
在一个数据库服务器中可以创建多个数据库(dataBases),每一个数据库都是一个单独的仓库。
2、什么是数据库
数据库就是存储和管理数据的仓库,通常情况下,一个网站的中的所有数据会存放在一个数据库中。例如:
jd.com db_jd(数据库)
taobao.com db_taobao(数据库)
...
3、什么是表
一个数据库中可以创建多张表,每张表用于存储一类信息(数据库),例如:
jd.com中的用户数据 tb_user(表)
jd.com中的商品数据 tb_product(表)
jd.com中的订单数据 tb_order(表)
…
4、什么表记录·
一张表中可以包含多行表记录,每一行表记录用于存储某一个具体的数据
学生编号 | 姓名 | 年龄 |
---|---|---|
1001 | 刘沛霞 | 35 |
1002 | 陈子枢 | 18 |
。。。 | 。。。 | 。。。 |
什么是SQL语言?
SQL是一们用于操作关系型数据库的通用的语言(使用SQL可以操作所有的关系型数据库)
使用SQL可以操作数据库、表、表记录
(1)创建数据库、删除数据库、修改数据库、查询数据库
(2)创建表、删除表、修改表、查询表
(3)新增表记录、删除表记录、修改表记录、查询表记录
使用SQL也可以操作存储过程/视图/索引等。
提示:SQL是一个标准通用的操作关系型数据库的语言(普通话),每个数据库厂商为了增强自己数据库的功能,都提供了支持自己数据库的语言,称之为数据库的方言。方言不通用!
连接mysql服务器
通过命令行工具可以登录MySQL客户端,连接MySQL服务器,从而访问服务器中的数据。
1、连接mysql服务器:
mysql -uroot -p密码
**-u:**后面的root是用户名,这里使用的是超级管理员root;
**-p:(小写的p)**后面的root是密码,这是在安装MySQL时就已经指定的密码;
2、连接mysql服务器并指定IP和端口:
mysql -uroot -proot -h127.0.0.1 -P3306
**-h:**后面给出的127.0.0.1是服务器主机名或ip地址,可以省略的,默认连接本机;
**-P:(大写的P)**后面的3306是连接端口,可以省略,默认连接3306端口;
3、退出客户端命令:quit或exit或 \q
4、FAQ:常见问题:
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-PZvSNUPD-1617701973310)(JAVAWEB-NOTE01.assets/e23a06d3921d5d68c48d6e51024aae8a.jpg)]
解决方法:复制mysql安装目录下的bin目录,将bin目录的路径添加到path环境变量中!!
扩展内容3:
(1)在cmd中连接mysql服务器之后,可以使用 #、/**/、-- 等符号添加注释,例如:
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-Veat5Jqy-1617701973318)(JAVAWEB-NOTE01.assets/7765164a8afaa8d26d9b31ff3356bc33.png)]
(2)在cmd中连接mysql服务器之后,在书写SQL语句时,可以通过 \c 取消当前语句的执行。例如:
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-e2bwhrXZ-1617701973321)(JAVAWEB-NOTE01.assets/4c6edbd263d6c870ac745e34d4b207d0.png)]
数据库及表操作
创建、删除、查看数据库
提示: (1)SQL语句对大小写不敏感。推荐关键字使用大写,自定义的名称(库名,表名,列名等)使用小写。
SHOW DATABASES; -- 查看当前数据库服务器中的所有库
CREATE DATABASE mydb1; -- 创建mydb1库
(2)并且在自定义名称时,针对多个单词不要使用驼峰命名,而是使用下划线连接。(例如:tab_name,而不是 tabName )
– 01.查看mysql服务器中所有数据库
show databases;
– 02.进入某一数据库(进入数据库后,才能操作库中的表和表记录)
– 语法:USE 库名;
use mysql;
– 查看已进入的库
SELECT DATABASE();
– 03.查看当前数据库中的所有表
-- 先进入某一个库
use mysql;
--再查看当前库中的所有表
show tables;
– 04.删除mydb1库
– 语法:DROP DATABASE 库名;
DROP DATABASE mysdb1;
-- 思考:当删除的库不存在时,如何避免错误产生?
DROP DATABASE IF EXISTS mydb1;
– 05.重新创建mydb1库,指定编码为utf8
– 语法:CREATE DATABASE 库名 CHARSET 编码
-- 如果存在mydb1,则先删除,再重建
drop database if exists mydb1;
create database mydb charset=utf8;
– 如果不存在则创建mydb1;
create database if not exists mydb1;
– 06.查看建库时的语句(并验证数据库库使用的编码)
– 语法:SHOW CREATE DATABASE 库名;
show create database mydb1;
创建、删除、查看表
– 07.进入mydb1库,删除stu学生表(如果存在)
– 语法:DROP TABLE 表名;
drop table if exists stu;
– 08.创建stu学生表(编号[数值类型]、姓名、性别、出生年月、考试成绩[浮点型]),建表的语法:
CREATE TABLE 表名(
列名 数据类型,
列名 数据类型,
...
列名 数据类型
);
SQL语句:
create table stu(id int default null,
name varchar(50) default null,
gender varchar(10) default null,
birthday date default null,
score double default null);
– 09.查看stu学生表结构
– 语法:desc 表名
desc stu;
![image-20200316141717739](JAVAWEB-NOTE01.assets/image-20200316141717739.png)
新增、更新、删除表记录
– 10.往学生表(stu)中插入记录(数据)
– 语法:INSERT INTO 表名(列名1,列名2,列名3…) VALUES(值1,值2,值3…);
-- 如果是在cmd中执行插入记录的语句,先 set names gbk; 再插入记录!
set names gbk;
insert into stu(
id,name,gender,birthday,score)
values
(1,'xubeihe','man','1995-12-16',90.9),
(1,'cai','woman','1997-12-16',90.9);
提示:
(1)当为所有列插入值时,可以省写列名,但值的个数和顺序必须和声明时列的个数和顺序保持一致!
(2)SQL语句中的值为字符串或日期时,值的两边要加上单引号(有的版本的数据库双引号也可以,但推荐使用单引号)。
(3)(针对cmd窗口)在插入数据之前,先设置编码:set names gbk;
或者用一下命令连接mysql服务器:
mysql --default-character-set=gbk -uroot -proot
– 11.查询stu表所有学生的信息
– 语法:SELECT 列名 | * FROM 表名
select * from stu;
– 12.修改stu表中所有学生的成绩,加10分特长分
– 修改语法: UPDATE 表名 SET 列=值,列=值,列=值…[WHERE子句];
update stu set score=score+10 where name='xubeihe';
– 13.修改stu表中编号为1的学生成绩,将成绩改为83分。
update stu set score=83 where id=1;
提示:where子句用于对记录进行筛选过滤,保留符合条件的记录,将不符合条件的记录剔除。
– 14.删除stu表中所有的记录不删除表
TRUNCATE TABLE 表名;
TRUNCATE table stu;
– 删除记录语法: DELETE FROM 表名 [where条件]
delete from stu;
– 仅删除符合条件的
delete from stu where id = 1;
查询表记录
– 准备数据: 以下练习将使用db10库中的表及表记录,请先进入db10数据库!!!
use db10;
基础查询
SELECT 语句用于从表中选取数据。结果被存储在一个结果表中(称为结果集)。
语法:SELECT 列名称 | * FROM 表名
提示:(1) *(星号)为通配符,表示查询所有列。
(2)但使用 *(星号)有时会把不必要的列也查出来了,并且效率不如直接指定列名
– 15.查询emp表中的所有员工,显示姓名,薪资,奖金
select name,sal,bonus from emp;
– 16.查询emp表中的所有部门和职位
select dept,job from emp;
思考:如果查询的结果中,存在大量重复的记录,如何剔除重复记录,只保留一条? */
– 在select之后、列名之前,使用DISTINCT 剔除重复的记录
select distinct dept,job from emp;
WHERE子句查询
WHERE子句查询语法:SELECT 列名称 | * FROM 表名称 WHERE 列 运算符 值
WHERE子句后面跟的是条件,条件可以有多个,多个条件之间用连接词(or | and)进行连接
下面的运算符可在 WHERE 子句中使用:
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-vA25FOHF-1617701973324)(JAVAWEB-NOTE01.assets/7b2b5598fb3af3f96d552b5d9779d892.png)]
– 17.查询emp表中薪资大于3000的所有员工,显示员工姓名、薪资
select name,sal from emp where sal>3000;
– 18.查询emp表中总薪资(薪资+奖金sal+bonus)大于3500的所有员工,显示员工姓名、总薪资
select name,sal+bonus from emp where sal+bonus>3500;
– ifnull(列bonus, 值)函数: 判断指定的列是否包含null值,如果有null值,用第二个值替换null值
select name,sal+ifnull(bonus,0) from emp where sal+ifnull(bonus,0)>3500;
– 注意查看上面查询结果中的表头,如何将表头中的 sal+bonus 修改为 “总薪资”
– 使用as可以为表头指定别名
select name,sal+bonus as 总薪资 from emp where (sal+bonus)>3500;
– 另外as可以省略
select name,sal+bonus 总薪资 from emp where (sal+bonus)>3500;
– 19.查询emp表中薪资在3000和4500之间的员工,显示员工姓名和薪资
select name,sal from emp where sal>3000 or sal<4500;
– 提示: between…and… 在…之间
select name,sal from emp where sal between 3500 and 4500;
– 20.查询emp表中薪资为 1400、1600、1800的员工,显示员工姓名和薪资
select name,sal from emp where sal=1400 and sal=1600 and sal=1800;
– 或者
select name,sal from emp where sal in(1400,1600,1800);
– 21.查询薪资不为1400、1600、1800的员工
方式一:
select name,sal from emp where sal!=1400 and sal!=1600 and sal!=1800;
方式二:
select name,sal from emp where not(sal=1400 or sal=1600 or sal=1800);
方式三:
select name,sal from emp where not in(1400,1600,1800);
– 22.(自己完成) 查询emp表中薪资大于4000和薪资小于2000的员工,显示员工姓名、薪资。
select name,sal from emp where sal>4000 or sal <2000;
– 23.(自己完成) 查询emp表中薪资大于3000并且奖金小于600的员工,显示员工姓名、薪资、奖金。
select name,sal,bonus from emp where sal>3000 and bonus<600;--不处理null值有误
– 处理null值
select name,sal,bonus from emp where sal>3000 and ifnull(bonus,0); --处理null值
– 24.查询没有部门的员工(即部门列为null值)
select name,dept from emp where dept is null;
– 思考:如何查询有部门的员工(即部门列不为null值)
方式一:
select * from emp where detp is not null;
–方式二:
select * from emp where not (detp is null);
模糊查询
LIKE 操作符用于在 WHERE 子句中搜索列中的指定模式。
可以和通配符(%、_)配合使用,其中"%“表示0或多个任意的字符。”_"表示一个任意的字符。
语法:SELECT 列 | * FROM 表名 WHERE 列名 LIKE 值
示例:
– 25.查询emp表中姓名中包含"涛"字的员工,显示员工姓名。
select * from emp where name like '%涛%';
– 26.查询emp表中姓名中以"刘"字开头的员工,显示员工姓名。
select name from emp where name like '刘%';
– 27.查询emp表中姓名以"刘"开头,并且姓名为两个字的员工,显示员工姓名。
select name from emp where name like '刘_';
多行函数查询
多行函数也叫做聚合(聚集)函数,根据某一列或所有列进行统计。
常见的多行函数有:
多行函数 | 作用 |
---|---|
COUNT( 列名 | * ) | 统计结果集中某一列或记录行的行数。 |
MAX( 列名 ) | 统计结果集中某一列值中的最大值 |
MIN( 列名 ) | 统计结果集中某一列值中的最小值 |
SUM( 列名 ) | 统计结果集中某一列所有值的和 |
AVG( 列名 ) | 统计结果集中某一列值的平均值 |
提示:(1)多行函数不能用在where子句中
(2)多行函数和是否分组有关,分组与否会直接影响多行函数执行的结果。
– 28.统计emp表中薪资大于3000的员工个数
select count(*) from emp where sal>300;
– 29.求emp表中的最高薪资
select sal from emp;-- 求所有薪资
select max(sal) from emp;-- 统计最高薪资
select min(sal) from emp;-- 统计最低薪资
select name,max(sal) from emp;-- 错误写法!name是整个结果中的第一个name,和最高薪资很可能不对应。
– 30.统计emp表中所有员工的薪资总和(不包含奖金)
select sum(sal) from emp;-- 统计薪资这一列中所有值的和
select sum(bonus) from emp;-- 统计奖金这一列中所有值的和,多行函数会处理null值
select sum(ifnull(bonus,0)) from emp;
– 31.统计emp表员工的平均薪资(不包含奖金)
select avg(sal) from emp;
select sum(sal)/ count(*) from emp;
多行函数需要注意的问题:
-
多行函数和是否分组有关,如果查询结果中的数据没有经过分组,默认整个查询结果是一个组,多行函数就会默认统计当前这一个组的数据。产生的结果只有一个。
-
如果查询结果中的数据经过分组(分的组不止一个),多行函数会根据分的组进行统计,有多少个组,就会统计出多少个结果。
例如:统计emp表中的人数(将emp表中的所有员工看作一个组)
select count(*) from emp;
结果返回的就是emp所有人数
再例如: 根据性别进行分组,再统计emp表中每组的人数,显示性别和对应人数
select gender,count(*) from emp group by gender;
分组查询
GROUP BY 语句根据一个或多个列对结果集进行分组。
在分组的列上我们可以使用 COUNT,SUM,AVG,MAX,MIN等函数。
语法:SELECT 列 | * FROM 表名 [WHERE子句] GROUP BY 列;
– 32.对emp表按照部门对员工进行分组,查看分组后效果。
select * from emp where group by dept;
– 33.对emp表按照职位进行分组,并统计每个职位的人数,显示职位和对应人数
select jop,count(*) from emp group by jop;
– 34.对emp表按照部门进行分组,求每个部门的最高薪资(不包含奖金),显示部门名称和最高薪资
select dept,max(sal) from emp group by dept;
排序查询
使用 ORDER BY 子句将结果集中记录根据指定的列排序后再返回
语法:SELECT 列名 FROM 表名 ORDER BY 列名 [ASC|DESC]
ASC(默认)升序,即从低到高;DESC 降序,即从高到低。
– 35.对emp表中所有员工的薪资进行升序(从低到高)排序,显示员工姓名、薪资。
select name,sal from emp order by sal asc;
– 36.对emp表中所有员工的奖金进行降序(从高到低)排序,显示员工姓名、奖金。
select name,bonus from emp order by bonus desc;
分页查询
在mysql中,通过limit进行分页查询:
limit (页码-1)*每页显示记录数, 每页显示记录数
– 37.查询emp表中的所有记录,分页显示:每页显示3条记录,返回第 1 页。
select * from emp limit 0,3;
– 38.查询emp表中的所有记录,分页显示:每页显示3条记录,返回第 2 页。
select * from emp limit 3,3;
– 补充练习:求emp表中薪资最高的前3名员工的信息,显示姓名和薪资
select name,sal from emp limit 0,3;
其他函数
函数名 | 解释说明 |
---|---|
curdate() | 获取当前日期,格式是:年月日 |
curtime() | 获取当前时间 ,格式是:时分秒 |
sysdate()/now() | 获取当前日期+时间,格式是:年月日 时分秒 |
year(date) | 返回date中的年份 |
month(date) | 返回date中的月份 |
day(date) | 返回date中的天数 |
hour(date) | 返回date中的小时 |
minute(date) | 返回date中的分钟 |
second(date) | 返回date中的秒 |
CONCAT(s1,s2…) | 将s1,s2 等多个字符串合并为一个字符串b |
CONCAT_WS(x,s1,s2…) | 同CONCAT(s1,s2,…)函数,但是每个字符串之间要加上x,x是分隔符 |
– 39.查询emp表中所有在1993和1995年之间出生的员工,显示姓名、出生日期。
select * from emp where birthday between 1993 and 1995;--错误的语法
select * from emp where birthday between '1993' and '1995';--错误的语法
select * from emp where birthday between '1993-1-1' and '1995-12-30';--正确
select * from emp where year(birthday) between 1993 and 1995;--正确
– 40.查询emp表中本月过生日b的所有员工
select * from emp where month(birthday) = month(curdate());
– 41.查询emp表中员工的姓名和薪资(薪资格式为: xxx(元) )
select name,concat(sal,'元') from emp;
– 补充练习:查询emp表中员工的姓名和薪资(薪资格式为: xxx/元 )
select name,concat_ws('/',sal,'元') from emp;
mysql的数据类型
数值类型
MySQL中支持多种整型,其实很大程度上是相同的,只是存储值的大小范围不同而已。
tinyint:占用1个字节,相对于java中的byte
smallint:占用2个字节,相对于java中的short
int:占用4个字节,相对于java中的int
bigint:占用8个字节,相对于java中的long
其次是浮点类型即:float和double类型
float:4字节单精度浮点类型,相对于java中的float
double:8字节双精度浮点类型,相对于java中的double
字符串类型
1、char(n) 定长字符串,最长255个字符。n表示字符数,例如:
– 创建user表,指定用户名为char类型,字符长度不超过10
create table if not exists user(
name cahr(10)
);
所谓的定长,是当插入的值长度小于指定的长度时,剩余的空间会用空格填充。(这样会浪费空间)
2、varchar(n) 变长字符串,最长不超过65535个字节,n表示字符数,一般超过255个字符,会使用text类型,例如:
iso8859-1码表:一个字符占用1个字节,1*n <= 65535, n最多等于 65535
utf8码表:一个中文汉字占用3个字节,3*n<=65535,n最多等于 65535/3
GBK码表:一个中文汉字占用2个字节,2*n<65535,n最多等于 65535/2
– 创建user表,指定用户名为varchar类型,长度不超过10
create table if not exists user(
name varchar(10)
)
所谓的不定长,是当插入的值长度小于指定的长度时,剩余的空间可以留给别的数据使用。(节省空间)
3、大文本(长文本)类型
最长65535个字节,一般超过255个字符列的会使用text。
– 创建user表:
create table if not exists user(
resume text
);
另,text也分多种,其中bigtext存储数据的长度约为4GB。
扩展内容3:(面试题)char(n)、varchar(n)、text都可以表示字符串类型,其区别在于:
(1)char(n)在保存数据时,如果存入的字符串长度小于指定的长度n,后面会用空格补全,因此可能会造成空间浪费,但是char类型的存储速度较varchar和text快。
因此char类型适合存储长度固定的数据,这样就不会有空间浪费,存储效率比后两者还快!
(2)varchar(n)保存数据时,按数据的真实长度存储,剩余的空间可以留给别的数据用,因此varchar不会浪费空间。
因此varchar适合存储长度不固定的数据,这样不会有空间的浪费。
(3)text是大文本类型,一般文本长度超过255个字符,就会使用text类型存储。
日期类型
date:年月日
time:时分秒
datetime:年月日 时分秒
timestamp:时间戳(实际存储的是一个时间毫秒值),与datetime存储日期格式相同。两者的区别是:
-
timestamp最大表示2038年,而datetime范围是1000~9999
-
timestamp在插入数据、修改数据时,可以自动更新成系统当前时间(后面用到时再做讲解)
mysql的字段约束
字段约束/列约束 --> 约束: 限制
主键约束
主键约束:如果为一个列添加了主键约束,那么这个列就是主键,主键的特点是唯一且不能为空。
主键的作用: 作为一个唯一标识,唯一的表示一条表记录(作用类似于人的身份证号,可以唯一的表示一个人一样。)
添加主键约束,例如将id设置为主键:
create table if not exists user3(
id int primary kk key
);
如果主键是数值类型,为了方便插入主键(并且保证插入数据时,主键不会因为重复而报错),可以设置一个主键自增策略。
主键自增策略是指:设置了自增策略的主键,可以在插入记录时,不给id赋值,只需要设置一个null值,数据库会自动为id分配一个值(AUTO_ 变量,默认从1开始,后面依次+1),这样既可以保证id是唯一的,也省去了设置id的麻烦。
将id主键设置为自增:
create table if not exists user4(
id int primary key auto_increment
);
非空约束
非空约束:如果为一个列添加了非空约束,那么这个列的值就不能为空,但可以重复。
添加非空约束,例如为password添加非空约束:
create table if not exists user5(
password varchar(20) not null
);
唯一约束
唯一约束:如果为一个列添加了唯一约束,那么这个列的值就必须是唯一的(即不能重复),但可以为空。
添加唯一约束,例如为username添加唯一(unique)约束及非空约束:
create table user(
username varchar(50) unique not null,
...
);
外键约束
外键其实就是用于通知数据库两张表数据之间对应关系的这样一个列。
这样数据库就会帮我们维护两张表中数据之间的关系。
(1) 创建表的同时添加外键
create table emp(
id int,
name varchar(50),
dept_id int,
foreign key(dept_id) references dept(id)
);
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-jLxrbcT9-1617701973328)(JAVAWEB-NOTE01.assets/6b56c23ba5cf6a63d9f9e8a8b5fd8467.png)]
(1)如果是要表示两张表的数据之间存在对应关系,只需要在其中的一张表中添加一个列,保存另外一张表的主键,就可以保存两张表数据之间的关系。
但是添加的这个列(dept_id)对于数据库来说就是一个普通列,数据不会知道两张表存在任何关系,因此数据库也不会帮我们维护这层关系。
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-CjWKmnCR-1617701973331)(JAVAWEB-NOTE01.assets/27084a70275c2ea84221346785d9821b.png)]
(2)如果将dept_id列设置为外键,等同于通知数据库,部门表和员工表之间存在对应关系,dept_id列中的数据要参考部门的主键,数据库一旦知道部门和员工表之间存在关系,就会帮我们维护这层关系型。
思考:如果在创建表时没有指定外键,那么后期该如何指定外键?以及如何删除外键?
表关系
常见的表关系分为以下三种:
一对多(多对一)、一对一、多对多
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-A1nclXoK-1617701973333)(JAVAWEB-NOTE01.assets/1e4a36daf517c88d356cbc3c8dbd3ea8.png)]
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-1gLccN8k-1617701973336)(JAVAWEB-NOTE01.assets/392ee4d4c0f8fdd86adc4b9abc3ad6f5.png)]
![](JAVAWEB-NOTE01.assets/55f5505ecfd67ae14e8c81da6ad7fd55.png)
多表查询
– 准备数据: 以下练习将使用db30库中的表及表记录,请先进入db30数据库!!!
连接查询
– 42.查询部门和部门对应的员工信息
select * from dept,emp;
上面的查询中存在大量错误的数据,一般我们不会直接使用这种查询。
笛卡尔积查询:所谓笛卡尔积查询就是指,查询两张表,其中一张表有m条记录,另一张表有n条记录,查询的结果是m*n条。
虽然笛卡尔积查询中包含大量错误数据,但我们可以通过where子句将错误数据剔除,保留下来的就是正确数据。
select * from dept,emp