mysql update sysdate_MySQL

MySQL

查询系统时间

SELECT SYSDATE();

R: 2018-01-26 14:05:41

注释 -- xxxxxxxxxxxxxx

SQL分类

数据定义语言DDL

数据操作语言DML

事物控制语言TCL

数据查询语言DQL

数据控制语言DCL

事务控制语言

用来维护数据一致性的语句

SQL 语句中关键字大写 非关键字小写

MySQL 数据类型

数值类型:

MySQL支持所有标准SQL数值数据类型。

这些类型包括严格数值数据类型(INTEGER、SMALLINT、DECIMAL和NUMERIC),以及近似数值数据类型(FLOAT、REAL和DOUBLE PRECISION)。

关键字INT是INTEGER的同义词,关键字DEC是DECIMAL的同义词。

BIT数据类型保存位字段值,并且支持MyISAM、MEMORY、InnoDB和BDB表。

作为SQL标准的扩展,MySQL也支持整数类型TINYINT、MEDIUMINT和BIGINT。下面的表显示了需要的每个整数类型的存储和范围。

类型大小范围(有符号)范围(无符号)用途

TINYINT1 字节(-128,127)(0,255)小整数值

SMALLINT2 字节(-32 768,32 767)(0,65 535)大整数值

MEDIUMINT3 字节(-8 388 608,8 388 607)(0,16 777 215)大整数值

INT或INTEGER4 字节(-2 147 483 648,2 147 483 647)(0,4 294 967 295)大整数值

BIGINT8 字节(-9 233 372 036 854 775 808,9 223 372 036 854 775 807)(0,18 446 744 073 709 551 615)极大整数值

FLOAT4 字节(-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)0,(1.175 494 351 E-38,3.402 823 466 E+38)单精度浮点数值

DOUBLE8 字节(-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)0,(2.225 073 858 507 201 4 E-308,1.797 693 134 862 315 7 E+308)双精度浮点数值

DECIMAL对DECIMAL(M,D) ,如果M>D,为M+2否则为D+2依赖于M和D的值依赖于M和D的值小数值

a指定指定小数点左边和右边可以存储的十进制数字的最大个数,最大精度38。

b指定小数点右边可以存储的十进制数字的最大个数。小数位数必须是从 0 到 a之间的值。默认小数位数是 0。

float:浮点型,含字节数为4,32bit,数值范围为-3.4E38~3.4E38(7个有效位)

double:双精度实型,含字节数为8,64bit数值范围-1.7E308~1.7E308(15个有效位)

decimal:数字型,128bit,不存在精度损失,常用于银行帐目计算。(28个有效位)

日期和时间类型

表示时间值的日期和时间类型为DATETIME、DATE、TIMESTAMP、TIME和YEAR。

每个时间类型有一个有效值范围和一个"零"值,当指定不合法的MySQL不能表示的值时使用"零"值。

TIMESTAMP类型有专有的自动更新特性,将在后面描述。

类型大小(字节)范围格式用途

DATE31000-01-01/9999-12-31YYYY-MM-DD日期值

TIME3'-838:59:59'/'838:59:59'HH:MM:SS时间值或持续时间

YEAR11901/2155YYYY年份值

DATETIME81000-01-01 00:00:00/9999-12-31 23:59:59YYYY-MM-DD HH:MM:SS混合日期和时间值

TIMESTAMP41970-01-01 00:00:00/2038结束时间是第 2147483647 秒,北京时间 2038-1-19 11:14:07,格林尼治时间 2038年1月19日 凌晨 03:14:07YYYYMMDD HHMMSS混合日期和时间值,时间戳

字符串类型

字符串类型指CHAR、VARCHAR、BINARY、VARBINARY、BLOB、TEXT、ENUM和SET。该节描述了这些类型如何工作以及如何在查询中使用这些类型。

CHAR和VARCHAR类型类似,但它们保存和检索的方式不同。它们的最大长度和是否尾部空格被保留等方面也不同。在存储或检索过程中不进行大小写转换。

BINARY和VARBINARY类类似于CHAR和VARCHAR,不同的是它们包含二进制字符串而不要非二进制字符串。也就是说,它们包含字节字符串而不是字符字符串。这说明它们没有字符集,并且排序和比较基于列值字节的数值值。

BLOB是一个二进制大对象,可以容纳可变数量的数据。有4种BLOB类型:TINYBLOB、BLOB、MEDIUMBLOB和LONGBLOB。它们只是可容纳值的最大长度不同。

有4种TEXT类型:TINYTEXT、TEXT、MEDIUMTEXT和LONGTEXT。这些对应4种BLOB类型,有相同的最大长度和存储需求。

类型大小用途

CHAR0-255字节定长字符串

VARCHAR0-65535 字节变长字符串

TINYBLOB0-255字节不超过 255 个字符的二进制字符串

TINYTEXT0-255字节短文本字符串

BLOB0-65 535字节二进制形式的长文本数据

TEXT0-65 535字节长文本数据

MEDIUMBLOB0-16 777 215字节二进制形式的中等长度文本数据

MEDIUMTEXT0-16 777 215字节中等长度文本数据

LONGBLOB0-4 294 967 295字节二进制形式的极大文本数据

LONGTEXT0-4 294 967 295字节极大文本数据

CREATE DATABASE emp;

--创建emp库 数据定义语言DDL

-- DDL 是对数据库对象进行操作的语言 数据库对象包括:表、视图、索引、序列

CREATE TABLEemployee(

idINTUNSIGNED AUTO_INCREMENT,

nameVARCHAR(20) NOT NULL,

genderCHAR(1)DEFAULT 'M',

birth DATE,

salaryDECIMAL(6,2),

jobVARCHAR(30),

deptnoINTUNSIGNED,PRIMARY KEY( `id` )

)ENGINE=InnoDB DEFAULT CHARSET=utf8;

-- 创建表 employee id 为无符号int 类型,自增 主键 ,name varchar(20) 可变长度20 不可为空 ,gender char(1)定长 1 默认值为 'M' ,birth date类型 日期 ,salary DECIMAL类型 6位小数两位,job varchar 30 deptno int无符号 表类型Inn'o'DB 字符类型utf8

DESC employee

-- 查看表结构

DROP TABLE employee

--删除表

RENAME TABLE employee TO myemp

--表重命名

ALTER TABLE myemp ADD sex CHAR(1) NOT NULL;

-- 向表中添加一个字段 添加到最后

ALTER TABLE myemp ADD sex2 CHAR(1) FIRST;

--FIRST 添加到最前面

ALTER TABLE myemp ADD sex3 CHAR(1) AFTER name;

-- AFTER 添加到某个字段后面

ALTER TABLE myemp DROP sex;

-- 删除一个字段

MySQL目前不支持列的Default 为函数的形式,

如达到你某列的默认值为当前更新日期与时间的功能,

你可以使用TIMESTAMP列类型

