阿里云-云小站(无限量代金券发放中)
【腾讯云】云服务器、云数据库、COS、CDN、短信等热卖云产品特惠抢购

MySQL官方文档阅读与记录

188次阅读
没有评论

共计 6532 个字符,预计需要花费 17 分钟才能阅读完成。

接触变成时间不久,之前对于 MySQL 的了解局限于简单的 CURD,没有系统和深入的学习过,最近想要更深入的学习和了解一下 MySQL,打算先从官方文档入手。

最新官方文档:https://dev.mysql.com/doc/refman/8.0/en

1. MySQL 对标准 SQL 的扩展

(1) MySQL 对标准的 SQL 进行了扩展,当进行迁移到其他数据库时,SQL 语句的语法可能存在不支持的情况,可以将 MySQL 的特定语法用如下的方式进行编写,MySQL 数据库将会对【/*!语句 */】内的语句进行解析并执行,其他数据库将会忽略该语句,因此可以实现迁移。

/*! MySQL-specific code */

示例:

SELECT /*! STRAIGHT_JOIN */ col1 FROM table1,table2 WHERE ...

(2) 不同的 MySQL 版本之间存在着语法的差异,因此,在特定场景下,可以指明该语句支持的 MySQL 语法支持的版本号

CREATE TABLE t1(a INT, KEY (a)) /*!50110 KEY_BLOCK_SIZE=1024 */;
-- 上述语句表明,当且仅当 MySQL 的版本大于等于指定版本 50110 的时候才会执行 KEY_BLOCK_SIZE=1024 语句

(3) 按类型列举 MySQL 对标准 SQL 的扩展

  • 数据的组织方式
    • 每个数据库对应数据目录下的一个目录
    • 每个表对应一个文件
    • 数据库表名是否大小写敏感与操作系统对文件名大小写是否敏感有关
  • 一般语法 SQL 语法
    • 字符串可以用 ” 或者 ’ 括起来,当ANSI_QUOTES 启用的时候则只能使用单引号
    • \ 表示转义字符
    • 可以用db_name.tbl_name 来访问不同数据库下的表,MySQL 不支持表空间
  • 数据类型
    • MEDIUMINTSET, ENUMBLOBTEXT
    • AUTO_INCREMENTBINARYNULLUNSIGNED, ZEROFILL
  • 函数与操作符
    • 函数支持别名
    • MySQL 认为 || 和 && 是逻辑 OR 和逻辑 AND,因此不支持标准 SQL 的 || 字符串连接,而是使用 concat 函数
    • COUNT(DISTINCT value_list)
    • 默认的字符串比较是不区分大小写的,如果需要区分大小写,则在声明某一列的时候需要加上BINARY 属性
    • % 取余,N%M 等于 MOD(N,M)
    • LAST_INSERT_ID()返回最新的 auto_increment 的数据
    • LIKE 可以用于数值型的数据
    • REGEXP NOT REGEXP
    • CONCAT()   CHAR() 
    • BIT_COUNT()CASEELT()FROM_DAYS()FORMAT()IF()PASSWORD()MD5()PERIOD_ADD()PERIOD_DIFF()TO_DAYS()WEEKDAY() 
    • TRIM()
    • STD()BIT_OR()BIT_AND()BIT_XOR(),GROUP_CONCAT()

2. MySQL 与标准 SQL 的语法差异

(1) SELECT … INTO TABLE

-- 不支持 SELECT ... INTO TABLE,可以用 INSERT INTO ... SELECT 代替(待验证),支持 SELECT ... INTO OUTFILE 和 CREATE TABLE ... SELECT.
INSERT
INTO tbl_temp2 (fld_id) SELECT tbl_temp1.fld_order_id FROM tbl_temp1 WHERE tbl_temp1.fld_order_id > 100;

(2) UPDATE


UPDATE
t1 SET col1 = col1 + 1, col2 = col1;

(3) FOREIGN KEY

(4) — 注释

/* comment */ 是标准 SQL 的注释,MySQL 支持该语法,用于实现 MySQL 的特殊语法

-- 注释的时候 -- 和后面的注释内容之间必须有空格
UPDATE account SET credit=credit-payment

-- 当 payment 等于 - 1 时:
UPDATE account SET credit=credit--1

-- 如果将 -- 后面的内容解析为注释,则上面的语句相当于:
UPDATE account SET credit=credit

3. MySQL 对于约束的处理

(1) 主键和唯一索引

当执行 INSERT 和 UPDATE 的时候将出现违反主键约束和唯一性约束的情况;

当表支持事务的时候,遇到违反约束的语句时将会停止执行并自动回滚;

当表不支持事务的时候,遇到违反约束的语句时将会停止执行该语句及之后的语句,已经执行的语句无法回滚

