MySQL基础

好久没用Mysql 有些东西都忘了 在这记录一下 方便以后查看 

 

查看所有数据库

show databases

选中数据库

use [数据库名字]

删除数据库

drop [数据库名字]

创建数据库

create database [数据库名字]

 

表:

查看所有的表

show tables

查询

select * from [table] [条件]

删除表

drop table [table]

truncate table [table]

delete from [table]

区别:

drop 删除表结构和数据、被依赖的约束、索引、触发器、与该表有关的存储过程、函数等会保留 但状态会变成invalid 释放空间 彻底删除 

truncate 清空表的数据 表的结构还在 释放空间 

delete 清空表的数据 结构还在 不释放空间 

delete 是数据操作语言 dml 可以激活触发器 数据是一行一行删除 会在事务日志中记录 所以可以回滚 

truncate 是数据定义语言 ddl 不可以激活触发器 数据是以页为单位删除 事务日志只会记录页的删除 

 

创建表

create table [table] ([列名][列的数据类型][列的约束],[列名2]...) [引擎编码等 可以不写]

举例:创建一个表名叫test的表 有3个列 分别是存储整数 文本和文本 

create table test (id int,name varchar(30),message varchar(100))

 

列的数据类型 

整数型

tinyint 占用一个字节 无符号最大存储 0到255 有符号 -128到127 

small 占用两个字节 无符号 0-65535 有符号 -32768-32767

mediumint 占用3个字节 无符号0-16777215 有符号 -8388608-8388607

int 占用4个字节 无符号 0-4294967295 有符号 -2147483648-2147483647

bigint 占用8个字节 无符号 0-18446744073709551615 有符号 -9223372036854775808--9223372036854775807

 

浮点型

单精度浮点型

float 占用4字节 无符号 0,(1.175 494 351 E-38,3.402 823 466 E+38) 有符号 (-3.402 823 466 E+38,-1.175 494 351 E-38),0,(1.175 494 351 E-38,3.402 823 466 351 E+38)

双精度浮点型

double 占用8字节 无符号 0,(2.225 073 858 507 201 4 E-308,1.797 693 134 862 315 7 E+308) 有符号 (-1.797 693 134 862 315 7 E+308,-2.225 073 858 507 201 4 E-308),0,(2.225 073 858 507 201 4 E-308,1.797 693 134 862 315 7 E+308)

小数

decimal(M,D) 定点型 占用M+2个字节 M是总位数 D是小数位数 范围与双精度浮点型相同 

声明方式

float(M,D)

double(M,D)

decimal(M,D)

M是总位数 D是小数点后的位数 

举例:float(10,2) 一共10位数 小数点后两位 例如12345678.30

一般float和double不指定位数 直接写float、double 浮点型在数据库中存储是近似值 不是准确值 定点型是准确值 好像是文本格式记录的 如果要保存需要非常精确的数值比如余额、科学计算等一般使用定点型 

mysql存储浮点型会四舍五入 比如限制了存储8位数 小数部分5位数 如果存储123.00 00 09 会被存储为123. 00 00 1 

 

字符型

char(M) 定长字符串 比如 char(10) 直接申请10字符的空间 新增的数据不足10字符也占10字符的空间 

varchar(M) 变长字符串 新增的数据不足M个字符 则只占新增的字符长度 

以下每个单词首字母需要大写 

tinyblob 最大存储255字节 

blob 最大存储65535字节 

mediumblob 最大存储16M数据 

longblob 最大存储4G数据 

binary(M) 定长 M<=255 插入数据 不足255 补0

VarBinary(M) 定长 M<=65536 插入数据 不足65536 不补0 

tinytext 0-255字节 

text 0-65535字节

mediumtext 0-16772150字节

longtext 0-4294967295字节

binary和Varbinary的M指定的是字节的长度 char和varchar是字符

 

日期时间

Date 日期 格式 YYYY-MM-DD 范围 1000-01-01到9999-12-31 占3bytes

Time 时间 格式HH:MM:SS 范围 -838:59:59到838:59:59 占3bytes

DateTime 日期时间 格式 YYYY-MM-DD HH:MM:SS 范围 1000-01-01 00:00:00-9999-12-31 23:59:59 占8bytes

Year 年 格式 YYYY 范围 1000-9999 占1bytes 