下面就详细说明TIMESTAMP列类型

ALTER TABLE myemp ADD hiredate2 TIMESTAMP DEFAULT NOW();

修改字段

可以修改字段的类型,长度,默认值是否非空,修改表结构都应当避免在表中有时间以后进行,

若表中有数据 修改表中字段时尽量不要修改类型修改长度尽量增大

ALTER TABLE myemp MODIFY name VARCHAR(100);

-- 修改字段长度

ALTER TABLE myemp CHANGE gender sex CHAR(1)

-- CHANGE old new

DML 语句

是对表中的数据进行的操作

伴随事务控制

增、删、改

INSERT INTO myemp (id,name,salary,deptno) VALUES (1,'jack',5000,10)

-- 插入一条数据

INSERT INTO myemp (name,salary,deptno,birth) VALUES ('jack',5000,10,now())

-- 插入一条数据

INSERT INTO myemp (name,salary,deptno,birth) VALUES ('jack',5000,10,'2018-01-02')

-- 插入一条数据

mysql日期和字符相互转换方法

date_format(date,'%Y-%m-%d') -------------->oracle中的to_char();

str_to_date(date,'%Y-%m-%d') -------------->oracle中的to_date();

%Y:代表4位的年份

%y:代表2为的年份

%m:代表月, 格式为(01……12)

%c:代表月, 格式为(1……12)

%d:代表月份中的天数,格式为(00……31)

%e:代表月份中的天数, 格式为(0……31)

%H:代表小时,格式为(00……23)

%k:代表 小时,格式为(0……23)

%h: 代表小时,格式为(01……12)

%I: 代表小时,格式为(01……12)

%l :代表小时,格式为(1……12)

%i: 代表分钟, 格式为(00……59)

%r:代表 时间,格式为12 小时(hh:mm:ss [AP]M)

%T:代表 时间,格式为24 小时(hh:mm:ss)

%S:代表 秒,格式为(00……59)

%s:代表 秒,格式为(00……59)

SELECT DATE_FORMAT(20130111191640,'%Y-%m-%d %H:%i:%s')

DATE_FORMAT(20130111191640,'%Y-%m-%d %H:%i:%s')

UPDATE myemp SET salary=6000;

--修改数据

UPDATE myemp SET salary=6000 WHERE id=1;

--修改数据 使用条件

DELETE FROM myemp;

--删除数据

DELETE FROM myemp WHERE id =2;

--删除数据 使用条件

程度从强到弱

1、drop table tb

drop将表格直接删除,没有办法找回

2、truncate (table) tb

删除表中的所有数据,不能与where一起使用

3、delete from tb (where)

删除表中的数据(可制定某一行)

区别:truncate和delete的区别

1、事务:truncate是不可以rollback的,但是delete是可以rollback的;

原因:truncate删除整表数据(ddl语句,隐式提交),delete是一行一行的删除,可以rollback

2、效果:truncate删除后将重新水平线和索引(id从零开始) ,delete不会删除索引

3、 truncate 不能触发任何Delete触发器。

4、delete 删除可以返回行数

各种函数和用法

SELECT name,salary*12 FROM myemp;

-- 年薪

SELECT CONCAT('name:',name) FROM myemp;

-- CONCAT 字符串连接函数

SELECT CONCAT(name,salary) FROM myemp

mysql CONCAT(str1,str2,…)

返回结果为连接参数产生的字符串。

如有任何一个参数为NULL ,则返回值为 NULL。

或许有一个或多个参数。 如果所有参数均为非二进制字符串,则结果为非二进制字符串。

如果自变量中含有任一二进制字符串,则结果为一个二进制字符串。

一个数字参数被转化为与之相等的二进制字符串格式;若要避免这种情况,可使用显式类型 cast,

例如: SELECT CONCAT(CAST(int_col AS CHAR), char_col)

CONCAT_WS(separator,str1,str2,…)

CONCAT_WS() 代表 CONCAT With Separator ,是CONCAT()的特殊形式。 第一个参数是其它参数的分隔符。

分隔符的位置放在要连接的两个字符串之间。分隔符可以是一个字符串,也可以是其它参数。

如果分隔符为 NULL,则结果为 NULL。函数会忽略任何分隔符参数后的 NULL 值。

SELECT CONCAT(name,',',salary) FROM myemp

--两个相等

SELECT CONCAT_WS(',',name,salary) FROMmyempSELECT 'zong' || 'xuan' FROM DUAL;

-> 0

--MySQL 没有 || 连接字符串

SELECT LENGTH(salary) FROMmyempSELECT LENGTH(name) FROM myemp

-- LENGTH()字符串的长度

SELECT UPPER('LiSan') ,LOWER('LiSan')

-- UPPER 转大写 LOWER 转小写

--MySQL 没有首字母大写函数

SELECT 1 + 1 FROM DUAL;

--DUAL 伪表 5.1以后

SELECT TRIM('zong' from 'zongxuan') FROM DUAL;

-> xuan

-- TRIM 把前面的从全部后面拿掉 不管有几次

SELECT TRIM('zongxuan') FROm DUAL;

->zongxuan

--TRIM 去空格

SELECT TRIM(LEADING 'x' FROM 'xxxbarxxx');

-> 'barxxx'

--删除指定的首字符 x

SELECT TRIM(BOTH 'x' FROM 'xxxbarxxx');

-> 'bar'

--删除指定的首尾字符 x

SELECT TRIM(TRAILING 'xyz' FROM 'barxxyz');

-> 'barx'

--删除指定的尾字符 x

SELECT LTRIM('barbar');

-- 左空格

SELECT RTRIM('barbar');

-- 右空格

SELECT name,LPAD(salary,7,'$') FROM myemp

-- 补位函数

SELECT name,LPAD(salary,7,' ') FROM myemp

--右对齐

SELECT name,RPAD(salary,10,' ') FROM myemp

--左对齐

LPAD(str,len,padstr)

用字符串 padstr对 str进行左边填补直至它的长度达到 len个字符长度,然后返回 str。如果 str的长度长于 len',那么它将被截除到 len个字符。

mysql> SELECT LPAD('hi',4,'??'); -> '??hi'RPAD(str,len,padstr)

用字符串 padstr对 str进行右边填补直至它的长度达到 len个字符长度,然后返回str。如果 str的长度长于 len',那么它将被截除到 len个字符。

mysql> SELECT RPAD('hi',5,'?'); ->'hi???'SELECT SUBSTR('zongxuan',5,4) FROM DUAL;

-> xuan

--SUBSTR 字符串截取 从第5个开始截取4个字符

SELECT SUBSTR('zongxuan',-4,4)

-- 倒数第4个开始截取4个字符

SELECT INSTR('zongxuan','n')

-- 查找 n 在zongxuan 中的位置 没有 更高级的用法

SELECT ROUND('123.123456',0) FROM DUAL;