MySQL 支持 IGNORE 关键字来忽略违反约束的语句并继续执行,可以使用mysql_info() 的 API 或者 SHOW WARNING 来查看实际插入或者更新的语句数量

(2) 外键

Mysql 支持在 CREATE TABLE 和ALTER TABLE 的时候对外键进行 UPADTE 和 DELETE,可选的操作包括:RESTRICT(默认), CASCADESET NULLNO ACTION

MySQL 要求外键必须有索引,如果没有索引的话将会创建自动索引

可以通过查询 INFORMATION_SCHEMA.KEY_COLUMN_USAGE 表来获取外键的信息

mysql> SELECT TABLE_SCHEMA, TABLE_NAME, COLUMN_NAME, CONSTRAINT_NAME
     > FROM INFORMATION_SCHEMA.KEY_COLUMN_USAGE
     > WHERE REFERENCED_TABLE_SCHEMA IS NOT NULL;
+--------------+---------------+-------------+-----------------+
| TABLE_SCHEMA | TABLE_NAME    | COLUMN_NAME | CONSTRAINT_NAME |
+--------------+---------------+-------------+-----------------+
| fk1          | myuser        | myuser_id   | f               |
| fk1          | product_order | customer_id | f2              |
| fk1          | product_order | product_id  | f1              |
+--------------+---------------+-------------+-----------------+
3 rows in set (0.01 sec) 

(3) 无效数据

默认情况下,MySQL 将会对非法的输入值进行强制处理使其符合规范,可以启用 strict SQL mode

(4) 枚举和集合

ENUM 枚举类型要求输入的数据必须是已定义的数据之一

SET 集合类型要求输入的数据必须是空字符串或者或者由已定义的元素组成

当 strict mode 启用的时候,将会拒绝并抛出错误