TimeStamp 格式和DateTime相同 范围 1970-01-01 00:00:00 UTC-2038-01-19 03:14:07UTC 北京时间 2038-01-14 11:14:07  占4bytes

 

增删改查

insert into [表名] ([列名],[列名]...) values ([值],[值]...)

insert into test (id,name) values (1,'test1')

delete from [表名] [条件]

delete from test where id = 1

update [表名] set [列名] = [值] [条件]

update test set name = 'test2' where id = 1

select [列名] from [表名] [条件] 条件可以不写 asc/desc 升序/降序

select id from test 

 

where 查询

运算符

<>,!= 不等于

<

>

>=

<=

between ... and 区间

and

or

not

in 

 

select * from test where id = 0

select * from test where id != 0 

其他同理

select * from test where id between 10 and 1000 取出id在10和1000之间的数据 左小右大 

取出id等于1同时name等于name的数据 

select * from test where id = 1 and name = ‘name’ 

取出id等于1 和 name 等于name的数据  

select * from test where id = 1 or name = ‘name’  

取出id不等于1的数据

select * from test where id not 1 结果:取出id除了1的所有数据 

取出指定集合里的数据

select * from test where id in (0,1) 结果 取出id 0 和 1的数据 

where后加if判断

select * from test where if(id>10,id = 1,id = 7) 结果 取出id = 1的数据 

解读

把if后面看成编程里的if语句 意思是where id字段里有大于10的 那么就取出id = 1 的内容 否则取出id = 7的数据 

可以多运算符同时使用 

 

group by 

分组 根据聚合函数运算后的结果 进行分组

select * from test group by id 

根据id分组 id相同的分成同一组 只显示一行 

聚合函数

sum

avg

max

min

...

group by 一般与运算函数一起使用 

 

select count(*) from test group by id

统计根据id分组后的行数 

 

having 子句

增加having子句是因为where不能与聚合函数一起使用 having可以对where和分组后的数据再筛选一遍 

select * from test where id >10 having < 20

结果 

取出id>10小于20的集 

 

order by 

排序

把结果排序 

select * from test order by id asc 

把结果根据 id 升序排列 

desc 降序

 

limit

指定取出多少行 

select * from test limit 10

取出10行

select * from test limit 5,3

跳过5行后取出3行

 

where子查询

select [列] from [表名] where id = ([新的查询语句])

select * from test where id = (select max(id) from test)

 

from子查询

把一个查询结果当成一个表查询

select * from (select* from test) as ceshi where id = 1

查询出id = 1 的结果 

 

exists子查询 

EXISTS 运算符用于测试子查询中是否存在任何记录 如果子查询返回一条或多条记录,EXISTS 运算符返回 TRUE 返回true后执行外层查询 

select * from test where exists (select * from test id = 1)

返回了test表所有数据 

select * from test where exists (select * from where id = 0)

返回0条数据

 

union

联合查询

把两个查询语句的结果联合成一个结果 

select name from test union select name from ceshi

查询两表的列必须是相同的数据类型 名字可以不一样 

联合后的结果会把重复的数据删除 保留1个  

可以用union all 来显示所有数据 

 

连接查询

连接两个表查询 通俗来说 把两个表连接成一张大表来查询 

left join 

左连接 

结果根据左表的列为准 如果左表的数据多于第二张表 则左表的数据都显示 右表没有的用NULL填充 反之 右表多的数据不显示 

select * from test left join test7 on test.id = test7.id

 

right join 

右连接

与左连接相反 

select * from test right join test2 on test.id = test2.id 

 

inner join

内连接

留下两个表相同的数据 其他的都不显示 

select * from test inner join test2 on test.id = test2.id 

只会留下两张表共有id的数据 

连接之后还可以添加其他筛选语句 

 

列的增删改

aler table [表名] add column [列名] [数据类型] [not null ...] 在最后增加一列

alter table [表名] add column [列名] [数据类型] [not null ...] after name 在name后增加一列

aler table [表名] add column [列名] [数据类型] [not null ...] first 增加到第一列 

alter table [表名] drop column [列名]

alter table [表名] modify [列名] [数据类型] [列的约束] 修改列的类型

alter table [表名] change [列名] [新列名] [数据类型] [约束] 修改列名和数据类型 

 

视图

把查询出来的结果保存成视图 

视图是虚拟的表 