SELECT ROUND('123.123456',-1) FROM DUAL;

SELECT ROUND('123.123456',5) FROM DUAL;

--正数 四舍五入到小数点后第m位

--0 四舍五入到整数位

--负数 四舍五入到小数点前m位

--MySQL 没有 TRUNC 函数 Oracle有

SELECT MOD(salary,1000) FROM myemp

-- 求余数

SELECT CEIL(45.678) FROM DUAL

--向上取整

SELECT FLOOR(45.678)

--向下取整

SELECT SYSDATE() FROM DUAL;

--系统时间

now()函数的同义词有:CURRENT_TIMESTAMP 、 CURRENT_TIMESTAMP()、LOCALTIMESTAMP 、 LOCALTIMESTAMP()、LOCALTIME 、 LOCALTIME()

SYSDATE( ):返回服务器的当前日期和时间

与now的不同点:(一般使用NOW而不用SYSDATE)

SYSDATE()返回的是函数执行时的时间

now()返回的是语句执行时的时间

SELECT CURTIME() FROMDUAL;--CURTIME() 返回当前时间 只包含时分秒

SELECT CURDATE() FROMDUAL;--CURDATE():返回当前日期,只包含年月日

SELECT TIMEDIFF(NOW(),'2018-01-31 10:59:00') FROMmyemp;--TIMEDIFF(expr1, expr2):返回两个日期相减(expr1 − expr2 )相差的时间数(两个参数类型必须相同)--是时间 不是日期

SELECT DATEDIFF(NOW(),'2018-02-05 10:59:00') FROMmyemp;--DATEDIFF(expr1, expr2):返回两个日期相减(expr1 − expr2 )相差的天数--是天数

SELECT DATE_ADD(NOW(),interval 1HOUR)--加一小时

SELECT DATE_ADD(NOW(),interval 1MINUTE)--加一分钟

SELECT DATE_ADD(NOW(),interval 1SECOND)--加一秒钟

SELECT DATE_ADD(NOW(),interval 1MICROSECOND)--加一毫秒

HOUR 小时| MINUTE 分 | SECOND 秒 | MICROSECOND 毫秒

YEAR 年 | MONTH 月 | DAY 日 | WEEK 周 | QUARTER 季

YEAR_MONTH 年和月 | DAY_HOUR 日和小时 | DAY_MINUTE 日和分钟 | DAY_SECOND 日和秒

HOUR_MINUTE 小时和分 | HOUR_SECOND 小时和秒 | MINUTE_SECOND分钟和秒

SELECT '2008-12-31 23:59:59' + INTERVAL 1SECOND;--不使用函数,也可以写表达式进行日期的加减

6、选取日期时间的各个部分:日期、时间、年、季度、月、日、小时、分钟、秒、微秒(常用)

SELECT now(),date(now()); --日期

SELECT now(),time(now()); --时间

SELECT now(),year(now()); --年

SELECT now(),quarter(now()); --季度

SELECT now(),month(now()); --月

SELECT now(),week(now()); --周

SELECT now(),day(now()); --日

SELECT now(),hour(now()); --小时

SELECT now(),minute(now()); --分钟

SELECT now(),second(now()); --秒

SELECT now(),microsecond(now()); --微秒

EXTRACT(unit FROM date):从日期中抽取出某个单独的部分或组合

SELECT now(),extract(YEAR FROM now()); --年

SELECT now(),extract(QUARTER FROM now()); --季度

SELECT now(),extract(MONTH FROM now()); --月

SELECT now(),extract(WEEK FROM now()); --周

SELECT now(),extract(DAY FROM now()); --日

SELECT now(),extract(HOUR FROM now()); --小时

SELECT now(),extract(MINUTE FROM now()); --分钟

SELECT now(),extract(SECOND FROM now()); --秒

SELECT now(),extract(YEAR_MONTH FROM now()); --年月

SELECT now(),extract(HOUR_MINUTE FROM now()); --时分

UNIX_TIMESTAMP() 当前系统时间戳

个性化显示时间日期

SELECTnow(),dayofweek(now());--dayofweek(date)--一周的第几天

SELECTnow(),dayofmonth(now());--dayofmonth(date)--一月的第几天

selectnow(),dayofyear(now());--dayofyear(date)--一年的第几天

set lc_time_names='zh_CN';--设定 名称是中文or英文的由系统变量lc_time_names控制(默认值是'en_US')

selectdayname(now())--dayname() 返回日期的星期

selectdayname(now()),monthname(now());--monthname()月份名称

show variableslike 'lc_time_names';--名称是中文or英文的由系统变量lc_time_names控制(默认值是'en_US')

--0时区 -> 东八区

SELECT CONVERT_TZ('2014-01-01 01:00:00','+00:00','+8:00') aa;--西八区 -> 东八区

SELECT CONVERT_TZ('2014-01-01 01:00:00','-8:00','+8:00') aa;-增加时间select ADDTIME(NOW(), '30:0:0') aa,ADDTIME('10:10:10', '30:0:0') bb;select ADDTIME(NOW(), '1 1:1:1.000002') aa,ADDTIME('10:10:10', '1 1:1:1.000002') bb;select ADDTIME(NOW(), '100 01:01:01.000002') aa,ADDTIME('10:10:10', '100 01:01:01.000002') bb;--当前时间戳

select CURRENT_TIMESTAMP() a1,CURRENT_TIMESTAMPa2,NOW() a3;selectNOW() a1,LOCALTIME a2,LOCALTIME() a3,LOCALTIMESTAMP a4,LOCALTIMESTAMP () a5;--两个时间相减(日期部分相减,返回天数)

select DATEDIFF('2014-1-1 10:00:00','2014-2-2 01:00:00') aa;select TIMEDIFF('2014-1-1 10:10:10','2014-2-2 01:20:30') aa;SELECT TIMESTAMPDIFF(YEAR,'2003-02-01','2003-05-01') aa;SELECT TIMESTAMPDIFF(MONTH,'2003-02-01','2003-05-01') aa;SELECT TIMESTAMPDIFF(DAY,'2003-02-01','2003-05-01');SELECT TIMESTAMPDIFF(HOUR,'2003-02-01','2003-05-01');--从数字数据获取时间

SELECT FROM_DAYS(730669.89) aa;SELECT FROM_UNIXTIME(1196440219.12) aa;SELECT FROM_UNIXTIME(1196440219) + 0aa;SELECT FROM_UNIXTIME(UNIX_TIMESTAMP(),'%Y %D %M %h:%i:%s %x');--LAST_DAY获取当前月份的最后一天

SELECT LAST_DAY('2014-02-05') aa;SELECT LAST_DAY('2014-02-05 11:11:11') aa;SELECTLAST_DAY(NOW()) aa;--UTC时间获取

selectUTC_DATE() a1,UTC_TIME() a2,UTC_TIMESTAMP() a3;--时间戳加减

