MySql基础知识

数据库是保存有组织数据的容器,DBMS是为管理数据库而设计的软件管理系统,MYSQL 等是数据库管理系统。MYSQL是一种数据库管理软件。

MYSQL命令

create database dbname -- 创建数据库 
use dbname-- 选择数据库 
drop database dbname -- 直接删除数据库 
show database -- 显示可用数据库列表 
show table -- 显示数据库中的表的列表 
show full columns from    tableName -- 与 DESCRIBE tableName 等效,显示表的字段信息 
show grants -- 显示授予用户的安全权限 
SHOW ERRORS -- 显示服务器错误
SHOW WARNINGS  -- 显示警告信息

SQL

SQL(STRUCTURED QUERY LANGUAGE) 结构化查询语言,一种专门用于与数据库通信的语言,不是DBMS专有的语言,很多DBMS都支持SQL,但是某个DBMS支持的SQL语法不能完全适用于其他DBMS。
SQL语句不区分大小写,多条语句需要分号分隔,单条语句可以不用分号。

创建表

use dbname
create
table 表名 ( id INT NOT NULL AUTO_INCREMENT, 字段2 VARCHAR(50) NOT NULL, 字段3 VARCHAR (50) NOT NULL DEFAULT 1, PRIMARY KEY(id) ) ENGINE =INNODB

检索列

select 字段1,字段2,... from 表名 where id=123 and ...

除非真的需要全部字段,否则建议不要使用用'*',这种操作是很耗性能的。

限制结果(用于分页查询)

  select 字段1,... from 表名 limit 10 检索前10行
  select 字段1,... from 表名 limit 11,20 检索11到20
  select 字段1,... from 表名 limit 11,-1 检索11到最后一行

结果排序

select 字段1,... from 表名 order by 字段 asc  -- asc 默认按照字段升序,此时可以不用orderby
select 字段1,... from 表名 order by 字段 desc  -- desc 按照字段降序 ,必须写 DESC
select 字段1,... from 表名  where id=order by 字段 desc  -- order by 放在where之后
select 字段1,... from 表名  where id=order by 字段a asc,字段b desc  -- 按照字段a正序,如果字段a相同,则按照字段b倒叙

where筛选过滤

select * from 表名 where 字段2 <> '' -- 不等于 和 != 效果相同
select * from 表名 where 数值字段 between 2 and 5 -- 检索2到5的记录,包括2和5
select * from 表名 where 字段2 is null -- 查找列无值的行,这里的无值不等于 值为0 和 空字符串
select * from 表名 where id in (1,2) -- 效果和or相似,但是据说一般or执行更快
select * from 表名 where id not in (1,2) -- 一般不建议这样使用
select * from 表名 where 字段a=10 or 字段b='lsk' and 字段c='avf123' -- 执行顺序是 字段a=10 or (字段b='lsk' and 字段c='avf123'),因为and的优先级高于or,添加括号的可增加可读性

过滤之通配符

    select * from 表名 where 字段a like 'abc%' -- 匹配abc开头的字段a,无论其后多少字符,注意:like '%'是不匹配值为NULL的行,且匹配值区分大小写
    select * from 表名 where 字段a like 'abc_', -- _仅仅匹配一个字符

除非必要,否则不建议使用 '%字符%'的形式进行匹配,非常好性能且查询效率非常慢。

过滤之正则表达式

select * from 表名 where 字段a regexp 'abc|XYZ' -- 等同于or
select * from 表名 where 字段a regexp '[abc]|XYZ' -- 匹配包含'aXYZ'或者'bXYZ'或者'cXYZ'    
select * from 表名 where 字段a regexp '.abc' -- 正则表达式. 匹配任意一个字符,等同于like的_
select * from 表名 where 字段a regexp 'abc' -- 匹配包含abc的字段a,等同于like的双侧%

[^123]表示123除外
[1-9],[A-Z],用'-'表示范围
匹配特殊字符(比如'.'和'_')可以采用'\'转义,比如匹配有'.'的可以这样表示'\.'

文本处理函数

    length(字符字段) -- 返回串长度
    lower(字符字段) -- 将串转换为小写
    upper(字符字段) -- 将串转换为大写
    ltrim(字符字段) -- 去掉串左边的空格
    rtrim(字符字段) -- 去掉串右边的空格

日期和时间处理函数

    curdate() -- 当前日期 yyyy-MM-dd
    curtime()  -- 当前时间 例10:52:10
    NOW()  -- yyyy-MM-dd hh-mm-ss
    DATE(须有值) -- 返回日期的日期部分 SELECT DATE(NOW()) yyyy-MM-dd
    DATE_FORMAT() -- 返回格式化的日期和时间串
    HOUR() -- 返回一个时间的小时部分
    MINUTE() -- 返回一个时间的分钟部分
    MONTH() -- 返回一个日期的月份部分
    SECOND() -- 返回一个时间的秒部分
    TIME() -- 返回一个日期时间的时间部分
    YEAR() -- 返回一个日期的年份部分