只有表结构 没有数据 数据还在原表 

要查询时数据取自原表 

原表更新 视图也会更新 

创建视图

create view [视图名字] as 查询语句

举例

create view test7 as select * from test

查询

跟表查询一致

删除

drop view [名字],[名字]..

视图不可以用 from子查询、系统变量和用户变量、预处理、触发器

 

触发器

4要点

触发地点(table)

触发操作(增删改)

触发时机(after/before)

触发事件(增删改)

 

create trigger [触发器名字]

after/before [触发时间]

insert on test [触发地点和操作]

for each row [行触发器] 固定的 颗粒度 行 影响一行触发一次 不写默认执行一个语句触发一次 mysql之前了解只有行触发器 现在不知道有没有其他级别触发器 

begin

[执行语句] 可以用if等语句

end

 

先更改语句结束符

delimiter @

 

create trigger test

after insert on test 

for each row

begin

insert into test3 (id,name) values (1,'n'); [在test表插入后执行此语句] 可以用变量等 

end @

 

之后可以改回原来的结束符

delimiter ;

5.7之前不能同时存在两种一样的触发器 

[创建等语句] after insert on test [执行一些语句]

[创建等语句] after insert on test [执行另一些语句]

不能同时监视相同操作 

想要实现可以在执行语句里加if判断来执行不同操作 

5.7之后可以 不过需要注意:

监视同一个操作 触发增加一行数据的操作 如果字段有唯一索引 或者自增 两个触发器增加相同数据会不成功类似问题 

 

查看所有触发器

show trigger

删除触发器 

drop trigger name

 

需要用到影响之前行的数据使用old.列名 

新行用new.列名 

订单表num列数值改变 就把库存表的number字段减去相对数值 

create trigger test 

after update on price 

for each row

begin

update inventory set number = old.number (修改之前的数值) + new.num (修改之后订单表num列的数值) - old.num (修改之前订单表num列的数值) where id = old.number ;

end@

 

before 在操作前触发 可以用来检测违法操作等 

 

存储引擎

 

特点/引擎 MyISAM InnoDB memory merge 点击查看简介 NDB
存储限制 256TB(好像可以修改) 64TB

RAM大小

max_heap_table_size = xxM

tmp_table_size = xxM

可以设置 最大存储大小

默认16M

没有
事务   支持     仅支持READ COMMMITTED (不支持MVCC)
表锁 行锁 表锁  表锁 行锁
B树索引 支持 支持 支持  支持 支持 
哈希索引     支持   支持
全文索引 支持

支持

(innoDB 1.2.x开始支持 很多文章还没更新)

     
集群索引   支持      
数据缓存   支持 支持   支持
索引缓存 支持 支持 支持  支持 支持
数据压缩 支持

支持

(1.2.x后支持MyISAM全部特性)

支持    
空间使用 N/A   低
内存使用 中等 低   高
批量插入速度 高   高
外键   支持     支持

 

存储引擎 使用场景
MyISAM 默认MySQL插件式存储引擎(5.5版本之前),如果应用是以读操作和插入操作为主,只有很少的更新和删除操作,并且对事务的完整性,并发性要求不是很高,那么选择这个存储引擎是非常适合的,MyISAM是在Web,数据仓库和其他应用环填下最常使用的存储引擎之一
innoDB 5.5版本之后的默认存储引擎 用于事务处理应用程序,支持外键,如果应用对事务的完整性有比较高的要求,在并发条件下要求数据的一致性,数据库操作除了插入和查询之外,还包括很多的更新、删除操作,那么InnoDB存储引擎应该是比较合适的选择,InnoDB存储引擎除了有效的降低由于删除和更新导致的锁定,还可以确保事务的完整提交(commit)和回滚(Rollback),对于类似计费系统或者财务系统等对数据准确性要求比较高的系统,InnoDB都是合适的选择
memory 将所有数据保存在RAM中,在需要快速定位记录和其他类似数据的环境下,可以提供极快的访问,MEMORY的缺陷是对表的大小有限制,太大的表无法承存在内存中,其次是要确保表的数据可以恢复,数据库异常终止之后表中的数据是可以恢复的,MEMORY表通常用于更新不太频繁的小表,用以快速得到访问结果
merge Merge存储引擎允许将一组使用MyISAM存储引擎的并且表结构相同(即每张表的字段顺序、字段名称、字段类型、索引定义的顺序及其定义的方式必须相同)的数据表合并为一个表,方便了数据的查询

 