SELECT TIMESTAMPADD(YEAR,1,'2014-01-02') aa;SELECT TIMESTAMPADD(MONTH,1,'2014-01-02') aa;SELECT TIMESTAMPADD(DAY,1,'2014-01-02') aa;SELECT TIMESTAMPDIFF(YEAR,NOW(),'2014-01-02') aa;SELECT TIMESTAMPDIFF(MONTH,NOW(),'2014-01-02') aa;SELECT TIMESTAMPDIFF(DAY,NOW(),'2014-01-02') aa;--时间格式化

select TIME_FORMAT('08:08:01','%h:%i:%s') aa;select TIME_FORMAT(current_time,'%H:%m:%s') aa;--秒换算时间

select SEC_TO_TIME(60) aa;select SEC_TO_TIME(3600) aa;select TIME_TO_SEC(current_time) aa;select TIME_TO_SEC('00:10:10') aa;--计算月份差

SELECT PERIOD_DIFF(201402,201403) aa;SELECT PERIOD_DIFF('201402','201403') aa;--增加月份

select PERIOD_ADD('201401',3) aa;select PERIOD_ADD('1401',3) aa;--创建日期、时间

select MAKEDATE(2014,10) aa;select MAKETIME(13,1,6) aa;

NULL

SELECT * FROM myemp WHERE birth IS NULL

SELECT * FROM myemp WHERE birth IS NOT NULL

--空值判断

NULL与任何数字运算还是NULLNULL与字符串拼接等于什么都没干SELECT IFNULL(null,'123')--IFNULL(expr1,expr2)--如果expr1不是NULL,IFNULL()返回expr1,否则它返回expr2。IFNULL()返回一个数字或字符串值,取决于它被使用的上下文环境。--MySQL 没有NVL()

SELECT name,salary+IFNULL(comm,0) FROMmyempSELECT name,salary,IF(comm IS NULL,'N','Y') FROMmyemp--MySQL 没有NUVL2() 用这个替换

SELECT name AS 姓名 FROMmyemp--AS 别名

SELECT name AS 姓名 FROM myemp WHERE id=1;--WHERE 查询条件--> 大于 < 小于 >= 大于等于 <= 小于等于 != 不等于 <> 不等于 = 等于

SELECT name FROM myemp WHERE id=1 OR id=2;--AND 并 OR 或 AND的优先级高于OR 使用() 来提高OR的优先级

SELECT * FROM myemp WHERE name LIKE '_zong%';--LIKE 模糊查询 两个通配符 % 和_ % 0到任意个 _任意一个

SELECT * FROM myemp WHERE jobid IN (1,2,3,4,5);SELECT * FROM myemp WHERE jobid NOT IN (1,2,3,4,5);--IN 和NOT IN 判断是否在列表中或不在列表中

SELECT * FROM myemp WHERE salary BETWEEN 1500 AND 3000;--BETWEEN AND 在一个区间

SELECT * FROM myemp WHERE deptno < ALL(SELECT comm FROMmyemp)SELECT * FROM myemp WHERE deptno > ANY(SELECT comm FROMmyemp)--ANY 和ALL 不能单独使用,需要配合比较操作符 > >= < <= 一起使用-->ANY 大于最小 ALL 大于最大

SELECT DISTINCT job FROMmyemp;--DISTINCT 去重

SELECT DISTINCT job,deptno FROMmyemp;--多字段去重 是指的job 和deptno 的组合没有重复的

SELECT DISTINCT count_time ,id c FROMes_gwSELECT COUNT(DISTINCT count_time ,id) c FROMes_gwSELECT * FROM myemp ORDER BYid--ORDER BY ASC 排序 升序 默认的--ORDER BY DESC 排序 降序--ORDER BY 写在最后--NULL 被认作 最大值

SELECT * FROM myemp ORDER BYdeptno,salary--多字段排序 先排第一个字段,在拍第二个字段

SELECT * FROM myemp ORDER BY deptno DESC,salarySELECT * FROM myemp ORDER BY deptno DESC ,salary DESC

聚合函数

MAX MIN COUNT AVG SUM

--聚合函数 对于NULL值都是忽略的

SELECT AVG(IFNULL(comm,0)) FROMmyemp--对于 NULL都是忽略 所以有部分的需求就是要 将NULL 转为 0

分组

配合聚合函数进行更细分的统计工作

GROUP BY

SELECT AVG(salary),deptno FROM myemp GROUP BYdeptno--按照 deptno 分组 统计 每个部门的评价工资

SLELCT deptno,job,AVG(salary) FROM myemp GROUP BYdeptno,job--按照多字段分组 就是 多个字段的组合为一组

当SELECT 子句中含有聚合函数时 那么凡看不在聚合函数中的其他单独字段都必须出现在GROUP BY 自居中, 反过来则不是必须的

-----------------------------------------------------------------------

SLELCT deptno,job,AVG(salary) FROM myemp GROUP BY deptno,job

SELECTR 中有deptno 和job 那么 GROUP BY 必须有 deptno 和job

SLELCT job,AVG(salary) FROM myemp GROUP BY deptno,job

GROUP BY 有 deptno 和job 那么 SELECTR 可以没有deptno

-----------------------------------------------------------------------

SELECT m_ip,COUNT(*) AS count_4ip FROM tou_connection2 GROUP BYm_ip--ip 登陆数统计

SELECT MAX(count_4ip) FROM (SELECT m_ip,COUNT(*) AS count_4ip FROM tou_connection2 GROUP BY m_ip) AS TEMP

--ip登陆数最大的

SELECT m_ip,COUNT(*) FROM tou_connection2 GROUP BY m_ip HAVING COUNT(*)= (SELECT MAX(count_4ip) FROM (SELECT m_ip,COUNT(*) AS count_4ip FROM tou_connection2 GROUP BY m_ip) AS TEMP)--ip 和 登陆数 HAVING 作为分组的条件查询

WHERE 中不能使用聚合函数作为过滤条件,过滤的时间不对 WHERE 是在数据库检索表中的数据时,对数据逐条过滤以决定是否查询出来该数据 所以 WHERE 时用来确认结果集的数据

使用聚合函数的结果作为过滤条件, 那么一定时数据从表中查询完毕(WHERE是查询过程中有用)得到结果及,并且分组完毕才进行聚合函数统计结果得到后才可以对分组进行过滤 这个过滤的时机是在where之后的

查询语句执行的顺序

SQL 语句的优化:

FROM 子句 从后往前 从右到左 数据量较少的表放后面

WHERE 子句 想下而上 从右到左 将能过滤最大数量记录的条件写在 WHERE 的最右

GROUP BY 子句 从左到右分组最好在GROUP BY 前使用WHERE 将不需要的记录在GROUP BY之前过滤掉

HAVING 子句 消耗资源 尽量避免使用 HAVING 会在检索出所有记录后才 对结果集进行过滤 需要排序等 操作