ENUM('a','b','c') -- 值 '''d''ax' 均为无效的错误数据
SET('a','b','c') -- 值'd''a,b,c,d' 均为无效的错误数

在 strict mode 情况下,可以用 INSERT IGNORE 或者 UPDATE IGNORE 来忽略错误,对于 ENUM 类型的非法数据将会插入非法数据 0,对于 SET 类型数据将会插入去除非法字符之后的数据

 3.Tutorial

(1) MySQL 提示符的含义

提示符 含义
mysql> 可以输入下一条执行语句
> 等待继续输入
‘> 等待以单引号开始的字符串的结尾
“> 等待以双引号开始的字符串的结尾
`> 等待以 ` 开始的字符串的结尾
/*>

等待以 /* 开始的注释的结束 */

(2) 创建并使用 database

-- 查看所有的数据库
show databases;

-- 使用某一个数据库,use 和其他的语句不同,不需要在结尾处加分号,但是必须在一行内输完,不能换行
use test

-- 创建数据库
CREATE DATABASE menagerie;

-- 登陆并连接使用数据库
mysql -h host -u user -p menagerie

-- 查看当前使用的数据库
SELECT DATABASE();

-- 查看数据库下有哪些表
SHOW TABLES;

-- 创建表
CREATE TABLE pet (name VARCHAR(20), owner VARCHAR(20), species VARCHAR(20), sex CHAR(1), birth DATE, death DATE);

-- 查看表定义
describe pet;

-- 从文件中导入数据到表中
-- pet.txt 内容为:Whistler Gwen bird \N 1997-12-09 \N
LOAD DATA LOCAL INFILE '/path/pet.txt' INTO TABLE pet;

-- 当文件中行结尾的分隔符为 \n\r 的时候需要特别的标明
LOAD DATA LOCAL INFILE '/path/pet.txt' INTO TABLE pet LINES TERMINATED BY '\r\n';
-- 当从文件中 load 数据失败时(ERROR 1148 (42000): The used command is not allowed with this MySQL version),需要查询是否启用了该功能,默认情况下是关闭的
show variables like '%LOCAL%';

-- 可以看到 local_infile 变量,默认情况下为 OFF,设置该属性为 ON,注意:使用这种方式修改后仍无法导入数据
SET global local_infile=1-- 需要在与数据库服务器建立连接的时候,设置该连接数据
mysql -h host -u user --local_infile -p menagerie

-- 插入数据
INSERT INTO pet VALUES ('Puffball','Diane','hamster','f','1999-03-30',NULL);

-- 查询数据
SELECT * FROM pet;

-- 删除所有数据
DELETE FROM pet;

-- 更新数据
UPDATE pet SET birth = '1989-08-31' WHERE name = 'Bowser';

-- 查询符合条件的数据
SELECT * FROM pet WHERE name = 'Bowser';
SELECT * FROM pet WHERE birth >= '1998-1-1';
SELECT * FROM pet WHERE species = 'dog' AND sex = 'f';
SELECT * FROM pet WHERE species = 'snake' OR species = 'bird';
SELECT * FROM pet WHERE (species = 'cat' AND sex = 'm') OR (species = 'dog' AND sex = 'f');

-- 排序
SELECT name, birth FROM pet;
SELECT owner FROM pet;
SELECT DISTINCT owner FROM pet;
SELECT name, species, birth FROM pet WHERE species = 'dog' OR species = 'cat';

MONTH(birth) = MOD(MONTH(CURDATE()), 12) + 1;MONTH(birth) = MOD(MONTH(CURDATE()), 12) + 1;
 
-- 时间日期计算

SELECT name, birth, CURDATE(), TIMESTAMPDIFF(YEAR,birth,CURDATE()) AS age FROM pet;
SELECT name, birth, CURDATE(), TIMESTAMPDIFF(YEAR,birth,CURDATE()) AS age FROM pet ORDER BY name;
SELECT name, birth, CURDATE(), TIMESTAMPDIFF(YEAR,birth,CURDATE()) AS age FROM pet ORDER BY age;
SELECT name, birth, death, TIMESTAMPDIFF(YEAR,birth,death) AS age FROM pet WHERE death IS NOT NULL ORDER BY age;

SELECT name, birth, MONTH(birth) FROM pet;
SELECT name, birth FROM pet WHERE MONTH(birth) = 5;
SELECT name, birth FROM pet WHERE MONTH(birth) = MONTH(DATE_ADD(CURDATE(),INTERVAL 1 MONTH));
SELECT name, birth FROM pet WHERE  
对 NULL 值的支持,NULL 表示值缺失

0 和 NULL 代表 false,其他均表示 true
SELECT 1 IS NULL, 1 IS NOT NULL;
任何数据和 NULL 进行 =、<>、>、< 均为 NULL
SELECT 1 = NULL, 1 <> NULL, 1 < NULL, 1 > NULL;
空字符串和 NULL 是不一致的
SELECT 0 IS NULL, 0 IS NOT NULL, IS NULL, IS NOT NULL

-- 模式匹配
SELECT * FROM pet WHERE name LIKE 'b%';
SELECT * FROM pet WHERE name LIKE '%fy';
SELECT * FROM pet WHERE name LIKE '%w%';
-- 查找名称长度为 5 个字符的数据
SELECT * FROM pet WHERE name LIKE '_____';

-- 正则表达式匹配
SELECT * FROM pet WHERE REGEXP_LIKE(name, '^b');
SELECT * FROM pet WHERE REGEXP_LIKE(name, 'fy$');
SELECT * FROM pet WHERE REGEXP_LIKE(name, 'w');
SELECT * FROM pet WHERE REGEXP_LIKE(name, '^.....$');

-- 当需要大小写敏感的时候需要指定大小写敏感的字符集
SELECT * FROM pet WHERE REGEXP_LIKE(name, '^b' COLLATE utf8mb4_0900_as_cs);
SELECT * FROM pet WHERE REGEXP_LIKE(name, BINARY '^b');
SELECT * FROM pet WHERE REGEXP_LIKE(name, '^b', 'c');
SELECT * FROM pet WHERE REGEXP_LIKE(name, '^.{5}$'); 
-- 统计记录数目
SELECT COUNT(*) FROM pet;
SELECT owner, COUNT(*) FROM pet GROUP BY owner;
SELECT species, COUNT(*) FROM pet GROUP BY species;
SELECT sex, COUNT(*) FROM pet GROUP BY sex;
SELECT species, sex, COUNT(*) FROM pet GROUP BY species, sex;
SELECT species, sex, COUNT(*) FROM pet WHERE species = 'dog' OR species = 'cat' GROUP BY species, sex;
 SELECT species, sex, COUNT(*) FROM pet WHERE sex IS NOT NULL GROUP BY species, sex; 
-- 多表操作
SELECT pet.name,TIMESTAMPDIFF(YEAR,birth,date) AS age,remark FROM pet INNER JOIN event ON pet.name = event.name WHERE event.type = 'litter';
SELECT p1.name, p1.sex, p2.name, p2.sex, p1.species FROM pet AS p1 INNER JOIN pet AS p2 ON p1.species = p2.species AND p1.sex = 'f' AND p2.sex = 'm';

本文永久更新链接地址:http://www.linuxidc.com/Linux/2018-01/150560.htm

正文完
星哥玩云-微信公众号
post-qrcode
 0
星锅
版权声明:本站原创文章,由 星锅 于2022-01-22发表,共计6532字。
转载说明:除特殊说明外本站文章皆由CC-4.0协议发布,转载请注明出处。
【腾讯云】推广者专属福利,新客户无门槛领取总价值高达2860元代金券,每种代金券限量500张,先到先得。
阿里云-最新活动爆款每日限量供应
评论(没有评论)
验证码
【腾讯云】云服务器、云数据库、COS、CDN、短信等云产品特惠热卖中