聚集函数

    select avg(数值字段) as avgNumber from 表名 -- 求平均值
    select count(*) from 表名 -- 记录数
    select column(name) from 表名 -- 表里列有值的记录数,值为NULL不计
    select sum(count) from 表名  -- 返回指定列的和,也可以 SUM(a*b)
    select max(score) from 表名 -- 查找最大值
    select min(score) from 表名 -- 查找最小值
    

数据分组

    select avg(score) from score group by classId  -- 检索不同课的平均分
    select avg(score) from score group by classId,depId, ... -- GROUP BY 后面可以多个字段,select的字段必须是group by 后面的字段或者是聚合函数
    select count(*) from 表名 group by id having count(*)>10 -- 分组后,对分组再进行过滤,需要使用having。

注意:    
如果分组列有NULL值的,将列值为NULL的分为一组;
GROUP BY 要在WHERE之后,ORDER BY 之前,也就是先过滤再分组再排序;

子查询

select * from 表A where id in (select a_Id from 表B where 表B字段1='')
select 字段1,(select field1 from 表2 where ...) as f1 from 表1 where 表1字段=''

建议少用子查询,耗性能

主键

唯一标识自己的一列或一组列,唯一区别表中的一行,主键不能为空,任意两行的主键值必须不同

主键不是必须的,但是建议每个表中有一个主键,这样操作管理更方便,主键值最好不更新,以免数据对应不上

一个表的主键只能有一个

外键

一个表的主键指向另一个表的外键,比如说student表的主键studentId,在score表中也有,并且是score表的外键

外键保持了数据完整性和一致。,比如你在student表里面修改了studentId后,则score的studentId也会联动更新(前提是做好了外键值关联)。并且score表中插入的studentId必须是student表里有的

联表查询

    select a.字段1,b.字段甲 from 表A as a,表B as b -- 两个表进行联结,进行的是笛卡尔积,结果的行数表A行数*表B的行数
    select a.字段1,b.字段甲 from 表A as a,表B as b where a.id=b.A_Id -- 在上面的基础上,加上WHERE进行过滤
    select a.字段1,b.字段甲 from 表A as a inner join 表B as b on a.id=b.A_Id -- 内部联结
    外部联结:left outer joinright outer joinLEFT JOIN 保证了保证左边表的所有行,右联结同理
    full join效果等同于 left outer join + right outer join ,即左表不匹配的右表不匹配的都会列出

UNION与UNION ALL

[SQL 语句 1] UNION [SQL 语句 2]
[SQL 语句 1] UNION ALL [SQL 语句 2]

UNION去重且排序
UNION ALL不去重不排序

使用时,必须确保搜出的结果集字段数量、类型、名称一致。

插入数据

    INSERT INTO 表名 VALUES ('值1','值2') -- 必须值的个数和顺序必须和字段的个数顺序一致
    INSERT INTO 表名 (字段1,字段2) VALUES ('值1','值2') -- 列和值,只要一一对应就好
    INSERT INTO 表名 VALUES ('值1','值2'),('值1','值2'),.. -- 插入多行
    INSERT INTO 表名 VALUES (SELECT 字段1 ,字段2 FROM 其他表) -- 查询出其他表的字段,列明无所谓,顺序要注意

更新数据

 update 表名 set 字段a ='值1',.. WHERE id=123 -- 需要注意的是WHERE一定不要漏,要不然会更新全表

更新多行的时候如果中途出现错误,会将更新的恢复回原来的值。

如果要做到即使中途发生错误也要继续更新可以采用 IGNORE关键字:update ignore 表名 set ....

删除数据

删除一个列的值:SET username=null,前提是该字段允许为null

删除一行:delete from 表名 where id=123

删除整个表的行 delete from 表名 ,(此时只是记录清空)。

 turncate 表名,也有记录清空的效果。不同的是turncate是先删除表,然后新建立一个同名表

表更新

ALTER TABLE 表名 ADD 字段a CHAR(20) -- 添加一列
ALTER TABLE 表名 DROP COLUMN 字段1 -- 删除一列
ALTER TABLE 表名 CONSTRAINT foregin_key FOREGIN KEY (字段a) REFERENCES 外表 (字段1) -- 定义外键
ALTER TABLE 表名 ADD PRIMARY KEY (id) -- 添加主键

视图

创建视图:CREATE VIEW 视图名称 AS SELECT 字段1,,... FROM 表名
删除视图:DROP VIEW MY_VIEW 