SELECT 子句 少使用*

ORDER BY子句 执行顺序从左到右 排序 消耗资源

关联查询

从多张表中查询对应记录的信息 关联查询的重点在于这些表中的记录的对应的关系,这个关系也称为连接条件

SELECTename,dnameFROMemp,deptWHERE emp.deptno=deptno.deptno

关联查询要添加连接条件,否则会产生笛卡儿积 通常是一个 无意义的结果集,他的记录是所有参与查询的表的记录数乘积的结果 要避免 数据量大的时候极易出现内存溢出等

N张表关联查询要右至少N-1个连接条件

INNER JOIN(内连接,或等值连接):获取两个表中字段匹配关系的记录。 (所有满足条件的)

LEFT JOIN(左连接):获取左表所有记录,即使右表没有对应匹配的记录。()

RIGHT JOIN(右连接): 与 LEFT JOIN 相反,用于获取右表所有记录,即使左表没有对应匹配的记录。

MySQL 不支持 全连接 :但是可以通过UNION 关键字来合并 LEFT JOIN 与 RIGHT JOIN来模拟FULL join.

select * from A left join B on B.name = A.name union select * from A right join B on B.name =A.name;--全连接

SELECTe.name,d.dnameFROM emp e JOINdept dON e.deptno=d.deptnoWHERE e.job='asd'

select * from A right join B on A.name =B.name;--右外连接

select * from A left join B on A.name=B.name where A.id is not null and B.id is not null;--左外连接

MySQL 不支持 (+) 代替 左右连接

自连接

当前表的的一条记录可以对应当前表自己的多条记录

自连接是为了解决同类型数据但是又存在上下级关系的树状结构数据使用

同类型数据有上下级

SELECT e.ename,m.ename,d.loc FROM emp e , emp m , dept d WHERE e.mgr=m.empno AND m.deptno=d.deptno AND e.ename='SMITH'

--JOIN ON 更好 更明了

SELECT e.ename,m.ename,d.loc FROM emp e JOIN emp m ON e.mgr=m.empno JOIN dept d ON m.deptno=d.deptno WHERE e.ename='SMITH'

同一张表可以看作员工表 有可以看作 领导表

表中有员工编号 又有 领导编号

子查询

子查询是一条SELECT 语句 但它是嵌套在其他SQL 语句中为的是给该SLQL 提供数据以支持其执行操作

SELECTename,salFROMempWHERE sal>(SELECT sal FROM emp WHERE ename='CLARK')CREATE TABLEtestAS

SELECTbts.id,b.name,bts.trs_nameFROM brokers AS b INNER JOIN broker_trade_services AS bts ON b.id=bts.broker_id--查询结果创建表

--使用 子查询 创建表的时候 字段有使用运算或者函数 必须使用别名DELETE FROMempWHERE deptno=(SELECTdeptnoFROMempWHERE name='CLARK')SELECTename,jobFROMempWHERE sal>(SELECT AVG(sal) FROM emp)

子查询根据查询结果集的不同分为

单行单列子查询 常用月过滤条件 可以配合 = > >= < <=使用

多行单列子查询 常用于过滤条件 由于查询多多个值 在判断=时要用IN 判断 > >= < <= 要配合 ANY ALL

多行多列子查询 那就是一张表

SELECTenameFROMempWHERE sal > ALL(SELECT sal FROM emp WHERE job IN('CLERK','SALESMAN'))

EXISTS 后面跟一个子查询 当该子查询可以查询出来至少一条记录时,则 EXISTS 表达式成立并返回true

SELECT MIN(sal)FROMempGROUP BYdeptnoHAVING MIN(sal) > (SELECT MIN(sal)FROMempWHERE deptno=30)

SELECT e.ename,e.sal,e.deptno,temp.avg_salFROM emp AS e LEFT JOIN (SELECT AVG(sal) ASavg_sal,deptnoFROMempGROUP BY deptno) AS temp ON e.deptno=temp.deptnoHAVING e.sal>temp.avg_sal

在SELECT 子句中使用子查询 可以将查询结果当作外层查询记录中的一个字段值显示, 这样就实现了外连接

分页查询

分页查看是将查询表中数据时分段查询,而不是一次性将所有数据查询出来 优势查询的数据两非常大 这回导致系统资源消耗大 响应时间长,数据冗余严重

所以我们要使用分页查询 数据库都支持分页 但是没有标准 所以不同的数据库有不同的方法

MySQL 使用limit 函数分页

SELECT * FROM emp LIMIT 0,3

--前三个

SELECT * FROM emp ORDER BY sal DESC LIMIT 6,5

--第6到第10 从第几个开始取,取多少个

DECODE 函数

MySQL 没有 可以用IF 来代替

row_number()

MySQL 没有

rank()

MySQL 没有

dense_rank

MySQL 没有

集合操作符

并 交 差

UNION 并

A表中有 B表中没有 出现一次

A表中没有 B表中有 出现一次

A表中有 B表中有 出现一次

A表中没有 B表中没有 不出现

UNION ALL 全并

A表中有 B表中没有 出现一次

A表中没有 B表中有 出现一次

A表中有 B表中有 出现两次

A表中没有 B表中没有 不出现

INTERSECT 交

A表中有 B表中没有 不出现

A表中没有 B表中有 不出现

A表中有 B表中有 出现一次

A表中没有 B表中没有 不出现

MINUS 差

A表中有 B表中没有 出现一次

A表中没有 B表中有 不出现

A表中有 B表中有 不出现

A表中没有 B表中没有 不出现

多条左集合操作的SELECT语句的列的个数和数据类型必须匹配

ORDER BY 只能放在最后一个查询语句中

高级分组函数

高级分组函数用在GROUP BY 子句中每个高级分组函数都有一套分组策略

MySQL 生成随机数

RAND()

SELECT CEIL(RAND()*2+2010)

生成2011-2012的随机数

ROLLUP 分组原则参数主次递减 一直到所有参数都不要,每一种分组都统计一次结果并且并在一个结果集显示.、

2、rollup注意:

1)ORDER BY不能在rollup中使用,两者为互斥关键字;

2)如果分组的列包含NULL值,那么rollup的结果可能不正确,因为在rollup中进行的分组统计时,null具有特殊意义。因此在进行rollup时可以先将

null转换成一个不可能存在的值,或者没有特别含义的值,比如:IFNULL(xxx,0)

3)mysql中没有像oracle那样的grouping()函数;

CREATE TABLE`sales_tab` (

`year_id`int(11) NOT NULL,

`month_id`int(11) NOT NULL,

`day_id`int(11) NOT NULL,

`sales_value`int(11) NOT NULL) ENGINE=InnoDB DEFAULT CHARSET=utf8

INSERT INTOsales_tabSELECT CEIL(RAND()*2+2010) ASyear_id,

