关系数据库与Sql
Posted 化身天使
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了关系数据库与Sql相关的知识,希望对你有一定的参考价值。
1. 数据库连接
1.1 Oracle
1.1.1 命令行连接
sqlplus /nolog
conn username/password@IP:[1521]/数据库服务名 [as sysdba]
使用tnsname
vi $ORACLE_HOME/NETWORK/ADMIN/tnsnames.ora
ORCL = “orcl服务名”
(DESCRIPTION =
(ADDRESS = (PROTOCOL = TCP)(HOST = 主机IP地址)(PORT = 1521))
(CONNECT_DATA =
(SERVER = DEDICATED)
(SERVICE_NAME = ABC)
)
sqlplus /nolog
conn username@orcl #使用配置文件内的名字连接,配置文件需在默认位置
1.1.2 cx_oracle
1.2 mysql
1.2.1 命令行连接
连接数据库mysql -u maria -p
使用数据库 use hangzhou;
1.2.2 pymysql
1.3
df sf
2. 数据库管理
2.1 数据库查询
查看数据库 show databases;
查看当前使用的数据库 select database();
show tables;
查看数据库状态 status;
查看表结构 desc t1;
#连接当前系统连接数
select count(*) from v$process
#查询最大连接数设置
select value from v$parameter where name = \'processes\'
2.2 用户管理
创建用户 CREATE USER u1 IDENTIFIED BY ’pass’;
修改用户名 RENAME USER u1 TO u2;
修改密码 SET PASSWORD FOR u1 = Password(‘pass’);
修改自己密码 SET PASSWORD = PASSWORD(‘pass’);
删除用户 DROP USER u1;
查看当前登录用户 SELECT USER();
查看用户表 use mysql;
select user from user;
2.3 权限管理
查看自己的授权 show grants;
查看其他用户权限 show grants for u1;
授权 grant 权限1,权限2,select on 数据库.表 to u1;
更新授权 flush privileges;
在数据库test的所有表授予u1用户查询权限
grant select on test.* to u1;
收回权限 grant revoke 权限1 on 数据库.表 to u1;
创建用户同时授权
grant reload,lock tables,replication client on *.* to \'uback’@’localhost\' identified by \'uback\';
权限清单
usage 无权限
all 除进行授权本身外的所有权限
2.4 事务
创建保存点a savepoint a;
回滚到保存点 rollback to a;
回滚到上次保存点 rollback;
提交修改 commit;
2.5 定时任务
2.5.1 创建
declare
job number;
BEGIN
DBMS_JOB.SUBMIT(
JOB => job, /*自动生成JOB_ID*/
WHAT => \'testJob;\', /*需要执行的存储过程名称或SQL语句*/
NEXT_DATE => sysdate, /*初次执行时间-立即执行*/
INTERVAL => \'trunc(sysdate,\'\'mi\'\')+1/(24*60)\' /*每隔1分钟执行一次*/
);
commit;
end;
2.5.2 查询
select * from all_jobs where what like ‘%存储过程名字%’;
2.5.3 启动
declare
begin
DBMS_JOB.RUN(24); /*24 job的id*/
commit;
end
2.5.4 停止
declare
begin
dbms_job.broken(24,true,sysdate); /*停止一个job,jobId, job的ID,里面参数true也可是false,next_date(某一时刻停止)也可是sysdate(立刻停止)。 */
commit;
end;
2.5.5 删除
declare
begin
dbms_job.remove(24); /*删除自动执行的job,参数是 job的id*/
commit;
end;
2.5.6 修改
修改执行间隔
declare
begin
dbms_job.interval(24,interval => \'TRUNC(SYSDATE)+1\'); /*第一个参数为job的ID,第二个参数interval: 计算下一次任务执行的时间表达式*/
commit;
end;
修改下一次执行的时间
declare
begin
dbms_job.next_date(24,to_date(\'2020-11-9 12:08:00\',\'yyyy-mm-dd hh24:mi:ss\')); /*第一个参数:job的ID;第二个参数:要修改后的计算下一次执行的时间表达式*/
commit;
end;
更换要执行的存储过程
declare
begin
dbms_job.what(24,\'testJob2();\'); /* 第一个参数:job的ID;第二个参数:要更改的新操作名称(操作名必须存在)*/
commit;
end;
2.5.7 时间参数说明
INTERVAL 部分参数值示例:
每天午夜12点: \'TRUNC(SYSDATE + 1)\'
每天早上8点30分: \'TRUNC(SYSDATE + 1) + (8*60+30)/(24*60)\'
每星期二中午12点: \'NEXT_DAY(TRUNC(SYSDATE ), \'\'TUESDAY\'\' ) + 12/24\'
每个月第一天的午夜12点: \'TRUNC(LAST_DAY(SYSDATE ) + 1)\'
每个季度最后一天的晚上11点: \'TRUNC(ADD_MONTHS(SYSDATE + 2/24, 3 ), \'Q\' ) -1/24\'
每星期六和日早上6点10分: \'TRUNC(LEAST(NEXT_DAY(SYSDATE, \'\'SATURDAY"), NEXT_DAY(SYSDATE, "SUNDAY"))) + (6×60+10)/(24×60)\'
每月25号00:00执行: \'TRUNC(LAST_DAY(SYSDATE ) + 25)\'
--------------------------
1:每分钟执行
Interval => TRUNC(sysdate,\'mi\') + 1/ (24*60)
或
Interval => sysdate+1/1440
2:每天定时执行
例如:每天的凌晨1点执行
Interval => TRUNC(sysdate) + 1 +1/ (24)
3:每周定时执行
例如:每周一凌晨1点执行
Interval => TRUNC(next_day(sysdate,\'星期一\'))+1/24
4:每月定时执行
例如:每月1日凌晨1点执行
Interval =>TRUNC(LAST_DAY(SYSDATE))+1+1/24
5:每季度定时执行
例如每季度的第一天凌晨1点执行
Interval => TRUNC(ADD_MONTHS(SYSDATE,3),\'Q\') + 1/24
6:每半年定时执行
例如:每年7月1日和1月1日凌晨1点
Interval => ADD_MONTHS(trunc(sysdate,\'yyyy\'),6)+1/24
7:每年定时执行
例如:每年1月1日凌晨1点执行
Interval =>ADD_MONTHS(trunc(sysdate,\'yyyy\'),12)+1/24
2.5.8 job参数说明
dba_jobs 表中字段含义:
JOB 任务的唯一标识码
LOG_USER 提交任务的用户
PRIV_USER 赋予任务权限的用户
SCHEMA_USER 对用户作语法分析的用户模式
LAST_DATE 最后一次成功执行任务的时间
LAST_SEC 最后一次成功执行任务的时间的时分秒
THIS_DATE 正在执行的任务的开始时间,若没有则为空
THIS_SEC 正在执行的任务的开始时间的时分秒,若没有则为空
NEXT_DATE 下一次执行定时任务的时间
NEXT_SEC 下一次执行定时任务的时间的时分秒
TOTAL_TIME 执行当前任务所需要的时间,单位:秒
BROKEN 标志参数,Y表示任务中断,以后不会再运行
INTERTAL 计算下一次执行定时任务的时间表达式
FAILURES 当前定时任务执行失败的总次数
WHAT 执行任务的PL/SQL代码块
NLS_ENV 任务执行的NLS会话设置
MISC_ENV 定时任务运行的其他一些参数设置
INSTANCE 标识当前任务运行是否受限,0 没有受限
3. 表操作
3.1 建表
创建表空间
create tablespace abc
datafile ‘/u01/app/oracle/oradata/TestDB11/catalog.dbf’ 自定义数据文件位置及名字
size 100M #大小
extent management local autoallocate 扩展自动管理
segment space management auto;
创建表
CREATE TABLE t1 (
c1 varchar2(50) PRIMARY KEY 主键
c2 date REFERENCES t2(c2) 外键关联到t2的c2列
c3 date UNIQUE 唯一约束,该列的值不可以重复
c4 date not null 非空
c5 date )
TABLESPACE abc; 指定表空间
复制创建表
CREATE TABLE test TABLESPACE abc AS SELECT * FROM test2;
仅复制表结构
CREATE TABLE tes AS SELECT * FROM test2 where rownum<1;
3.2 新增
增加列 ALTER TABLE test ADD hobby nvarchar2(20);
添加索引 CREATE INDEX index1 ON t1(c1)
添加唯一索引 CREATE UNIQUE INDEX index1 ON t1(c1)
添加主键 ALTER TABLE t1 ADD PRIMARY KEY(字段名);
添加主键 ALTER TABLE test ADD CONSTRAINT test_pk PRIMARY KEY(ename);
添加外键 ALTER TABLE t ADD CONSTRAINT f3 FOREIGN KEY(c2) REFERENCES t3(id);
//外键条件,一个表的非主键连接另一个表的主键
添加唯一约束 ALTER TABLE test ADD UNIQUE(id);
check约束 ALTER TABLE test ADD CONSTRAINT emp_chk CHECK(empno>7000);
3.3 删除
#删除表空间
drop tablespace abc
including contents 同时删除表空间中的数据
cascade constraints; 同时删除完整性限制
#删除表
DROP TABLE test1
CASCADE CONSTRAINTS PURGE; #同时删除引用及数据文件
删除列 ALTER TABLE test DROP COLUMN hobby;
删除唯一约束 ALTER TABLE test DROP UNIQUE(id); #列名
删除约束 ALTER TABLE test DROP CONSTRAINT emp_chk ; #约束的名字
删除主键 ALTER TABLE t1 DROP PRIMARY KEY ;
删除主键 ALTER TABLE test DROP CONSTRAINT id_pk; #主键名字
删除外键 ALTER TABLE new DROP CONSTRAINT new_test3; #外键名字
3.4 修改
改表名 ALTER TABLE t6 RENAME TO t3;
改列名 ALTER TABLE test RENAME COLUMN id TO id1;
改列数据类型 ALTER TABLE t MODIFY id integer;
设置列非空 ALTER TABLE test MODIFY ename NOT NULL;
设置列可以为空 ALTER TABLE test MODIFY ename NULL;
列设置默认值 ALTER TABLE test MODIFY(id DEFAULT 22);
3.5 查询
获取表的全部字段
select wm_concat(column_name) from user_tab_columns where table_name=upper(\'tableName\')
查询表字段类型
select * from all_tab_cols where table_name=upper(\'表名\')
查询建表语句(可以看字段类型)
select dbms_metadata.get_ddl(\'TABLE\',upper(\'表名\')) from dual;
查询索引
select * from all_ind_columns where table_name=upper(\'表名\')
4. 数据类型
5. 插入行
INSERT INTO test VALUES(\'smith\',7369,5);
INSERT INTO t1(c1,c2,c3) VALUES(’abc’,’123’,’hahaha’);
从另一个表查询数据插入
INSERT INTO test(id,ename) SELECT id,ename FROM emp;
insert into tab1 select * from tab2 #一个表全部数据插入另一个表
Merge into存在更新,不存在插入,
merge into yytbt t1
using yytb_tmp t2
on (t1.编号=t2.编号)
when matched then
update set t1.a=t2.a
when not matched then
insert (c1,c2,c3)
values (t2.a,t2.b,t2.c) \'\'\'
不能更新on连接的字段,有时update出错,需带where字句
6. 删除行
删除某条记录 DELETE FROM t1 WHERE id=5;
删除全部行,可撤销 DELETE FROM test;
删除全部行,不可撤销 TRUNCATE TABLE t1;
删除指定列重复的数据
delete from t1 where rowid not in (select max(rowId) from t1 group by user_Id)
7. 更新列
更新单字段
UPDATE t1 SET c2=’123’ WHERE c1=young;
更新多字段
update t1 set (字段1, c2) = (值1, v2) WHERE c1=young;
忽略错误,更新能正常更新的行
UPDATE IGNORE t1 set
使用子查询更新
UPDATE test SET id =(SELECT id FROM t2 WHERE eno=7369) WHERE name=’smith’;
清空列内容 UPDATE test SET id=null;
8. 数据库及表信息查询
8.1 数据库查询
查看数据库 show databases;
查看当前使用的数据库 select database();
show tables;
查看数据库状态 status;
查看表结构 desc t1;
#连接当前系统连接数
select count(*) from v$process
#查询最大连接数设置
select value from v$parameter where name = \'processes\'
8.2 表查询
获取表的全部字段
select wm_concat(column_name) from user_tab_columns where table_name=upper(\'tableName\')
查询表字段类型
select * from all_tab_cols where table_name=upper(\'表名\')
查询建表语句(可以看字段类型)
select dbms_metadata.get_ddl(\'TABLE\',upper(\'表名\')) from dual;
查询索引
select * from all_ind_columns where table_name=upper(\'表名\')
select * from all_indexes where table_name = upper(\'表名\') ; --更详细
查询更新时间等
select * from ALL_OBJECTS where object_name =upper(\'表名\')
9. 查询select
9.1 基础查询
9.1.1 去重
单字段 select distinct c1 from t1;
多字段 select distinct(c1,c2) from t1;
查询某个字段不重复的全部记录
SELECT * FROM t1 WHERE rowid IN (SELECT max(rowid) FROM t1 WHERE lng IS NOT NULL GROUP BY channel_id )
9.1.2 模糊查询
like
where c1 like ‘_alo%’; not like
_ 单个字符
% 零或多个字符,无法匹配null
[abc] 其中一个字符
[^abc] 不在其中的字符
[!abc] 同上
regexp_like 正则表达式
where regexp_like(列名c1,\'\\d+$\'); 匹配出0到多个数字结尾的项
9.1.3 限制返回行数
Oracle查询前6行 select c1 from t where rownum<6; 符号只能< ,<= ,!=
Mysql查询前6行 select c1 from t limit 6; 计数从0开始
Mysql查询从第7行开始的6行
select c1 from t1 limit 6,6;
Oracle查询第6行以后的数据
SELECT c1 ,c2 from ( SELECT ROWNUM N, c1 ,c2 FROM t1 ) WHERE N>6
注:内层rownum 一定要用别名, 不然会冲突出错
9.1.4 日期查询
字符串转时间to_date(c1, \'YYYY-MM-DD HH24:MI:SS\' )
时间转字符串 to_char(时间列, \'YYYY-MM-DD\' )
时间列查询 20210930以后的数据
where to_char(时间列, \'YYYY-MM-DD\') > 20210930
9.1.5 表连接与子查询
INNER JOIN
SELECT a.a1,b.b1 FROM a,b WHERE a.a2=b.b2;
SELECT a.a1,b.b1 FROM a JOIN b ON a.a2=b.b2;
full join 全连接
left join,左连接
right join
union
将两个select语句的查询结果合并到一起,表1的数据在上,对应列的数据类型要相同,union得到的结果会去重,
union all
同上,不去重
嵌套子查询
先执行括号内的子查询,再执行父查询
相关子查询
先执行父查询,父查询的结果逐行传递给子查询执行.即父查询的每一行都执行一遍子查询
select c1,c2,c3 from t1 where c3>
(select avg(c4) from t1 where c1=t1.c1)
9.1.6 列转行
把查询出的某列显示在一行 select wm_concat(c1) from t1
根据分组把某列展示为多行 select wm_concat(c1) from t1 group by c2
9.1.7 操作符
IN:
WHERE LastName IN (\'Adams\',\'Carter\');
WHERE LastName IN (select c2 from t2);
NOT IN
IN 先查询外表(左侧),再查询内表(右侧)
比or快,可以代替or
EXISTS:
WHERE LastName EXISTS (select c2 from t2);
EXISTS 先查询内表(右侧),再查外表
用NOT EXISTS 代替NOT IN ,
BETWEEN区间:
WHERE Name BETWEEN \'Adams\' AND \'Carter\'
NOT BETWEEN (数值,文本,日期等)
空值 where id is null; is not null
9.2 字符串函数
删除指定字符(删除c1字段中的楼和层, a可以是任意c1中不存在的字符)
select translate(c1 ,\'a楼层\',\'a\') from t1
获取指定字符的位置,instr
instr(c1,’字符’)
instr(c1,’字符’,3,4)
3起始位置, 缺省1
4匹配次数, 缺省全部
拆分列(以-符号拆分c1列, -之前c2, 之后c3)
select c1,
substr(c1, instr(c1, \'-\')-1) as c2
substr(c1, instr(c1, \'-\')+1) as c3
from t1;
正则表达式
1.匹配出c1列开头的数字
select regexp_substr(c1 , \'^\\d+\') from t1;
9.3
数据库系统原理之SQL与关系数据库基本操作
SQL与关系数据库基本操作
第一节 SQL概述
结构化查询语言(Structured Query Language,SQL)是一种专门用来与数据库通信的语言,它可以帮助用户操作关系数据库。
一、SQL的发展
- 从20世纪80年代以来,SQL一直是关系数据库管理系统(RDBMS)的标准语言。
- SQL-89 SQL-92(SQL2) SQL-99(SQL3)
- 目前没有一个数据库系统能够支持SQL标准的全部概念和特性。
二、SQL的特点
- SQL不是某个特定数据库供应商专有的语言
- SQL简单易学
- SQL尽管看上去很简单,但它实际上是一种强有力的语言,灵活使用其语言元素,可以进行非常复杂和高级的数据库操作
- SQL语句不区分大小写(开发人员习惯于对所有SQL关键字使用大写,对所有列和表的名称使用小写)
三、SQL的组成
SQL集数据查询(Data Query)、数据定义(Data Definition)、数据操纵(Data Manipulation)和数据控制(Data Control)四大功能于一体,其核心主要包含有以下几个部门:
- 数据定义语言(Data Definition Language,DDL):主要用于对数据库及数据库中的各种对象进行创建、删除、修改等操作。其中数据库对象主要有表、默认约束、规则、视图、触发器、存储过程等。
- CREATE:用于创建数据库或数据库对象
- ALTER:用于对数据库或数据库对象进行修改
- DROP:用于删除数据库或数据库对象
- 数据操纵语言(Data Manipulation Language,DML):主要用于操纵数据库中各种对象、特别是检索和修改数据。
- SELECT:用于从表或视图中检索数据,其是数据库中使用最为频繁的SQL语句之一
- INSERT:用于将数据插入到表或视图中
- UPDATE:用于修改表或视图中的数据,其即可修改表或视图中一行数据,也可同时修改多行或全部数据
- DELETE:用于从表或视图中删除数据,其中可根据条件删除指定的数据
- 数据控制语言(Data Control Language,DCL):主要用于安全管理,例如确定哪些用户可以查看或修改数据库中的数据。
- GRANT:用于授予权限,可把语句许可或对象许可的权限授予其他用户或角色
- REVOKE:用于收回权限,其功能与GRANT相反,但不影响该用户或角色从其他角色中作为成员继承许可权限
- 嵌入式和动态SQL规则:规定了SQL语句在高级程序设计语言中使用的规范方法,以便适应较为复杂的应用。
- SQL调用和会话规则:SQL调用包括SQL例程和调用规则,以便提高SQL的灵活性、有效性、共享性以及使SQL具有更多的高级语言的特征。SQL会话规则则可使应用程序连接到多个SQL服务器中的某一个,并与之交互。
第二节 MySQL预备知识
MySQL是一个关系数据库管理系统(RDBMS),它具有客户/服务器体系结构 ,最初是由瑞典MySQL AB公司开发。
一、MySQL使用基础
- LAMP(Linux+Apache+MySQL+PHP/Perl/Python),即使用Linux作为操作系统,Apache作为Web服务器,MySQL作为数据库管理系统,PHP、Perl或Python语言作为服务器端脚本解释器。
- WAMP(Windows+Apache+MySQL+PHP/Perl/Python),即使用Windows作为操作系统,Apache作为Web服务器,MySQL作为数据库管理系统,PHP、Perl或Python语言作为服务器端脚本解释器。
二、MySQL中的SQL
MySQL在SQL标准的基础上增加了部分扩展的语言要素:
- 常量:是指在程序运行过程中值不变的量,也称为字面值或标量值
- 常量的使用格式取决于值的数据类型,可分为字符串常量、数值常量、十六进制常量、时间日期常量、位字段值、布尔值和NULL值
- 变量:用于临时存储数据,变量中的数据可以随着程序的运行而变化。
- 变量有名字和数据类型两个属性
- 变量的名字用于标识变量
- 变量的数据类型用于确定变量中存储数值的格式和可执行的运算
- 在MySQL中,变量分为用户变量和系统变量
- 用户变量前添加一个“@”
- 系统变量前添加两个“@”
- 变量有名字和数据类型两个属性
- 运算法
- 算术运算符:+(加)、-(减)、*(乘)、/(除)和%(求模)5种运算
- 位运算符:&(位与)、|(位或)、^(位异或)、~(位取反)、>>(位右移)、<<(位左移)
- 比较运算符:=(等于)、>(大于)、<(小于)、>=(大于等于)、<=(小于等于)、<>(不等于)、!=(不等于)、<=>(相等或都等于空)
- 逻辑运算符:NOT或!(逻辑非)、AND或&&(逻辑与)、OR或||(逻辑或)、XOR(逻辑异或)
- 表达式:是常量、变量、列名、复杂计算、运算符和函数的组合
- 根据表达式的值的数据类型,表示式可分为字符型表达式、数值型表达式和日期表达式
- 内置函数
- 数学函数,例如ABS()函数、SORT()函数
- 聚合函数,例如COUNT()函数
- 字符串函数,例如ASCII()函数、CHAR()函数
- 日期和时间函数,例如NOW()函数、YEAR()函数
- 加密函数,例如FNCODE()函数、ENCRYPT()函数
- 控制流程函数,例如IF()函数、IFNULL()函数
- 格式化函数,例如FORMAT()函数
- 类型转换函数,例如CAST()函数
- 系统信息函数,例如USER()函数、VERSION()函数
第三节 数据定义
SQL标准不提供修改数据库模式定义和修改视图定义的操作,用户如果需要修改这些对象,可先将它们删除然后再重建,或者也可以使用具体的关系数据库关系系统所提供的扩展语句来实现。
SQL标准没有提供索引相关的语句。
SQL标准提供的数据定义语句
操作对象 | 创建 | 删除 | 修改 |
---|---|---|---|
模式 | CREATE SCHEMA 语句 | DROP SCHEMA 语句 | |
表 | CREATE TABLE 语句 | DROP TABLE 语句 | ALTER TABLE 语句 |
视图 | CREATE VIEW 语句 | DROP VIEW 语句 |
一、数据库模式定义
- 创建数据库
CREATE DATABASE|SCHEMA[IF NOT EXISTS] db_name
[DEFAULT] CHARACTER SET [=] charset_name
|[DEFAULT] COLLATE [=] collation_name
此语法说明:
"[]" 标示其内容为可选项
"|" 用于分隔花括号中的选择项,表示可任选其中一项来与花括号外的语法成分共同组成SQL语句命令,即选项彼此间是“或”的关系
"db_name" 用于标示具体的数据库命名,且该数据库名必须符号操作系统文件夹命名规则,在MySQL中不区分大小写
"DEFAULT" 指定默认值
"CHARACTER SET" 指定数据库字符集(Charset)
"COLLATE" 指定字符集的校对规则
"IF NOT EXISTS" 用于在创建数据库前进行判断,只有该数据库目前尚不存在时才执行CREATE DATABASE操作,即此选项可以避免出现数据库已经存在而再新建的错误
在MySQL中创建一个名为mysql_test的数据库
➜ mysql -uroot -p
Enter password:
Welcome to the MySQL monitor. Commands end with ; or \\g.
Your MySQL connection id is 1370
Server version: 8.0.32 Homebrew
Copyright (c) 2000, 2023, Oracle and/or its affiliates.
Oracle is a registered trademark of Oracle Corporation and/or its
affiliates. Other names may be trademarks of their respective
owners.
Type \'help;\' or \'\\h\' for help. Type \'\\c\' to clear the current input statement.
mysql> create database if not exists mysql_test;
Query OK, 1 row affected (0.03 sec)
mysql> show databases;
+--------------------+
| Database |
+--------------------+
| information_schema |
| mysql |
| mysql_test |
| performance_schema |
| sys |
| win_pro |
| win_pro_test |
+--------------------+
7 rows in set (0.01 sec)
mysql>
- 选择数据库
USE db_name;
只有使用USE命令指定某个数据库为当前数据库之后,才能对该数据库及其存储的数据对象执行各种后续操作
- 修改稿数据库
ALTER DATABASE | SCHEMA [db_name]
alter_specification ...
修改已有数据库mysql_test的默认字符集和校对规则
mysql> use mysql_test;
Database changed
mysql> alter database mysql_test
-> default character set gb2312
-> default collate gb2312_chinese_ci;
Query OK, 1 row affected (0.02 sec)
mysql>
- 删除数据库
DROP DATABASE|SCHEMA [IF EXISTS] db_name
分别不使用和使用关键字”IF EXISTS“删除一个系统中尚未创建的数据库”mytest“
mysql> drop database mytest;
ERROR 1008 (HY000): Can\'t drop database \'mytest\'; database doesn\'t exist
mysql> drop database if exists mytest;
Query OK, 0 rows affected, 1 warning (0.00 sec)
mysql>
可选项”IF EXISTS“可以避免删除不存在的数据库时出现的MySQL错误信息。
使用DROP DATABASE 或 DROP SCHEMA 语句会删除指定的整个数据库,该数据库中的所有表(包括其中的数据)也将永久删除,因而使用该语句时,需谨慎,以免错误删除。
- 查看数据库
SHOW DATABASES|SCHEMAS
[LIKE \'pattern\'|WHERE expr]
"LIKE" 匹配指定的数据库名称
"WHERE" 指定数据库名称查询范围的条件
使用SHOW DATABASES或SHOW SCHEMAS语句,只会列出当前用户权限范围内所能查看到的所有数据库名称。
列出当前用户可查看的数据库列表
mysql> show databases;
+--------------------+
| Database |
+--------------------+
| information_schema |
| mysql |
| mysql_test |
| performance_schema |
| sys |
| win_pro |
| win_pro_test |
+--------------------+
7 rows in set (0.00 sec)
mysql>
二、表定义
数据表是关系数据库中最重要、最基本的数据对象,也是数据存储的基本单位。
数据表被定义为字段的集合,数据在表中是按照行和列的格式来存储的,每一行代表一条记录,每一列代表记录中一个字段的取值。
创建数据表的过程,实质上就是定义每个字段的过程,同时也是实施数据完整性约束的过程。
-
创建表
CREATE [TEMPORARY] TABLE tbl_name
(
字段名1 数据类型 [列级完整性约束条件] [默认值]
[, 字段名2 数据类型 [列级完整性约束条件] [默认值]]
[, ......]
[, 表级完整性约束条件]
) [ENGINE=引擎类型];
在一个已有数据库mysql_test中新建一个包含客户姓名、性别、地址、联系方式等内容的客户基本信息表,要求将客户的id号指定为该表的主键。
mysql> use mysql_test;
Database changed
mysql> create table customers
-> (
-> cust_id int not null auto_increment,
-> cust_name char(50) not null,
-> cust_sex char(1) not null default 0,
-> cust_address char(50) null,
-> cust_contact char(50) null,
-> primary key (cust_id)
-> );
Query OK, 0 rows affected (0.04 sec)
mysql> show tables;
+----------------------+
| Tables_in_mysql_test |
+----------------------+
| customers |
+----------------------+
1 row in set (0.01 sec)
mysql>
- 临时表与持久表
- 添加可选性“TEMPORARY”关键字创建的表为临时表,否则为持久表。
- 临时表的生命周期较短,只对创建它的用户可见,当断开与该数据库的连接时,MySQL会自动删除。
- 两个不同的连接可以使用相同的临时表名称,同时两个临时表不会互相冲突,也不与原有的同名的非临时表冲突。
- 数据类型
- 数据类型是指系统中所允许的数据的类型。
- 数据库中每个列都应有适当的数据类型,用于限制或允许该列中存储的数据。
- 在创建表时必须为每个表列指定正确的数据类型及可能得数据长度。
- 在MySQL中,主要的数据类型包括数值类型、日期和时间类型、字符串类型、空间数据类型等。
- 关键字 AUTO_INCREMENT
- 关键字“AUTO_INCREMENT”可以为表中数据类型为整型的列设置自增属性。
- 顺序是从数字1开始。
- 每个表只能有一个AUTO_INCREMENT列,并且它必须被索引。
- 其值是可以被覆盖的。
- 指定默认值
- 在MySQL中,默认值使用关键字“DEFAULT”来指定。
- 如果没有为列指定默认值,MySQL会自动为其分配一个。
- 如若该列可以去NULL值,则默认值为NULL。
- 如若该列被定义为NOT NULL,则默认值取决于该列的类型。
- NULL值
- NULL值是指没有值或缺值。
- 在MySQL中是通过关键字“NULL”来指定。
- 允许NULL的列,可以在插入行时不给出该列的值。
- 不能将NULL值与空串相混淆,NULL值是没有值,它不是空串。
- 主键
- 通过 PRIMARY KEY 关键字来指定。
- 主键值必须唯一,即表中的每个行必须具有唯一的主键值,而且主键一定要为NOT NULL。
- 如果主键使用单个列,则它的值必须唯一。
- 如果使用多个列,则这些列的组合值必须唯一。
-
更新表
-
ADD [COLUMN] 子句 (向表中增加新列,且其可同时增加多个列)
- 通过“FIRST”关键字将新列作为原表的第一列
- 若不指定关键字(after/first),则新列会添加到原表的最后
- 可以在ALTER TABLE 语句中通过使用 ADDPRIMARY KEY 子句、ADDFOREIGN KEY 子句、ADD INDEX 子句为原表添加一个主键、外键和索引等
向数据库mysql_test的表customers中添加一列,并命名为cust_city,用于描述用户所在的城市,要求其不能为NULL,默认值为字符串’Wuhan‘,且该列位于原表cust_sex列之后
- SQL语句
mysql> alter table mysql_test.customers
-> add column cust_city char(10) not null default \'Wuhan\' after cust_sex;
Query OK, 0 rows affected (0.02 sec)
Records: 0 Duplicates: 0 Warnings: 0
mysql> desc customers;
+--------------+----------+------+-----+---------+----------------+
| Field | Type | Null | Key | Default | Extra |
+--------------+----------+------+-----+---------+----------------+
| cust_id | int | NO | PRI | NULL | auto_increment |
| cust_name | char(50) | NO | | NULL | |
| cust_sex | char(1) | NO | | 0 | |
| cust_city | char(10) | NO | | Wuhan | |
| cust_address | char(50) | YES | | NULL | |
| cust_contact | char(50) | YES | | NULL | |
+--------------+----------+------+-----+---------+----------------+
6 rows in set (0.00 sec)
mysql>
-
CHANGE [COLUMN] 子句
- 可同时修改表中指定列的名称和数据类型
- 可同时放入多个子句,只需彼此间用逗号分隔
将数据库mysql_test中表customers的cust_sex列重命名为sex,且将其数据类型更改为字符长度为1的字符数据类型char(1),允许其为NULL,默认值为字符常量’M‘
mysql> alter table mysql_test.customers
-> change column cust_sex sex char(1) null default \'M\';
Query OK, 0 rows affected (0.03 sec)
Records: 0 Duplicates: 0 Warnings: 0
mysql> desc customers;
+--------------+----------+------+-----+---------+----------------+
| Field | Type | Null | Key | Default | Extra |
+--------------+----------+------+-----+---------+----------------+
| cust_id | int | NO | PRI | NULL | auto_increment |
| cust_name | char(50) | NO | | NULL | |
| sex | char(1) | YES | | M | |
| cust_city | char(10) | NO | | Wuhan | |
| cust_address | char(50) | YES | | NULL | |
| cust_contact | char(50) | YES | | NULL | |
+--------------+----------+------+-----+---------+----------------+
6 rows in set (0.01 sec)
mysql>
- ALTER [COLUMN] 子句
- 修改或删除表中指定列的默认值
将数据库mysql_test中表customers的cust_city列的默认值修改为字符常量’Beijing‘
mysql> alter table mysql_test.customers
-> alter column cust_city set default \'Beijing\';
Query OK, 0 rows affected (0.02 sec)
Records: 0 Duplicates: 0 Warnings: 0
mysql> desc customers;
+--------------+----------+------+-----+---------+----------------+
| Field | Type | Null | Key | Default | Extra |
+--------------+----------+------+-----+---------+----------------+
| cust_id | int | NO | PRI | NULL | auto_increment |
| cust_name | char(50) | NO | | NULL | |
| sex | char(1) | YES | | M | |
| cust_city | char(10) | NO | | Beijing | |
| cust_address | char(50) | YES | | NULL | |
| cust_contact | char(50) | YES | | NULL | |
+--------------+----------+------+-----+---------+----------------+
6 rows in set (0.01 sec)
mysql>
- MODIFY [COLUMN] 子句
- 只会修改指定列的数据类型,而不会干涉它的列名
- 可以通过关键字“FIRST”或“AFTER”修改指定列在表中的位置
将数据库mysql_test中表customers的cust_name列的数据类型由之前的字符长度为50的定长字符数据类型char(50)更改为字符长度为20的定长字符数据类型char(20),并将此列设置成表的第一列。
mysql> alter table mysql_test.customers
-> modify column cust_name char(20) first;
Query OK, 0 rows affected (0.04 sec)
Records: 0 Duplicates: 0 Warnings: 0
mysql> desc customers;
+--------------+----------+------+-----+---------+----------------+
| Field | Type | Null | Key | Default | Extra |
+--------------+----------+------+-----+---------+----------------+
| cust_name | char(20) | YES | | NULL | |
| cust_id | int | NO | PRI | NULL | auto_increment |
| sex | char(1) | YES | | M | |
| cust_city | char(10) | NO | | Beijing | |
| cust_address | char(50) | YES | | NULL | |
| cust_contact | char(50) | YES | | NULL | |
+--------------+----------+------+-----+---------+----------------+
6 rows in set (0.01 sec)
mysql>
- DROP [COLUMN] 子句
删除数据库mysql_test中表custmoers的 cust_contact列
mysql> alter table mysql_test.customers
-> drop column cust_contact;
Query OK, 0 rows affected (0.02 sec)
Records: 0 Duplicates: 0 Warnings: 0
mysql> desc customers;
+--------------+----------+------+-----+---------+----------------+
| Field | Type | Null | Key | Default | Extra |
+--------------+----------+------+-----+---------+----------------+
| cust_name | char(20) | YES | | NULL | |
| cust_id | int | NO | PRI | NULL | auto_increment |
| sex | char(1) | YES | | M | |
| cust_city | char(10) | NO | | Beijing | |
| cust_address | char(50) | YES | | NULL | |
+--------------+----------+------+-----+---------+----------------+
5 rows in set (0.01 sec)
mysql>
- RENAME [TO] 子句
- 为表重新赋予一个表名
使用RENAME [TO] 子句,重命名数据库mysql_test中表customers的表名为backup_customers
mysql> show databases;
+--------------------+
| Database |
+--------------------+
| information_schema |
| mysql |
| mysql_test |
| performance_schema |
| sys |
| win_pro |
| win_pro_test |
+--------------------+
7 rows in set (0.01 sec)
mysql> use mysql_test;
Reading table information for completion of table and column names
You can turn off this feature to get a quicker startup with -A
Database changed
mysql> show tables;
+----------------------+
| Tables_in_mysql_test |
+----------------------+
| customers |
+----------------------+
1 row in set (0.00 sec)
mysql> alter table mysql_test.customers
-> rename to mysql_test.backup_customers;
Query OK, 0 rows affected (0.02 sec)
mysql> show tables;
+----------------------+
| Tables_in_mysql_test |
+----------------------+
| backup_customers |
+----------------------+
1 row in set (0.00 sec)
mysql>
-
重命名表
用RENAME TABLE 语句来更改表的名字,并可同时重命名多个表
RENAME TABLE tbl_name TO new_tal_name
[, tbl_name2 TO new_tbl_name2] ...
使用RENAME TABLE 语句,将表backup_customers命名为customers
mysql> rename table mysql_test.backup_customers to mysql_test.customers;
Query OK, 0 rows affected (0.01 sec)
mysql> show tables;
+----------------------+
| Tables_in_mysql_test |
+----------------------+
| customers |
+----------------------+
1 row in set (0.00 sec)
mysql>
-
删除表
DROP [TEMPORARY] TABLE [IF EXISTS]
tbl_name [, tbl_name] ...
[RESTRICT | CASCADE]
DROP TABLE 语句可以同时删除多个表(包括临时表),但操作者必须拥有该命令的权限
当表被删除时,其中存储的数据和分区信息均会被删除,所以使用该语句须格外小心,但操作者在该表上的权限并不会自动被删除
-
查看表
(1)显示表的名称
SHOW [FULL] TABLES [FROM | IN db_name]
[LIKE \'pattern\' | WHERE expr]
显示数据库mysql_test中所有的表名
mysql> use mysql_test;
Database changed
mysql> show tables;
+----------------------+
| Tables_in_mysql_test |
+----------------------+
| customers |
+----------------------+
1 row in set (0.00 sec)
mysql>
(2)显示表的结构
第一种
SHOW [FULL] COLUMNS FROM | IN tbl_name [FROM | IN db_name]
[LIKE \'pattern\' | WHERE expr]
第二种
DESCRIBE | DESC tbl_name [col_name | wild]
说明:MySQL支持用DESCRIBE 作为 SHOW COLUMNS FROM 的一种快捷方式
显示数据库mysql_test中表customers的结构
mysql> desc mysql_test.customers;
+--------------+----------+------+-----+---------+----------------+
| Field | Type | Null | Key | Default | Extra |
+--------------+----------+------+-----+---------+----------------+
| cust_name | char(20) | YES | | NULL | |
| cust_id | int | NO | PRI | NULL | auto_increment |
| sex | char(1) | YES | | M | |
| cust_city | char(10) | NO | | Beijing | |
| cust_address | char(50) | YES | | NULL | |
+--------------+----------+------+-----+---------+----------------+
5 rows in set (0.00 sec)
mysql>
三、索引定义
索引是DBMS根据表中的一列或若干列按照一定顺序建立的列值与记录行之间的对应关系表,因而索引实质上是一张描述索引列的列值与原表中记录行之间一一对应关系的有序表。
索引是提高数据文件访问效率的有效方法。
1)索引是以文件的形式存储的,DBMS会将一个表的所有索引保存在同一个索引文件中,索引文件需要占用磁盘空间。
2)索引在提供查询速度的同时,却会降低更新表的速度。表中的索引越多,则更新表的时间就会越长。
- 普通索引(INDEX):INDEX KEY
- 唯一性索引(UNIQUE):索引列中的所有值都只能出现一次,必须是唯一的,UNIQUE。
- 主键(PRIMARY KEY):
- 主键是一种唯一性索引
- 创建主键时,必须指定关键字 PRIMARY KEY,且不能有空值
- 主键一般是在创建表的时候指定,也可以通过修改表的方式添加主键,并且每个表只能有一个主键
单列索引:一个索引只包含原表中的一个列
组合索引:也称复合索引或多列索引,就是原表中多个列共同组成一个索引
一个表可以有多个单列索引,但这些索引不是组合索引
按照最左前缀的法则,一个组合索引实质上为表的查询提供了多个索引,以此来加快查询速度
1 索引的创建
- 使用CREATE INDEX 语句创建索引
- 可以在一个已有的表上创建索引,但不能创建主键
CREATE [UNIQUE] INDEX index_name
ON tbl_name (index_col_name, ...)
其中,index_col_name的格式为:
col_name [(length)][ASC | DESC]
可选项”UNIQUE“关键字用于指定创建唯一性索引
”Index_name“用于指定索引名,一个表可以创建多个索引,但每个索引在该表中的名称必须是唯一的
”tbl_name"用于指定要建立索引的表名
“Index_col_name”是关于索引列的描述
例子:在数据库mysql_test 的表customers 上,根据客户姓名列的前三个字符创建一个升序索引 index_customers
mysql> create index index_customers
-> on mysql_test.customers(cust_name(3) asc);
Query OK, 0 rows affected (0.01 sec)
Records: 0 Duplicates: 0 Warnings: 0
mysql> show index from mysql_test.customers;
+-----------+------------+-----------------+--------------+-------------+-----------+-------------+----------+--------+------+------------+---------+---------------+---------+------------+
| Table | Non_unique | Key_name | Seq_in_index | Column_name | Collation | Cardinality | Sub_part | Packed | Null | Index_type | Comment | Index_comment | Visible | Expression |
+-----------+------------+-----------------+--------------+-------------+-----------+-------------+----------+--------+------+------------+---------+---------------+---------+------------+
| customers | 0 | PRIMARY | 1 | cust_id | A | 0 | NULL | NULL | | BTREE | | | YES | NULL |
| customers | 1 | index_customers | 1 | cust_name | A | 0 | 3 | NULL | YES | BTREE | | | YES | NULL |
+-----------+------------+-----------------+--------------+-------------+-----------+-------------+----------+--------+------+------------+---------+---------------+---------+------------+
2 rows in set (0.00 sec)
mysql>
例子:在数据库mysql_test的表customers 上,根据客户姓名列和客户id号创建一个组合索引 index_cust
mysql> create index index_cust
-> on mysql_test.customers(cust_name, cust_id);
Query OK, 0 rows affected (0.02 sec)
Records: 0 Duplicates: 0 Warnings: 0
mysql> show index from mysql_test.customers;
+-----------+------------+-----------------+--------------+-------------+-----------+-------------+----------+--------+------+------------+---------+---------------+---------+------------+
| Table | Non_unique | Key_name | Seq_in_index | Column_name | Collation | Cardinality | Sub_part | Packed | Null | Index_type | Comment | Index_comment | Visible | Expression |
+-----------+------------+-----------------+--------------+-------------+-----------+-------------+----------+--------+------+------------+---------+---------------+---------+------------+
| customers | 0 | PRIMARY | 1 | cust_id | A | 0 | NULL | NULL | | BTREE | | | YES | NULL |
| customers | 1 | index_customers | 1 | cust_name | A | 0 | 3 | NULL | YES | BTREE | | | YES | NULL |
| customers | 1 | index_cust | 1 | cust_name | A | 0 | NULL | NULL | YES | BTREE | | | YES | NULL |
| customers | 1 | index_cust | 2 | cust_id | A | 0 | NULL | NULL | | BTREE | | | YES | NULL |
+-----------+------------+-----------------+--------------+-------------+-----------+-------------+----------+--------+------+------------+---------+---------------+---------+------------+
4 rows in set (0.01 sec)
mysql>
- 使用 CREATE TABLE 语句创建索引
例子:在已有数据库mysql_test上新建一个包含产品卖家id号、姓名、地址、联系方式、售卖产品类型、当月销量等内容的产品卖家信息表 seller,要求在创建表的同时,为该表添加由卖家id号和售卖产品类型组成的联合主键,并在当月销量上创建索引
mysql> use mysql_test;
Database changed
mysql> create table seller
-> (
-> seller_id int not null auto_increment,
-> seller_name char(50) not null,
-> seller_address char(50) null,
-> seller_contact char(50) null,
-> product_type int(5) null,
-> sales int null,
-> primary key (seller_id, product_type),
-> index index_seller(sales)
-> );
ERROR 1171 (42000): All parts of a PRIMARY KEY must be NOT NULL; if you need NULL in a key, use UNIQUE instead
mysql>
这个例子说明:MySQL可以在一个表上同时创建多个索引,并且使用 PRIMARY KEY 的列必须是一个具有 NOT NULL属性的列
mysql> create table seller
-> (
-> seller_id int not null auto_increment,
-> seller_name char(50) not null,
-> seller_address char(50) null,
-> seller_contact char(50) null,
-> product_type int(5) not null,
-> sales int null,
-> primary key (seller_id, product_type),
-> index index_seller(sales)
-> );
Query OK, 0 rows affected, 1 warning (0.02 sec)
mysql> desc mysql_test.seller;
+----------------+----------+------+-----+---------+----------------+
| Field | Type | Null | Key | Default | Extra |
+----------------+----------+------+-----+---------+----------------+
| seller_id | int | NO | PRI | NULL | auto_increment |
| seller_name | char(50) | NO | | NULL | |
| seller_address | char(50) | YES | | NULL | |
| seller_contact | char(50) | YES | | NULL | |
| product_type | int | NO | PRI | NULL | |
| sales | int | YES | MUL | NULL | |
+----------------+----------+------+-----+---------+----------------+
6 rows in set (0.01 sec)
mysql>
- 使用 ALTER TABLE 语句创建索引
例子:使用 ALTER TABLE 语句在数据库 mysql_test 中表 seller的姓名列上添加一个非唯一的索引,取名为 index_seller_name
mysql> alter table mysql_test.seller
-> add index index_seller_name(seller_name);
Query OK, 0 rows affected (0.01 sec)
Records: 0 Duplicates: 0 Warnings: 0
mysql> show index from mysql_test.seller;
+--------+------------+-------------------+--------------+--------------+-----------+-------------+----------+--------+------+------------+---------+---------------+---------+------------+
| Table | Non_unique | Key_name | Seq_in_index | Column_name | Collation | Cardinality | Sub_part | Packed | Null | Index_type | Comment | Index_comment | Visible | Expression |
+--------+------------+-------------------+--------------+--------------+-----------+-------------+----------+--------+------+------------+---------+---------------+---------+------------+
| seller | 0 | PRIMARY | 1 | seller_id | A | 0 | NULL | NULL | | BTREE | | | YES | NULL |
| seller | 0 | PRIMARY | 2 | product_type | A | 0 | NULL | NULL | | BTREE | | | YES | NULL |
| seller | 1 | index_seller | 1 | sales | A | 0 | NULL | NULL | YES | BTREE | | | YES | NULL |
| seller | 1 | index_seller_name | 1 | seller_name | A | 0 | NULL | NULL | | BTREE | | | YES | NULL |
+--------+------------+-------------------+--------------+--------------+-----------+-------------+----------+--------+------+------------+---------+---------------+---------+------------+
4 rows in set (0.00 sec)
mysql>
2 索引的查看
SHOW INDEX | INDEXES | KEYS
FROM | IN tbl_name
[FROM | IN db_name]
[WHERE expr]
3 索引的删除
- 使用 DROP INDEX 语句删除索引
DROP INDEX index_name ON tbl_name
例子:删除 customers表创建的索引 index_cust
mysql> show index from mysql_test.customers;
+-----------+------------+-----------------+--------------+-------------+-----------+-------------+----------+--------+------+------------+---------+---------------+---------+------------+
| Table | Non_unique | Key_name | Seq_in_index | Column_name | Collation | Cardinality | Sub_part | Packed | Null | Index_type | Comment | Index_comment | Visible | Expression |
+-----------+------------+-----------------+--------------+-------------+-----------+-------------+----------+--------+------+------------+---------+---------------+---------+------------+
| customers | 0 | PRIMARY | 1 | cust_id | A | 0 | NULL | NULL | | BTREE | | | YES | NULL |
| customers | 1 | index_customers | 1 | cust_name | A | 0 | 3 | NULL | YES | BTREE | | | YES | NULL |
| customers | 1 | index_cust | 1 | cust_name | A | 0 | NULL | NULL | YES | BTREE | | | YES | NULL |
| customers | 1 | index_cust | 2 | cust_id | A | 0 | NULL | NULL | | BTREE | | | YES | NULL |
+-----------+------------+-----------------+--------------+-------------+-----------+-------------+----------+--------+------+------------+---------+---------------+---------+------------+
4 rows in set (0.00 sec)
mysql> drop index index_cust on mysql_test.customers;
Query OK, 0 rows affected (0.01 sec)
Records: 0 Duplicates: 0 Warnings: 0
mysql> show index from mysql_test.customers;
+-----------+------------+-----------------+--------------+-------------+-----------+-------------+----------+--------+------+------------+---------+---------------+---------+------------+
| Table | Non_unique | Key_name | Seq_in_index | Column_name | Collation | Cardinality | Sub_part | Packed | Null | Index_type | Comment | Index_comment | Visible | Expression |
+-----------+------------+-----------------+--------------+-------------+-----------+-------------+----------+--------+------+------------+---------+---------------+---------+------------+
| customers | 0 | PRIMARY | 1 | cust_id | A | 0 | NULL | NULL | | BTREE | | | YES | NULL |
| customers | 1 | index_customers | 1 | cust_name | A | 0 | 3 | NULL | YES | BTREE | | | YES | NULL |
+-----------+------------+-----------------+--------------+-------------+-----------+-------------+----------+--------+------+------------+---------+---------------+---------+------------+
2 rows in set (0.00 sec)
mysql>
- 使用 ALTER TABLE 语句删除索引
例子:使用 ALTER TABLE 语句删除数据库 mysql_test 中表 customers 的主键和索引 index_customers
# 删除索引报错
ERROR 1075 (42000): Incorrect table definition; there can be only one auto column and it must be defined as a key
# 解决:修改为唯一约束
mysql> alter table mysql_test.customers add unique(cust_id);
Query OK, 0 rows affected (0.01 sec)
Records: 0 Duplicates: 0 Warnings: 0
mysql> alter table mysql_test.customers drop primary key, drop index index_customers;
Query OK, 0 rows affected (0.04 sec)
Records: 0 Duplicates: 0 Warnings: 0
mysql> show index from mysql_test.customers;
+-----------+------------+----------+--------------+-------------+-----------+-------------+----------+--------+------+------------+---------+---------------+---------+------------+
| Table | Non_unique | Key_name | Seq_in_index | Column_name | Collation | Cardinality | Sub_part | Packed | Null | Index_type | Comment | Index_comment | Visible | Expression |
+-----------+------------+----------+--------------+-------------+-----------+-------------+----------+--------+------+------------+---------+---------------+---------+------------+
| customers | 0 | cust_id | 1 | cust_id | A | 0 | NULL | NULL | | BTREE | | | YES | NULL |
+-----------+------------+----------+--------------+-------------+-----------+-------------+----------+--------+------+------------+---------+---------------+---------+------------+
1 row in set (0.00 sec)
第四节 数据更新
一、插入数据
1 使用 INSERT...VALUES 语句插入单行或多行元组数据
INSERT [INTO] tbl_name [(col_name, ...)]
VALUES | VALUE (expr | DEFAULT, ...), (...), ...
例子:使用 INSERT...VALUES 语句向数据库 mysql_test 的表 customers 中插入这样一行完整数据:(901,张三,F,北京市,朝阳区)
mysql> insert into mysql_test.customers
-> values (901,\'张三\',\'F\',\'北京市\',\'朝阳区\');
ERROR 1366 (HY000): Incorrect integer value: \'张三\' for column \'cust_id\' at row 1
mysql> desc customers;
+--------------+----------+------+-----+---------+----------------+
| Field | Type | Null | Key | Default | Extra |
+--------------+----------+------+-----+---------+----------------+
| cust_name | char(20) | YES | | NULL | |
| cust_id | int | NO | PRI | NULL | auto_increment |
| sex | char(1) | YES | | M | |
| cust_city | char(10) | NO | | Beijing | |
| cust_address | char(50) | YES | | NULL | |
+--------------+----------+------+-----+---------+----------------+
5 rows in set (0.00 sec)
mysql> alter table mysql_test.customers modify cust_name char(20) after cust_id;
Query OK, 0 rows affected (0.04 sec)
Records: 0 Duplicates: 0 Warnings: 0
mysql> desc customers;
+--------------+----------+------+-----+---------+----------------+
| Field | Type | Null | Key | Default | Extra |
+--------------+----------+------+-----+---------+----------------+
| cust_id | int | NO | PRI | NULL | auto_increment |
| cust_name | char(20) | YES | | NULL | |
| sex | char(1) | YES | | M | |
| cust_city | char(10) | NO | | Beijing | |
| cust_address | char(50) | YES | | NULL | |
+--------------+----------+------+-----+---------+----------------+
5 rows in set (0.01 sec)
mysql> insert into mysql_test.customers values (901,\'张三\',\'F\',\'北京市\',\'朝阳区\');
Query OK, 1 row affected (0.00 sec)
mysql> select * from customers;
+---------+-----------+------+-----------+--------------+
| cust_id | cust_name | sex | cust_city | cust_address |
+---------+-----------+------+-----------+--------------+
| 901 | 张三 | F | 北京市 | 朝阳区 |
+---------+-----------+------+-----------+--------------+
1 row in set (0.00 sec)
mysql>
例子:使用 INSERT...VALUES 语句向数据库 mysql_test 的表 customers 中插入一行数据,要求该数据目前只用明确给出cust_name列和cust_address列的信息,即分别为“李四”和“武汉市”,而cust_id列的值由系统自动生成,cust_sex列选用表中默认值,另外cust_contact列的值暂不确定,可不用指定。
mysql> insert into mysql_test.customers values(0,\'李四\',default,\'武汉市\',null);
Query OK, 1 row affected (0.01 sec)
mysql> select * from customers;
+---------+-----------+------+-----------+--------------+
| cust_id | cust_name | sex | cust_city | cust_address |
+---------+-----------+------+-----------+--------------+
| 901 | 张三 | F | 北京市 | 朝阳区 |
| 902 | 李四 | M | 武汉市 | NULL |
+---------+-----------+------+-----------+--------------+
2 rows in set (0.00 sec)
为了编写更为安全的 INSERT...VALUES 语句,通常需要在表名后的括号中明确地给出列名清单
mysql> insert into mysql_test.customers(cust_id,cust_name,cust_sex,cust_address,cust_contact)
-> values(0,\'李四\',default,\'武汉市\',null);
ERROR 1054 (42S22): Unknown column \'cust_sex\' in \'field list\'
mysql> desc customers;
+--------------+----------+------+-----+---------+----------------+
| Field | Type | Null | Key | Default | Extra |
+--------------+----------+------+-----+---------+----------------+
| cust_id | int | NO | PRI | NULL | auto_increment |
| cust_name | char(20) | YES | | NULL | |
| sex | char(1) | YES | | M | |
| cust_city | char(10) | NO | | Beijing | |
| cust_address | char(50) | YES | | NULL | |
+--------------+----------+------+-----+---------+----------------+
5 rows in set (0.01 sec)
mysql> alter table mysql_test.customers change column sex cust_sex char(1) null default \'M\';
Query OK, 0 rows affected (0.01 sec)
Records: 0 Duplicates: 0 Warnings: 0
mysql> insert into mysql_test.customers(cust_id,cust_name,cust_sex,cust_address,cust_contact) values(0,\'李四\',default,\'武汉市\',null);
ERROR 1054 (42S22): Unknown column \'cust_contact\' in \'field list\'
mysql> alter table mysql_test.customers
-> add column cust_contact char(50) null;
Query OK, 0 rows affected (0.01 sec)
Records: 0 Duplicates: 0 Warnings: 0
mysql> desc customers;
+--------------+----------+------+-----+---------+----------------+
| Field | Type | Null | Key | Default | Extra |
+--------------+----------+------+-----+---------+----------------+
| cust_id | int | NO | PRI | NULL | auto_increment |
| cust_name | char(20) | YES | | NULL | |
| cust_sex | char(1) | YES | | M | |
| cust_city | char(10) | NO | | Beijing | |
| cust_address | char(50) | YES | | NULL | |
| cust_contact | char(50) | YES | | NULL | |
+--------------+----------+------+-----+---------+----------------+
6 rows in set (0.00 sec)
mysql> insert into mysql_test.customers(cust_id,cust_name,cust_sex,cust_address,cust_contact) values(0,\'李四\',default,\'武汉市\',null);
Query OK, 1 row affected (0.00 sec)
mysql> select * from customers;
+---------+-----------+----------+-----------+--------------+--------------+
| cust_id | cust_name | cust_sex | cust_city | cust_address | cust_contact |
+---------+-----------+----------+-----------+--------------+--------------+
| 901 | 张三 | F | 北京市 | 朝阳区 | NULL |
| 902 | 李四 | M | 武汉市 | NULL | NULL |
| 903 | 李四 | M | Beijing | 武汉市 | NULL |
+---------+-----------+----------+-----------+--------------+--------------+
3 rows in set (0.00 sec)
mysql>
2 使用 INSERT...SET 语句插入部分列值数据
INSERT [INTO] tbl_name
SET col_name=expr | DEFAULT, ...
例子:
mysql> insert into mysql_test.customers
-> set cust_name=\'李四\',cust_address=\'武汉市\',cust_sex=DEFAULT;
Query OK, 1 row affected (0.00 sec)
mysql> select * from customers;
+---------+-----------+----------+-----------+--------------+--------------+
| cust_id | cust_name | cust_sex | cust_city | cust_address | cust_contact |
+---------+-----------+----------+-----------+--------------+--------------+
| 901 | 张三 | F | 北京市 | 朝阳区 | NULL |
| 902 | 李四 | M | 武汉市 | NULL | NULL |
| 903 | 李四 | M | Beijing | 武汉市 | NULL |
| 904 | 李四 | M | Beijing | 武汉市 | NULL |
+---------+-----------+----------+-----------+--------------+--------------+
4 rows in set (0.00 sec)
mysql>
3 使用 INSERT...SELECT 语句插入子查询数据
INSERT [INTO] tbl_name [(col_name, ...)]
SELECT ...
SELECT 子句返回的是一个查询到的结果集
二、 删除数据
DELETE FROM tbl_name
[WHERE wherr_condition]
[ORDER BY ...]
[LIMIT row_count]
- DELETE 语句删除的是表中的数据,而不是关于表的定义
例子:使用DELETE 语句删除数据库 mysql_test 的表 customers 中客户名为“王五”的客户信息
mysql> insert into mysql_test.customers set cust_name=\'王五\',cust_address=\'武汉市\',cust_sex=DEFAULT;
Query OK, 1 row affected (0.00 sec)
mysql> select * from customers;
+---------+-----------+----------+-----------+--------------+--------------+
| cust_id | cust_name | cust_sex | cust_city | cust_address | cust_contact |
+---------+-----------+----------+-----------+--------------+--------------+
| 901 | 张三 | F | 北京市 | 朝阳区 | NULL |
| 902 | 李四 | M | 武汉市 | NULL | NULL |
| 903 | 李四 | M | Beijing | 武汉市 | NULL |
| 904 | 李四 | M | Beijing | 武汉市 | NULL |
| 905 | 王五 | M | Beijing | 武汉市 | NULL |
+---------+-----------+----------+-----------+--------------+--------------+
5 rows in set (0.00 sec)
mysql> delete from mysql_test.customers
-> where cust_name=\'王五\';
Query OK, 1 row affected (0.00 sec)
mysql> select * from customers;
+---------+-----------+----------+-----------+--------------+--------------+
| cust_id | cust_name | cust_sex | cust_city | cust_address | cust_contact |
+---------+-----------+----------+-----------+--------------+--------------+
| 901 | 张三 | F | 北京市 | 朝阳区 | NULL |
| 902 | 李四 | M | 武汉市 | NULL | NULL |
| 903 | 李四 | M | Beijing | 武汉市 | NULL |
| 904 | 李四 | M | Beijing | 武汉市 | NULL |
+---------+-----------+----------+-----------+--------------+--------------+
4 rows in set (0.00 sec)
mysql>
三、修改数据
UPDATE tbl_name
SET col_name=expr|DEFAULT[,col_name2=expr|DEFAULT]...
[WHERE where_condition]
[ORDER BY ...]
[LIMIT row_count]
例子:使用 UPDATE 语句将数据库 mysql_test 的表 customers 中姓名为“张三”的客户的地址更新为“武汉市”
mysql> update mysql_test.customers
-> set cust_address=\'武汉市\'
-> where cust_name=\'张三\';
Query OK, 1 row affected (0.01 sec)
Rows matched: 1 Changed: 1 Warnings: 0
mysql> select * from customers;
+---------+-----------+----------+-----------+--------------+--------------+
| cust_id | cust_name | cust_sex | cust_city | cust_address | cust_contact |
+---------+-----------+----------+-----------+--------------+--------------+
| 901 | 张三 | F | 北京市 | 武汉市 | NULL |
| 902 | 李四 | M | 武汉市 | NULL | NULL |
| 903 | 李四 | M | Beijing | 武汉市 | NULL |
| 904 | 李四 | M | Beijing | 武汉市 | NULL |
+---------+-----------+----------+-----------+--------------+--------------+
4 rows in set (0.00 sec)
mysql>
- 使用UPDATE语句可同时修改数据行的多个列值,只需其彼此间用逗号分隔即可
- 如若删除表中某个列的值,在允许该列可为空值的前提下,可将它设置为NULL来实现
第五节 数据查询
一、SELECT 语句
SELECT
[ALL | DISTINCT | DISTINCTROW]
select_expr [,select_expr ...]
FROM table_references
[WHERE where_condition]
[GROUP BY col_name|expr|position
[ASC|DESC],...[WITH ROLLUP]]
[HAVING where_condition]
[ORDER BY col_name|expr|position
[ASC|DESC],...]
[LIMIT [offset,]row_count|row_count OFFSET offset]
SELECT 语句中各子句的使用次序及说明
子句 | 说明 | 是否必须使用 |
---|---|---|
SELECT | 要返回的列或表达式 | 是 |
FROM | 从中检索数据的表 | 仅在从表选择数据时使用 |
WHERE | 行级过滤 | 否 |
GROUP BY | 分组说明 | 仅在按组计算聚合时使用 |
HAVING | 组级过滤 | 否 |
ORDER BY | 输出排序顺序 | 否 |
LIMIT | 要检索的行数 | 否 |
二、列的选择与指定
在 SELECT 语句中,语法项”select_expr“主要用于指定需要查询的内容,其指定方法通常有以下几种:
1 选择指定的列
例子:查询数据库mysql_test的表customers中各个客户的姓名、性别和地址信息
mysql> select cust_name,cust_sex,cust_address
-> from mysql_test.customers;
+-----------+----------+--------------+
| cust_name | cust_sex | cust_address |
+-----------+----------+--------------+
| 张三 | F | 武汉市 |
| 李四 | M | NULL |
| 李四 | M | 武汉市 |
| 李四 | M | 武汉市 |
+-----------+----------+--------------+
4 rows in set (0.00 sec)
mysql>
例子:查询数据库mysql_test的表customers中各个客户的所有信息
mysql> select * from mysql_test.customers;
+---------+-----------+----------+-----------+--------------+--------------+
| cust_id | cust_name | cust_sex | cust_city | cust_address | cust_contact |
+---------+-----------+----------+-----------+--------------+--------------+
| 901 | 张三 | F | 北京市 | 武汉市 | NULL |
| 902 | 李四 | M | 武汉市 | NULL | NULL |
| 903 | 李四 | M | Beijing | 武汉市 | NULL |
| 904 | 李四 | M | Beijing | 武汉市 | NULL |
+---------+-----------+----------+-----------+--------------+--------------+
4 rows in set (0.00 sec)
mysql>
2 定义并使用列的别名
将SELECT语句的语法项”select_expr“指定为如下语法格式:
column_name [AS] column_alias
例子:查询数据库mysql_test的表customers中客户的cust_name列、cust_address列和cust_contact,要求将结果集中cust_address列的名称使用别名”地址“替代
mysql> select cust_name,cust_address as 地址,cust_contact
-> from mysql_test.customers;
+-----------+-----------+--------------+
| cust_name | 地址 | cust_contact |
+-----------+-----------+--------------+
| 张三 | 武汉市 | NULL |
| 李四 | NULL | NULL |
| 李四 | 武汉市 | NULL |
| 李四 | 武汉市 | NULL |
+-----------+-----------+--------------+
4 rows in set (0.00 sec)
mysql>
3 替换查询结果集中的数据
将SELECT语句的语法项”select_expr“指定为如下语法格式:
CASE
WHERE 条件1 THEN 表达式1
WHERE 条件2 THEN 表达式2
...
ELSE 表达式
END[AS] colunm_alias
例子:查询数据库mysql_test的表customers中客户的cust_name列和cust_sex列,要求判断结果集中cust_sex列的值,如果该列的值为M,则显示输出”男“,否则为”女“,同时在结果集的显示中奖cust_sex列用别名”性别“标注
mysql> select cust_name,
-> case
-> when cust_sex=\'M\' then \'男\'
-> else \'女\'
-> end as 性别
-> from mysql_test.customers;
+-----------+--------+
| cust_name | 性别 |
+-----------+--------+
| 张三 | 女 |
| 李四 | 男 |
| 李四 | 男 |
| 李四 | 男 |
+-----------+--------+
4 rows in set (0.00 sec)
mysql>
4 计算列值
例子:查询数据库 mysql_test 的表 customers 中每个客户的 cust_name 列、cust_sex列,以及对cust_id列加上数字100后的值
mysql> select cust_name,cust_sex,cust_id+100
-> from mysql_test.customers;
+-----------+----------+-------------+
| cust_name | cust_sex | cust_id+100 |
+-----------+----------+-------------+
| 张三 | F | 1001 |
| 李四 | M | 1002 |
| 李四 | M | 1003 |
| 李四 | M | 1004 |
+-----------+----------+-------------+
4 rows in set (0.00 sec)
5 聚合函数
聚合函数通常是数据库系统中一类系统内置函数,常用于对一组值进行计算,然后返回单个值。
除COUNT函数外,聚合函数都会忽略空值。
MySQL中常用聚合函数表
函数名 | 说明 |
---|---|
COUNT | 求组中项数,返回INT类型整数 |
MAX | 求最大值 |
MIN | 求最小值 |
SUM | 返回表达式中所有值的和 |
AVG | 求组中值的平均値 |
STD 或 STDDEV | 返回给定表达式中所有值的标准值 |
VARIANCE | 返回给定表达式中所有值的方差 |
GROUP_CONCAT | 返回由属于一组的列值连接组合而成的结果 |
BIT_AND | 逻辑或 |
BIR_OR | 逻辑与 |
BIT_XOR | 逻辑异或 |
三、FROM 子句与多表连接查询
若一个查询同时涉及两个或两个以上的表,则称之为多表连接查询,也称多表查询或连接查询。
多表连接查询是关系数据库中最主要的查询。
通过在FROM子句中指定多个表时,SELECT操作会使用“连接”运算将不同表中需要查询的数据组合到一个结果集中,并同样以一个临时表的形式返回,其连接方式主要包括交叉连接、内连接和外连接。
1 交叉连接
- 交叉连接,又称笛卡尔积
- 在MySQL中,它是通过在FROM子句中使用关键字“CROSS JOIN” 来连接两张表,从而实现一张表的每一行与另一张表的每一行的笛卡尔乘积,并返回两张表的每一行相乘的所有可能的搭配结果,供SELECT 语句中其他语法元素(如 WHERE 子句、GROUP BY 子句等)进行过滤和筛选操作
例子:假设数据库中有两张表,分别是 tbl1 和tbl2,现要求输出这两张表执行交叉联接后的所有数据集
mysql> SELECT * FROM tbl1 CROSS JOIN tbl2;
mysql> SELECT * FROM tbl1,tbl2;
- 交叉连接返回的查询结果集的记录行数,等于其所连接的两张表记录行数的乘积
- 对于存在大量数据的表,应该避免使用交叉连接
2 内连接
SELECT some_columns
FROM table1
INNER JOIN
table2
ON some_conditions;
连接条件“some_sonditions”一般使用的语法格式是:
[<table1>.]<列名或列别名><比较运算符>[<table2>.]<列名或列别名>
示例:
mysql> SELECT * FROM tb_student INNER JOIN tb_score ON tb_student.studentNo = tb_score.studentNo;
- 由于内连接是系统默认的表连接,因而在 FROM 子句中可以省略关键字 “INNER”,而只用关键字“JOIN”连接表
(1)等值连接
在FROM子句中使用关键字“INNER JOIN” 或 “JOIN”连接两张表时,如若在ON子句的连接条件中使用运算符 “=”(即等号),即进行相等性测试,则此连接方式称为等值连接,也称为相等连接。通常,在等值连接的条件设置中会包含一个主键和一个外键。
(2)非等值连接
在FROM子句中使用关键字“INNER JOIN” 或 “JOIN”连接两张表时,如若在ON子句的连接条件中使用除运算符 “=”之外的其他比较运算符,即进行不相等性测试,则此连接方式称为非等值连接,也称为不等连接。
(3)自连接
在FROM子句中使用关键字“INNER JOIN” 或 “JOIN”连接两张表时,可以将一个表与它自身进行连接,这种连接方式称为自连接。
自连接时一种特殊的内连接,若需要在一个表中查找具有相同列值的行,则可以考虑使用自连。
使用自连接时,需要为表指定两个不同的别名,且对所有查询列的引用均必须使用表别名限定,否则SELECT操作会失败。
3 外连接
外连接是首先将连接的两张表分为基表和参考表,然后再以基表为依据返回满足和不满足条件的记录。
外连接可以在表中没有匹配记录的情况下仍返回记录。
(1)左外连接
左外连接,也称左连接,它的使用语法格式与内连接大致相同,区别仅在于它在 FROM 子句中使用关键字 “LEFT OUTER JOIN” 或关键字 “LEFT JOIN" 来连接两张表,而不是使用关键字 ”INNER JOIN“ 或 ”JOIN“,如此可用于接收关键字 ”LEFT OUTER JOIN“ 或 ”LEFT JOIN“ 左边表(也称为基表)的所有行,并用这些行与该关键字右边表(也称为参考表)中的行进行匹配,即匹配左表中的每一行及右表中符号条件的行。
左外连接的结果集中的NULL值表示右表中没有找到与左表相符的记录
(2)右外连接
右外连接,也称右连接,它的使用语法格式与内连接大致相同,区别仅在于它在 FROM 子句中使用关键字 ”RIGHT OUTER JOIN“ 或关键字 ”RIGHT JOIN“ 来连接两张表,而不是使用关键字 ”INNER JOIN“ 或 ”JOIN“。
右外连接是以右表为基表,其连接方式与左外连接完全一样。
在右外连接的结果集中除了匹配的行之外,还包括右表中有的,但在左表中不匹配的行,对于这样的行,从左表中被选择的列的值被设置为NULL。
左外连接例子:
mysql> SELECT * FROM tb_student LEFT JOIN tb_score ON tb_student.studentNo = tb_score.studentNo;
四、WHERE 子句与条件查询
WHERE 子句中设置过滤条件的几个常用方法:
1 比较运算
比较运算符表
以上是关于关系数据库与Sql的主要内容,如果未能解决你的问题,请参考以下文章 |
---|