附录:MySQL引擎之间的区别和优劣之分 - 知乎 (zhihu.com) 简单介绍了十种存储引擎 

面试 | MySQL 中 InnoDB 和 MyISAM 的联系与区别 - 知乎 (zhihu.com) 不是很全 不过也有比较详细的地方 

 

深入理解InnoDB -- 架构篇 - 知乎 (zhihu.com) 详细 

 

 

事务

事务4特点

原子性 事务中的语句要么都成功执行 要么都失败 体现在最终是commit还是rollback 不可再分割 有的成功有的失败是不可能的 commit就都执行 rollback就都失败 

一致性 事务使库从一个状态转到另一个状态 比如A给B转账 A减少 B增加 从最终结果看是 合理的 一致的 如果A减少 B没增加 就不一致了 

隔离性 事务允许并发 多个事务有相应的隔离机制

持久性 事务结束后 结果持久化在硬盘上 

 

MySQL事务【详解-最新的总结】_mysql的事务_悬浮海的博客-CSDN博客 

 

隐式事务

即insert delete update 等都会在语句结束自动提交 

查看是否开启自动提交 

show variables like ‘autocommit’

设置是否自动提交

set autocommit = 0/1

 

显式事务

即手动控制commit rollback

开启事务

start transaction 或 begin

start transaction read only开启只读事务 事务中只能执行读操作 可能对读有优化 执行其他操作会报错 

commit 提交事务 

rollback 回滚事务 

savepoint save1 加保存点 跟游戏里保存点一样 不想回滚所有操作 可以提前加保存点 再执行 rollback to save1 可以回滚保存点到现在的操作 

 

事务的隔离级别 

在并发的时候需要

想象一个场景 两个客户端同时连接MySQL 同时开启了事务 事务A修改了一个数据 事务B读取了该数据 事务A因为一些原因回滚了 事务B读的数据跟数据库的实际数据不一样 

上面这个问题的名字就叫 脏读 

还有一些因为隔离级别不同差不多的情况 

读已提交 顾名思义 可以读到其他事务提交了的数据  

不可重复读 还是脏读的场景 有两个事务同时操作数据库 不同的是 这次事务B先提交了 把数据库的数据更新完了 A去读取数值就是新数值 跟之前查询的不一样了 就是不可重复读 

可重复读 顾名思义 可以重复读 每次结果都一样 

幻读 事务A按照条件进行查询 事务B插入了符合事务A条件的数据 事务A之前查询没有这条数据 插入了一条数据  此列有唯一索引 发现插入失败 再查一次 又发现有这条数据 就像幻觉一样 

根据需求不同 有不同的隔离级别解决相应的问题 一共有4个级别 

 

读未提交 read uncommitted 最宽松的 顾名思义 可以读没有提交的数据 会出现脏读的现象 

读已提交 READ COMMITTED 顾名思义 这个级别可以读已提交的数据 读不到没提交的数据 

可重复读 repeatable read 顾名思义 可以重复读 解决了不可重复读问题 但会出现幻读 

串行化 SERIALIZABLE 事务都是互斥的 事务不能同时 读写 写读 写写 可以同时读 

隔离级别越高 并发的能力就越低 

 

查看

查看隔离级别

show variables like 'transaction_isolation'

设置隔离级别

全局

set global transaction isolation level = REPEATABLE READ

会话

set session transaction isolation level = [...]

 

数据库备份 

mysqldump 

原理:连接到数据库 把需要备份的数据查询出来 转成insert 语句 恢复直接执行insert 语句

常用参数

--host /-h         主机地址

--port / -p         端口

--user / -u        用户名

--password / -p         密码

--databases         要备份的数据库

--all-databases       备份所有数据库

--compack         压缩

--lock-tables         备份前锁定所有表

--no-create-db         禁止创建 创建数据库语句

--force         当备份出错时继续 

--default-character-set        指定字符集

--add-locks       备份前锁定要备份的表 

 

例子

mysqldump -u root -p [] --all-databases > ../xxx/all.db

 

还原

use [数据库]

source 路径  

 

未完

posted @ 2023-07-13 20:31  mlto  阅读(18)  评论(0编辑  收藏  举报