CEIL(RAND()*12) ASmonth_id,

CEIL(RAND()*31) ASday_id,RAND()*100+210 ASsales_valueFROM dual

delimiter $$create procedureinsert_info()begin

DECLARE done INT DEFAULT 0;while(done<=23) doINSERT INTO sales_tab SELECT CEIL(RAND()*2+2010) AS year_id,CEIL(RAND()*12) AS month_id,CEIL(RAND()*31) AS day_id,RAND()*100+210 AS sales_value FROMdual;set done=done+1;end while;end$$

call insert_info();

SELECT year_id,month_id,day_id,SUM(sales_value) FROM sales_tab GROUP BY year_id,month_id,day_id WITHROLLUP--和ORACLE 的用法不一样 效果一样

CUBE() 每种组合分一次组

分组次数是2的参数个数次方

但是MySQL 只定义了 没有实现

视图

视图 VIEW 也被称作虚表 是一个数据的逻辑表示

视图对应于一条SELECT 语句 结果集被赋予一个名字 就是视图名字

视图本身不包含任何数据 它只包含映射到基表的一个查询语句 当基表的数据发生变化 视图数据也随之变化

视图在SQL语句中体现的角色与表相同但是视图并不是一张真实存在的表而只是对应一个SELECT 语句的查询结果集

并将它当作表来看待

使用视图的目的是简化SQL语句的复杂度重用了查询,限制数据访问 视图本质就是一条SELECT 语句 所以当访问视图的时候只能

访问到对应的SELECT语句中涉及的列,对基表中其他列起到安全和保密的作用

CREATE VIEW

--创建视图

CREATE VIEWv_emp_10AS

SELECT empno,ename,sal,deptno FROM emp WHERE deptno=10;--创建一个视图 视图的名字一般用v_ 开始

SELECT * FROMv_emp_10--在视图中查询

修改视图

OR REPLACE 短语修改视图对应的SLQL 查询语句

CREATE OR REPLACE VIEWv_emp_10AS

SELECT empno id,ename name,sal salary,deptno FROMempWHERE deptno=10;

视图分类

简单视图

--最简单的没有函数没有处理

复杂视图

--有函数之类的

连接视图

--多个表联查

对视图进行DML操作

仅能对简单视图进行DML操作

对视图进行操作可能对基表进行数据污染

WITH CHECK OPTION

--通过视图所做的就该必须在视图的可见范围内 假设INSERT 新增的记录在视图仍可查看

假设UPDATE 修改后的结果必须能通过视图查看到

为视图添加只读选项

WITH READ ONLY

--只能看不能改

DROP VIEW v_emp_10

--删除视图 不会影响基表数据

数据字典

--MySQL 数据库中的一些信息 比如 有多少表,表的名字

information_schema库中

SCHEMATA表:提供了当前mysql实例中所有数据库的信息。是show databases的结果取之此表。

TABLES表:提供了关于数据库中的表的信息(包括视图)。详细表述了某个表属于哪个schema,表类型,表引擎,创建时间等信息。是show tables from schemaname的结果取之此表。

COLUMNS表:提供了表中的列信息。详细表述了某张表的所有列以及每个列的信息。是show columns from schemaname.tablename的结果取之此表。

STATISTICS表:提供了关于表索引的信息。是show index from schemaname.tablename的结果取之此表。

USER_PRIVILEGES(用户权限)表:给出了关于全程权限的信息。该信息源自mysql.user授权表。是非标准表。

SCHEMA_PRIVILEGES(方案权限)表:给出了关于方案(数据库)权限的信息。该信息来自mysql.db授权表。是非标准表。

TABLE_PRIVILEGES(表权限)表:给出了关于表权限的信息。该信息源自mysql.tables_priv授权表。是非标准表。

COLUMN_PRIVILEGES(列权限)表:给出了关于列权限的信息。该信息源自mysql.columns_priv授权表。是非标准表。

CHARACTER_SETS(字符集)表:提供了mysql实例可用字符集的信息。是SHOW CHARACTER SET结果集取之此表。

COLLATIONS表:提供了关于各字符集的对照信息。

COLLATION_CHARACTER_SET_APPLICABILITY表:指明了可用于校对的字符集。这些列等效于SHOW COLLATION的前两个显示字段。

TABLE_CONSTRAINTS表:描述了存在约束的表。以及表的约束类型。

KEY_COLUMN_USAGE表:描述了具有约束的键列。

ROUTINES表:提供了关于存储子程序(存储程序和函数)的信息。此时,ROUTINES表不包含自定义函数(UDF)。名为“mysql.proc name”的列指明了对应于INFORMATION_SCHEMA.ROUTINES表的mysql.proc表列。

VIEWS表:给出了关于数据库中的视图的信息。需要有show views权限,否则无法查看视图信息。

TRIGGERS表:提供了关于触发程序的信息。必须有super权限才能查看该表

MySQL序列是一组整数:1, 2, 3, ...,由于一张数据表只能有一个字段自增主键, 如果你想实现其他字段也实现自动增加,就可以使用MySQL序列来实现。

MySQL中最简单使用序列的方法就是使用 MySQL AUTO_INCREMENT 来定义列。

CREATE TABLEinsect

(

idINT UNSIGNED NOT NULLAUTO_INCREMENT,PRIMARY KEY(id),

nameVARCHAR(30) NOT NULL,

date DATENOT NULL,

originVARCHAR(30) NOT NULL);

在MySQL的客户端中你可以使用 SQL中的LAST_INSERT_ID( ) 函数来获取最后的插入表中的自增列的值。

如果你删除了数据表中的多条记录,并希望对剩下数据的AUTO_INCREMENT列进行重新排列,那么你可以通过删除自增的列,然后重新添加来实现。 不过该操作要非常小心,如果在删除的同时又有新记录添加,有可能会出现数据混乱。操作如下所示:

ALTER TABLE insect DROPid;ALTER TABLEinsectADD id INT UNSIGNED NOT NULLAUTO_INCREMENT FIRST,ADD PRIMARY KEY (id);

一般情况下序列的开始值为1,但如果你需要指定一个开始值100,那我们可以通过以下语句来实现:

CREATE TABLEinsect

(

idINT UNSIGNED NOT NULLAUTO_INCREMENT,PRIMARY KEY(id),

nameVARCHAR(30) NOT NULL,

date DATENOT NULL,

originVARCHAR(30) NOT NULL)engine=innodb auto_increment=100 charset=utf8;

或者你也可以在表创建成功后,通过以下语句来实现:

ALTER TABLE t AUTO_INCREMENT = 100;SET auto_increment_increment=10; #自增量每次增加的值改为10,SET auto_increment_offset=2; #第一次加载数值时的偏移值的个位值

SHOW VARIABLESLIKE 'AUTO_INC%'; #查看修改后变量的值

SHOWTABLE STATUS FROM NAME_DB; -->显示数据库NAME_DB中所有表的信息