当我们查询后出现一个结果,包装成一个虚拟表,也就是视图,我们可以把视图当成表使用(仅查看);
视图本身不包含数据,数据是从其他表检索出来;
使用视图可以重用SQL,并且可以保护数据,可以授予用户部分数据权限而不是全部数据;
如视图中存在分组(GROUP BY)、联结、子查询、并(UNOIN)、聚合函数(SUM/COUNT等)、计算字段、DISTINCT等都不能对视图进行更新操作;

存储过程

有些时候SQL也需要有逻辑判断(If),这样就出现一种“把多条SQL语句封装在一起执行”的场景,也就是存储过程,这样不仅简单安全而且性能也会更高(减少了多次的SQL网络传输时间)

注意:存储过程并不显示结果,只是将结果返回给你指定的变量。

执行的方式:1、创建一个存储过程;2、调用存储过程,将参数传入;3、SELECT参数输出结果。

参数类型说明: in 传递给存储过程,out 从存储过程传出,inout 对存储过程传入传出。结果将返回给OUT变量

示例:

创建存储过程:

CREATE PROCEDURE pro_name (
    IN @inputParaValue INT,
    OUT @outParaVlaue INT
)
BEGIN
    SELECT
        count(id)
    FROM
        tableA
    WHERE
        age =@inputParaValue INTO @outParaVlaue
    END

调用 和 输出:

call pro_name (123@outParaVlaue);

select @outParaVlaue-- 调用并且会输出@outParaVlaue

另外:

show create procedure 存储过程名字 -- 显示此存储过程的创建语句
show procedure status -- 列出所有存储过程

删除:drop procedure procedurename

触发器

我们还有一种需求:事件(INSERT UPDATE DELETE)发生的前后自动执行某些语句,此时就是触发器的作用了。

CREATE TRIGGER <触发器名称>  -- 触发器必须有名字,最多64个字符,可能后面会附有分隔符.它和MySQL中其他对象的命名方式基本相象.
{ BEFORE | AFTER }  -- 触发器有执行的时间设置:可以设置为事件发生前或后。
{ INSERT | UPDATE | DELETE }  -- 同样也能设定触发的事件:它们可以在执行insert、update或delete的过程中触发。
ON <表名称>  -- 触发器是属于某一个表的:当在这个表上执行插入、 更新或删除操作的时候就导致触发器的激活. 我们不能给同一张表的同一个事件安排两个触发器。
FOR EACH ROW  -- 触发器的执行间隔:FOR EACH ROW子句通知触发器 每隔一行执行一次动作,而不是对整个表执行一次。
<触发器SQL语句>  -- 触发器包含所要触发的SQL语句:这里的语句可以是任何合法的语句, 包括复合语句,但是这里的语句受的限制和函数的一样。
-- 必须拥有相当大的权限才能创建触发器(CREATE TRIGGER),如果你已经是Root用户,那么就足够了。这跟SQL的标准有所不同。

一个表最多6个触发器,插入删除更新的前后。

create trigger trigger_Name 
after 
insert on tabe1 for each row 
begin
  -- 要做的一些操作 
  insert into tab2(tab2_id) values(new.tab1_id);
end

在触发器中可以引用NEW新的虚拟表,访问插入的行。可以引用OLD虚拟表,访问被删除的行。

注意: 只有表支持触发器,视图不支持; 触发器中不能调用存储过程。

删除触发器:DROP TRIGGER trigger_Name

 MySql引擎

使用命令,查看库的所有引擎:

show storage Engines

 

查看目的表的引擎方法:

show create table 表名

新建标的时候,在create语句的最后,加上 ENGINE =引擎名,可为表指定引擎。

修改表的引擎:

alter table 表名 engine=引擎名;

 MyISAM读操作性能好;InnoDB支持事务,写操作更好并发度。

 MyISAMInnoDB
存储文件 三个文件组成。.frm(存储表定义).MYD (存储数据文件)。.MYI (存储索引)。 所有的表都保存在同一个数据文件中(也可能是多个文件,或者是独立的表空间文件),InnoDB表的大小只受限于操作系统文件的大小,一般为2GB。
外键 不支持 支持
事务 不支持 支持
自增列 必须是索引列,不需要一定是组合索引的第一列 必须是索引,若是组合索引,必须是组合索引的第一列
查询行数 存储了表的行数,查找时候很快 必须扫描全表
增删改查优势 适合查询 新增和修改
表锁 行锁、表锁
删除表数据 重建表方式 一行一行删
存储空间 表支持3种存储格式,静态表(默认存储格式,每条记录固定长度),动态表(每条记录不是固定长度,存储的空间减少了),压缩表(每条记录被单独压缩,占据空间更少) 需要更多的内存和存储,它会在主内存中建立其专用的缓冲池用于高速缓冲数据和索引
可移植性 数据以文件形式存储,跨平台转移方便 所有表都保存在同一个数据文件(也可能是多个文件),相对来说不好备份
posted @ 2017-08-01 14:54  Eric-Lee  阅读(341)  评论(0编辑  收藏  举报