SHOW CREATE TABLE NAME_TBL; -->显示表NAME_TBL创建时的信息

SHOW VARIABLES LIKE 'AUTO_INC%' -->显示MySQL的AUTO_INC开头的用户会话变量(SHOW GLOBAL VARIABLES)

索引

MySQL索引的建立对于MySQL的高效运行是很重要的,索引可以大大提高MySQL的检索速度。

打个比方,如果合理的设计且使用索引的MySQL是一辆兰博基尼的话,那么没有设计和使用索引的MySQL就是一个人力三轮车。

索引分单列索引和组合索引。单列索引,即一个索引只包含单个列,一个表可以有多个单列索引,但这不是组合索引。组合索引,即一个索引包含多个列。

创建索引时,你需要确保该索引是应用在SQL 查询语句的条件(一般作为 WHERE 子句的条件)。

实际上,索引也是一张表,该表保存了主键与索引字段,并指向实体表的记录。

上面都在说使用索引的好处,但过多的使用索引将会造成滥用。因此索引也会有它的缺点:虽然索引大大提高了查询速度,同时却会降低更新表的速度,如对表进行INSERT、UPDATE和DELETE。因为更新表时,MySQL不仅要保存数据,还要保存一下索引文件。

建立索引会占用磁盘空间的索引文件。

索引 一般是以 idx 开头

索引的统计

CREATE INDEX idx_name ONemp(ename);--给emp 表的ename 字段创建一个索引名字是idx_name

ALTER TABLE emp ADD INDEXidx_id(id);--修改表结构(添加索引)

复合索引 多个字段CREATE INDEX idx_name2 ONemp(ename,job);--不一样的

CREATE INDEX idx_name1 ONemp(job,ename);SELECT ename,sal,job FROM emp ORDER BYjob,ename--用到的是idx_name1这个索引 顺序有要求

基于函数的索引

......

如果经常在索引列上 执行DML操作 需要定期重建索引

提高索引的空间利用率

alter index idx_a1 rebuild online;

DROP INDEX idx_name2;

删除索引

添加索引的原则:

不要在小表上 添加

为经常出现在WHERE 子句中的列创建索引

为经常出现在ORDER BY DISTINCT 后面的字段建立索引

如果要建立复合索引 索引的字段顺序要和这些关键字后面的字段一致

为经常作为表的连接条件的列上创建索引

限制表上的索引数目索引不是越多越好

删除很少被使用的不合理的索引

约束的作用

非空约束 =》NOT NULL NN

唯一约束 =》Unique NK

主键约束 =》Primary Key PK

外键约束 =》Foreign Key FK

检查约束 =》Check CK

列级别:

CREATE TABLEtable_name(column_name data_type[[NOT NULL] | [UNIQUE [KEY] | PRIMARY KEY]|CHECK(expr)],…)

表级别:

CREATE TABLEtable_name(

column_name data_type[NOT NULL],

column_name data_type[not null],…,[CONSTRAINT constraint_name] PRIMARY KEY (col_name,...)|[CONSTRAINT constraint_name] unique (col_name,...)|[CONSTRAINT constraint_name] foreign KEY (col_name) REFERENCEStbl_name (index_col_name)|check(expr)

);

1)NOT NULL约束只能在列级别定义,作用在多个列上的约束只能定义在表级别,例如复合主键约束;

2)列级别上不能定义外键约束,并且不能给约束起名字,由MySQL自动命名(NOT NULL除外);

3)表级别上定义的约束可以给约束起名字(CHECK约束除外)

mysql> create tabletemp_uk(-> id int not null unique,-> name varchar(20) unique);

mysql> insert into temp_uk values(1,'a');

mysql> insert into temp_uk values(2,'a');

ERROR1062 (23000): Duplicate entry 'a' for key 'name'mysql> insert into temp_uk values(2,null);

mysql> insert into temp_uk values(3,null);

mysql> select * fromtemp_uk;+----+------+

| id | name |

+----+------+

| 2 | NULL |

| 3 | NULL |

| 1 | a |

+----+------+

2、unique唯一约束

1.唯一约束是指定table的列或列组合不能重复,保证数据的唯一性,约束的列不允许有重复值;

2.唯一约束不允许出现重复的值,但是可以为多个null;

3.同一个表可以有多个唯一约束,多个列组合的约束

可见,唯一性约束的列可以有多个null值,因为null <> null

4.在创建唯一约束时,如果不给唯一约束名称,就默认和列名相同;

5.唯一约束不仅可以在一个表内创建,而且可以同时多表创建组合唯一约束。

mysql> create tabletest(-> id int not null,-> name varchar(20),-> password varchar(16),---使用表级约束语法

-> constraint uk_name_pwd unique(name,password)->); #表示用户名和密码组合不能重复

Query OK,0 rows affected (0.08sec)

查询数据字典,查看唯一键约束的信息

mysql> select * frominformation_schema.table_constraints-> where table_name='test';

mysql> create tabletemp_pk(-> id int primary key);

mysql> insert into temp_pk values(1),(2);

mysql> insert into temp_pk values(1);

ERROR1062 (23000): Duplicate entry '1' for key 'PRIMARY'mysql> update temp_pk set id=1 where id=2;

ERROR1062 (23000): Duplicate entry '1' for key 'PRIMARY'mysql> insert into temp_pk values(null);

ERROR1048 (23000): Column 'id' cannot be null

3、primary key主键约束

primary key = not null + unique

主键:用来唯一的标示表中的每一行(类型一般为整型或者字符串)

具有主键约束的列不允许有null值,并且不允许有重复值;

每个表最多只允许一个主键(可定义联合主键),主键名总是PRIMARY。

!!给主键一个新的名字,但在数据字典中,主键名还是显示primary

mysql> create tabletemp_pk(-> id int,-> name varchar(20),-> constraint pk_id_name primary key(id,name)->);

Query OK,0 rows affected (0.06sec)

mysql> desctemp_pk;+-------+-------------+------+-----+---------+-------+

| Field | Type | Null | Key | Default | Extra |

+-------+-------------+------+-----+---------+-------+

| id | int(11) | NO | PRI | NULL | |

| name | varchar(20) | NO | PRI | NULL | |

+-------+-------------+------+-----+---------+-------+

rows in set (0.00sec)

mysql> insert into temp_pk values(1,'张三');

Query OK,1 row affected (0.00sec)

mysql> insert into temp_pk values(2,'李四');

Query OK,1 row affected (0.01sec)

mysql> insert into temp_pk values(1,'王五');

Query OK,1 row affected (0.00sec)

mysql> insert into temp_pk values(1,'张三');

ERROR1062 (23000): Duplicate entry '1-张三' for key 'PRIMARY'mysql> select * fromtemp_pk;+----+--------+

| id | name |

+----+--------+

| 1 | 张三 |

| 1 | 王五 |

| 2 | 李四 |

+----+--------+

rows in set (0.00 sec)

联合主键(用几个列进行唯一标识一行)

//创建父表

mysql> create tabledept(-> deptid int,-> dname varchar(20),-> constraint dept_deptid_pk primary key(deptid)->);

mysql> insert into dept(deptid,dname) values(10,'市场部');

mysql> insert into dept(deptid,dname) values(20,'销售部');

//创建子表(表级别创建外键约束)

mysql> create tableemp(-> id int,-> name varchar(20),-> deptid int,-> constraint emp_id_pk primary key(id),-> constraint emp_deptid_fk foreign key(deptid)-> referencesdept(deptid)-> );

4、foreign key外键约束

外键约束:

参照完整性约束,保证一个或两个表之间的参照完整性,外键是构建于一个表的两个字段或是两个表的两个字段之间的参照关系。

注意:

1)具有外键约束的列的值不能随便给,必须满足外键所引用的主键的取值;

2)一张表中可以定义多个外键;

3)外键列默认可以给null值。

按照定义,外键必须引用一个主键或者唯一键,引用的主键一般在另外一张表中,也可以是本表的主键(后者称为“自引用”)。

父子表:

外键所在的表叫做子表、从表

外键所引用的主键所在的表叫做父表、主表

注意:父子表是相对而言的,表a可以是表b的子表,但同时也可以是表c的父表

查询数据字典,查看外键约束的信息:

上面创建子表的时候给外键约束命名emp_deptid_fk;

如果不给外键约束命名,那么默认的名字是表名_ibfk_n, n是整数,从1开始;

mysql> insert into emp(id,name,deptid) values(1,'张三',10);

Query OK,1 row affected (0.00sec)

mysql> insert into emp(id,name,deptid) values(2,'李四',10);

Query OK,1 row affected (0.00sec)

mysql> insert into emp(id,name,deptid) values(3,'王五',50);  #insert主表deptid列没有的数据

ERROR1452 (23000): Cannot add or update a child row: a foreign key constraint fails (`test`.`emp`, CONSTRAINT `emp_deptid_fk` FOREIGN KEY (`deptid`) REFERENCES`dept` (`deptid`))

mysql> update emp set deptid=30 where id=1;  #update主表deptid列没有的数据

ERROR1452 (23000): Cannot add or update a child row: a foreign key constraint fails (`test`.`emp`, CONSTRAINT `emp_deptid_fk` FOREIGN KEY (`deptid`) REFERENCES`dept` (`deptid`))

外键的update更新操作规则如下删除规则……

mysql> delete from dept where deptid=10;  #delete父表中的行(子表中有引用的数据行)

ERROR1451 (23000): Cannot delete or update a parent row: a foreign key constraint fails (`test`.`emp`, CONSTRAINT `emp_deptid_fk` FOREIGN KEY (`deptid`) REFERENCES `dept` (`deptid`))

外键的默认删除规则:

当删除父表中的行时,如果子表中有依赖于被删除父行的子行存在,那么就不允许删除,并抛出异常(默认对外键使用on delete restrict 或on delete no action选项)

reference_definition:

REFERENCES tbl_name (index_col_name,...)

[MATCH FULL | MATCH PARTIAL | MATCH SIMPLE]

[ON DELETE reference_option]

[ON UPDATE reference_option]

reference_option:

RESTRICT | CASCADE | SET NULL | NO ACTION

在定义外键约束时,通过使用on delete cascade或者on delete set null选项,可以改变外键的默认删除规则:

①ON DELETE CASCADE:级联删除。当删除父表中的行时,如果子表中有依赖于被删除父行的子行存在,那么连同子行一起删除(很危险!!!)

②ON DELETE SET NULL:当删除父表中的行时,如果子表中有依赖于被删除父行的子行存在,那么不删除,而是将子行的外键列设置为null

5、check约束

MySQL可以使用check约束,但check约束对数据验证没有任何作用。

Oracle中可以使用check约束,是有相应的作用的。

mysql> create tabletest_ck(-> id int check(id>0)->);

mysql> insert into test_ck values(-100);

mysql> select * fromtest_ck;+------+

| id |

+------+

| -100 |

+------+

The CHECK clause is parsed but ignored by all storage engines。

定义数据库列时,可以使用ENUM(enumeration,枚举)和SET(集合)类型:变通的实现CHECK约束

两者的区别是:

使用ENUM,只能选一个值;

使用SET,可以选多个值;

ENUM和SET中的值都必须是字符串类型。

1、enum枚举类型

注意:

在内部存储ENUM值时,MYSQL给ENUM中的每个值一个顺序号码:第一个值的顺序号码是1,第二个值的顺序号码是2,以此类推。当排序或比较ENUM的时候,使用这些顺序号码进行。

mysql> select * from student order bysex;+----+--------+------+

| id | name | sex |

+----+--------+------+

| 5 | 王五 | NULL |

| 1 | 张三 | M |

| 2 | 李四 | F |

+----+--------+------+

2、set类型:因为可以列举多个值,所以在建表列约束中常被使用到。

mysql> create tableteam(-> teamno int not null,-> division set('north','south','east','west')->);

mysql> insert into team values(1,'west');

mysql> insert into team values(2,'west,south');

mysql> insert into team values(4,null);

mysql> insert into team values(3,'east,asia');

ERROR1265 (01000): Data truncated for column 'division' at row 1mysql> select * fromteam;+--------+------------+

| teamno | division |

+--------+------------+

| 1 | west |

| 2 | south,west |

| 4 | NULL |

+--------+------------+

MySQL中,CURRENT_DATE()函数返回当前的系统日期,可以用于INSERT或UPDATE语句中,例如:INSERT INTO my_table (date_column) VALUES (CURRENT_DATE())。而curdate()函数也返回当前的系统日期,以YYYY-MM-DD的格式,可以直接存储到DATE字段中。\[1\]\[2\] 另外,current_date也可以返回当前的系统日期,与CURRENT_DATE()函数功能相同。\[2\]而now()函数返回当前的系统日期和时间,以YYYY-MM-DD HH:MM:SS的格式,可以直接存储到DATETIME字段中。\[3\]sysdate()函数也返回当前的系统日期和时间。\[3\] 综上所述,如果你需要获取当前的系统日期,你可以使用CURRENT_DATE()、curdate()或current_date函数。如果你需要获取当前的系统日期和时间,你可以使用now()或sysdate()函数。 #### 引用[.reference_title] - *1* [MySQL中CURDATE和CURRENT_DATE函数](https://blog.csdn.net/db_dc_dmc/article/details/131123566)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control_2,239^v3^insert_chatgpt"}} ] [.reference_item] - *2* *3* [MySQL 获取系统时间/系统日期/日期时间的函数](https://blog.csdn.net/liaowenxiong/article/details/107665629)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control_2,239^v3^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值