数据库:保存有组织的数据的容器(通常是一个文件或一组文件)。

:某种特定类型数据的结构化清单。

表名:表名的唯一性取决于多个因素,如数据库名或表名等的结合。同一个数据库中不能有相同的表名,但是不同的数据库之间可以有相同的表名。

模式:关于数据库和表的布局及特性的信息。

:表中的一个字段。所有表都是由一个或多个列组成。

数据类型:所容许的数据的类型。每个表列都有相应的数据类型,它限制(或容许)该列中存储的数据。

:表中的一个记录。

主键:一列(或一组列),其值能够唯一区分表中每个行。应该保证每个表中具有一个主键,以便于以后的数据操纵和管理。主键应该满足的要求:

  • 任意两行都具有相同的主键值
  • 每个行都必须具有一个主键值(主键值不允许NULL值)

如果主键是由多列组成,上述条件应用到构成主键的所有列,所有列值的组合必须是唯一的(但单个列的值可以不唯一)。

主键的最好习惯:

  • 不更新主键列中的值
  • 不重用主键列的值
  • 不在主键列中使用可能会更改的值。

检索数据

select

从一个或多个表中检索信息。必须给出两条信息:

  • 想选择什么
  • 从什么地方选择

检索单个列

select prod_name from products:利用select语句从products表中检索一个名为prod_name的列。所需的列名在select关键字之后给出,from关键字指处从其中检索数据的表名。

返回的数据不一定是排序后的数据。

结束SQL语句:多条SQL语句必须以分号分割。

SQL语句不区分大小写。但是一个良好的习惯是对SQL关键字大写,对所有列和表名使用小写。并且再次处理SQL语句时,其中所有空格都被忽略,可以分成一行或多行,分成多行容易阅读和调试。

检索多个列

在select关键字之后给出多个列名,列名之间必须以逗号分隔。(在选择多个列时,一定要在列名之间加上逗号,但最后一个列名后不加。如果在最后一个列名后加了逗号,将出现错误)。

SELECT prod_id,prod_name,prod_price
FROM products;

这条语句使用SELECT语句从表products中选择数据。指定了3个列名,列名之间用逗号分隔。

数据表示:SQL语句一般返回原始的无格式的数据。表示一般在显示该数据的应用程序中规定。

检索所有列

在实际列名的位置使用星号*通配符来达到。

SELECT *
FROM products;

如果给定一个通配符*,则返回表中的所有列。列的顺序一般是列在表定义中出现的顺序。但有时候不是这样的,由于表的模式的变化可能会导致顺序的变化。

检索不同的行

使用DISTINCT关键字,指示MySQL只返回不同的值

SELECT DISTINCT vend_id
FROM products;

上述语句告诉MySQL只返回不同(唯一)的vend_id行。如果使用DISTINCT关键字,它必须直接放在列名的前面。

不能部分使用DISTINCT DISTINCT关键字应用于所有列而不仅是前置它的列。如果给出SELECT DISTINCT vend_id, prod_price FROM products;除非指定的两个列的某一行均与其他一行相同,才被认为是相同的行,如果仅是某个列的一行与其他行相同,不能认为重复。

限制结果

为了返回前几行,使用LIMIT子句。例:

SELECT prod_name
FROM products
LIMIT 5;

LIMIT 5指示MySQL返回不多于5行。

SELECT prod_name
FROM products
LIMIT 5,5;

LIMIT 5,5指示MySQL返回从行5开始的5行。第一个数为位置,第二个数为要检索的行数。

因此,LIMIT带一个数的,总是从第一行开始,给出的数为返回的行数。带两个值得LIMIT可以指定从行号为为第一个值得位置开始。

行0:检索出来的第一行为行0而不是行1.因此,LIMIT 1,1将检索出第二行而不是第一行。

行数不够LIMIT中指定要检索的行数为检索的最大行数。如果没有足够的行,MySQL将只返回它能返回的那么多行。

LIMIT的另一种写法:LIMIT 4 OFFSET 3代表从行3开始取4行。就像LIMIT 3,4一样。

使用完全限定的表名

使用完全限定的列名

SELECT products.prod_name FROM products;

使用完全限定的表名(假定products表位于crashcourse数据库中)

SELECT products.prod_name FROM crashcourse.products;

排序检索数据

按单列排序

关系数据库设计理论认为,如果不明确规定排序顺序,则不应该假定检索出的数据的顺序有意义。

子句:SQL语句由子句构成,有些子句是必需的,而有的是可选的。一个子句通常由一个关键字和所提供的数据组成。

为了明确地排序用SELECT语句检索出的数据,可使用ORDER BY子句。ORDER BY子句取一个或多个列的名字,据此对输出进行排序。

SELECT prod_name FROM products ORDER BY prod_name;

通过非选择列进行排序:通常,ORDER BY子句中使用的列将是为显示所选择的列。但是,实际上并不一定要这样,用非检索的列排序数据是完全合法的。

按多个列排序

按多个列排序,只要指定列名,列名之间用逗号分开即可,就像选择多个列时一样。

SELECT prod_id,prod_price,prod_name 
FROM products 
ORDER BY prod_price,prod_name;

在排序过程中,仅在多个行具有相同的prod_price值时才对产品按prod_name进行排序。如果prod_price列中的所有值都是唯一的,则不会按prod_name排序。

指定排序方向

默认的为升序排序,另外可以指定DESC关键字,进行降序排序

SELECT prod_id,prod_price,prod_name
FROM products
ORDER BY prod_price DESC;

如果对多个列进行排序,每个列排序的方式不同,可采用下列方式:

SELECT prod_id,prod_name,prod_price
FROM products
ORDER BY prod_price DESC, prod_name;

先以prod_price降序排序,如果遇到重复的prod_price,再以prod_name升序排序。

DESC关键字只应用到直接位于其前面的列名。如果想对多个列进行降序排序,必须对每个列指定DESC关键字。

与DESC相反的关键字是ASC。但是一般没有什么用处。

利用ORDER BY和LIMIT的组合,能够找出当前排序下的对应行。在给出ORDER BY 子句时,应该保证它位于FROM子句之后。如果使用LIMIT,它必须位于ORDER BY之后。使用子句的次序不对将产生错误信息。

过滤数据

使用WHERE子句

检索数据时,通常只会根据特定操作或报告的需要提取表数据的子集。只检索所需数据需要指定搜索条件,搜索条件也被称为过滤条件。

在SELECT语句中,数据根据WHERE子句中指定的搜索条件进行过滤。WHERE子句在表名(FROM子句)之后给出,如下所示:

SELECT prod_name,prod_price
FROM products
WHERE prod_price = 2.50;

这条语句从products表中检索出两个列,但不返回所有行,只返回prod_price=2.50的行。

WHERE子句的位置:在同时使用ORDER BY和WHERE子句时,应该让ORDER BY位于WHERE之后,否则将会发生错误。

WHERE子句操作符

=    等于
<>    不等于
!=    不等于
<    小于
<=    小于等于
>    大于
>=    大于等于
BETWEEN    在指定的两个值之间

检查单个值

SELECT prod_anme,prod_price
FROM products
WHERE prod_price <= 10;

检索价格小于等于10的所有产品

不匹配检查

SELECT vend_id,prod_name
FROM products
WHERE vend_id <> 1000 
ORDER BY vend_id 
LIMIT 1 OFFSET 0;

查找vend_id不等于1000的按照vend_id递增排序的第一行。

使用引号:在WHERE子句中可能使用引号,单引号用来限定字符串。如果将值与串类型的列进行比较,则需要限定引号。用来与数值列进行比较的值不用引号。

范围值检查

使用BETWEEN操作符。

SELECT prod_name,prod_price
FROM products
WHERE prod_price BETWEEN 5 AND 10;

检索价格在5到10之间。BETWEEN匹配范围中的所有制,包括指定的开始值和结束值。

空值检查

在创建表时,可以指定其中的列是否可以不包含值。在一个列不包含值时,称其为包含空值NULL。

NULL 无值,与字段包含0、空字符串或仅仅包含空格不同。

特殊的WHERE子句,用来检查具有NULL的列。

SELECT prod_name,prod_price
FROM products
WHERE prod_price IS NULL;

返回没有价格的所有产品。

​ NULL与不匹配:在通过过滤选择出不具有特定值的行时,可能希望返回具有NULL值得行。但是,不行。因为未知具有特殊的含义,数据库不知道他们是否匹配,所以在匹配过滤或不匹配过滤时不返回它们。

因此,在过滤数据时,一定要验证返回数据中确实给出了被过滤列具有NULL的行。


数据过滤

组合WHERE子句建立功能更强的更高级的搜索条件。MySQL允许给出多个WHERE子句,这些子句可以以两种方式使用:以AND子句的方式或OR子句的方式使用。

AND子句:

SELECT prod_id,prod_price,prod_name
FROM products
WHERE vend_id = 1003 AND prod_price <= 1000;

返回vend_id=1003且价格小于等于1000的所有产品的id、价格和名称。

OR子句:

SELECT prod_name,prod_id
FROM products
WHERE vend_id = 1002 OR vend_id = 1003;

返回vend_id=1002或者vend_id=1003的商品名称和id。

计算次序

SELECT prod_name,prod_price
FROM products
WHERE vend_id = 1002 OR vend_id = 1003 AND prod_price >=10;

SQL在处理OR操作符前,优先处理AND操作符。当SQL看到上述WHERE子句时,理解为由vend_id=1003制造的任何价格在10(含)以上的产品,或者由vend_id=1002制造的任何产品,而不管其价格如何。

解决方法是使用圆括号明确地分组相应的操作符。如下:

SELECT prod_name,prod_price
FROM products
WHERE (vend_id = 1002 OR vend_id = 1003) AND prod_price >= 10;

圆括号具有较AND和OR操作符高的计算次序。

在WHERE子句中使用圆括号:任何时候使用具有AND和OR操作符的WHERE子句,都应该使用圆括号明确地分组操作符。不要过分依赖默认计算次序,即使它确实是你想要的东西也是如此。

IN操作符

IN操作符用来指定条件范围,范围中的每个条件都可以进行匹配。IN取合法值的由逗号分隔的清单,全都在圆括号中。

SELECT prod_name, prod_price
FROM products
WHERE vend_id IN (1002,1003)
ORDER BY prod_name;

此SELECT语句检索vend_id为1002和1003的所有数据。IN操作符后的合法值清单,必须在圆括号中。与OR等价。

IN操作符的优点:

  • 在使用长的合法选项清单时,IN操作符的语法更清楚且更直观
  • 在使用IN时,计算的次序更容易管理(因为使用的操作符更少)
  • IN操作符一般比OR操作符清单执行更快
  • IN的最大优点是可以包含其他SELECT语句,使得能够更动态的建立WHERE子句。

NOT操作符

WHERE子句中的NOT操作符有且只有一个功能,那就是否定它之后所跟的任何条件。

SELECT prod_name, prod_price
FROM products
WHERE vend_id NOT IN (1002,1003)
ORDER BY prod_name;

这里的NOT否定跟在它之后的条件,因此,现在匹配vend_id不是1002和1003的所有其他数据。

MySQL中的NOT 支持使用NOT对IN、BETWEEN和EXISTS子句取反。

NOT仅对其后的一个条件有影响,再之后的条件无影响

SELECT DISTINCT prod_name, prod_price
FROM products
WHERE prod_price NOT IN(5,10) AND prod_price = 9;

上面的语句是找出价格不等于5和10并且价格等于9的所有数据。

用通配符进行过滤

LIKE操作符

通配符:用来匹配值得一部分的特殊字符

搜索模式:由字面值、通配符或两者组合构成的搜索条件。

为在搜索子句中使用通配符,必须使用LIKE操作符。LIKE指示MySQL后跟的搜索模式利用通配符匹配而不是直接相等匹配进行比较。

谓词:操作符何时不是操作符?答案是它作为谓词时。从技术上说,LIKE是谓词而不是操作符。虽然最终的结果是相同的,但应该对此术语有所了解。

百分号(%)通配符

%表示任何字符出现任意次数。

SELECT prod_id, prod_name
FROM products
WHERE prod_name LIKE 'jet%';

使用搜素模式‘jet%’。在执行这条子句时,将检索任何以‘jet’起头的词。‘%’告诉MySQL接受jet之后的任意字符,不管有多少字符。

如果需要区分大小写,需要在MySQL中进行配置,因此搜索可以是区分大小写的。

通配符可以在搜索模式中任意位置使用,并且可以使用多个通配符。

SELECT prod_id, prod_name
FROM products
WHERE prod_name LIKE '%anvil%';

搜索模式‘%anvil%’表示匹配任何位置包含文本anvil的值,而不论它之前或之后出现什么字符。

通配符也可以出现在搜索模式的中间,但不太有用。

SELECT prod_name, prod_price
FROM products
WHERE prod_name LIKE 's%e';

%可以匹配0个字符、1个字符或多个。

注意尾空格:尾空格可能会干扰通配符匹配。一个简单的办法是在搜索模式最后附加一个%,一个更好的办法是使用函数去掉首尾空格。

注意NULL:虽然似乎%通配符可以匹配任何东西,但有一个例外,即NULL。即使是WHERE prod_name LIKE '%`也不能匹配用值NULL作为产品名的行。

下划线(_)通配符

下划线的用途与%一样,但下划线只匹配单个字符而不是多个字符。

SELECT prod_id,prod_name
FROM products
WHERE prod_name LIKE '_ ton anvil'

此WHERE子句中的搜索模式给出了后面跟有文本的两个通配符。结果只显示匹配搜索模式的行。即必须是如1 ton anvil,但是34 ton anvil无法匹配到

与%能匹配0个字符不一样,_总是匹配一个字符,不能多也不能少。

使用通配符的技巧

  • 不要过度使用通配符。如果其他操作符能达到相同的目的,应该使用其他操作符。
  • 在确实需要使用通配符时,除非绝对有必要,否则不要把他们用在搜索模式的开始处。把通配符置于搜索模式的开始处,搜索起来是最慢的。
  • 仔细注意通配符的位置。如果放错地方,可能不会返回想要的数据。

用正则表达式进行搜索

正则表达式的作用是匹配文本,将一个模式(正则表达式)与一个文本串进行比较。MySQL用WHERE子句对正则表达式提供了初步的支持,允许指定正则表达式,过滤SELECT检索出的数据。

基本字符匹配

SELECT prod_name, prod_price
FROM products
WHERE prod_name REGEXP '1000'
ORDER BY prod_name;

除关键字LIKEREGEXP替代外,这条语句非常像使用LIKE的语句。它告诉MySQL:REGEXP后所跟的东西作为正则表达式处理。

SELECT DISTINCT prod_name,prod_price
FROM products
WHERE prod_name REGEXP '.000'
ORDER BY prod_name;

这里使用正则表达式.000。其中.是正则表达式语言中一个特殊的字符。它表示匹配任意一个字符,因此,1000和2000都匹配且返回。

LIKE和REGEXP:在LIKE和REGEXP之间有一个重要的差别。如下:

SELECT prod_name, prod_price
FROM products
WHERE prod_name LIKE '1000'
ORDER BY prod_name;
SELECT prod_name, prod_price
FROM products
WHERE prod_name REGEXP '1000'
ORDER BY prod_name;

LIKE匹配整个列。如果被匹配的文本在列值中出现,LIKE将不会找到它,相应的行也不会被返回(除非使用通配符)。而REGEXP在列值内进行匹配,如果被匹配的文本在列值中出现,REGEXP将会找到它,相应的行将被返回。这是一个重要的差别。

匹配不区分大小写:正则表达式匹配,如果要区分大小写,使用BINARY关键字,如WHERE prod_name REGEXP BINARY ‘JetPack .000’

进行OR匹配

为搜索两个串之一(或者为这个串,或者为另一个串),使用|,如下所示:

SELECT prod_name
FROM products
WHERE prod_name REGEXP '1000|2000'
ORDER BY prod

语句中使用了正则表达式1000|2000|为正则表达式的OR操作符。它表示匹配其中之一,因此1000和2000都匹配并返回。

匹配几个字符之一

匹配特定的字符,可以通过指定一组用[]括起来的字符来完成。如下:

SELECT prod_name
FROM products
WHERE prod_name REGEXP '[123] Ton'
ORDER BY prod_name;

这里,使用了正则表达式[123] Ton[123]定义了一组字符,它的意思是匹配1或2或3,因此1 ton 和2 ton和3 ton都匹配且返回。

[]是另一种形式的OR语句。事实上,正则表达式[123]Ton[1|2|3]ton的缩写。但是如下写法存在一定问题:

SELECT prod_name
FROM products
WHERE prod_name REGEXP '1|2|3 Ton'
ORDER BY prod_name;

这时的意思是‘1’或‘2’或‘3 ton’。除非把字符|括在一个集合中,否则将应用于整个串。

字符集合也可以被否定,即,它们将匹配除指定字符外的任何东西。为否定一个字符集,在集合开始处放置一个^即可。因此,尽管[123]匹配字符123,但[^123]却匹配除这些字符外的任何东西。

匹配范围

在集合中使用-来定义一个范围。[0-9]等同于[0123456789]。范围不限于完整的集合,[1-3][6-9]也是合法的范围。此外,范围不一定是数值,[a-z]匹配任意字母字符。

SELECT prod_name
FROM products
WHERE prod_name REGEXP '[1-5] Ton'
ORDER BY prod_name;

匹配1到5。

匹配特殊字符

如果要找出.字符的值,如下:

SELECT vend_name
FROM vendors
WHERE vend_name REGEXP '.'
ORDER BY vend_name;

这并不是期望的输出,.匹配任意字符,因此每个行都检索出来,为了匹配特殊字符,必须用‘\\’为前导。\\-表示查找-\\.表示查找.

SELECT vend_name
FROM vendors
WHERE vend_name REGEXP '\\.'
ORDER BY vend_name;

查找.。这就是所谓的转义,正则表达式内具有特殊意义的所有字符都必须以这种方式转义。这包括. | []以及迄今为止使用过的其他特殊字符。

\\也用来引用元字符(具有特殊含义的字符),如下表所示:

元字符 说明
\\f 换页
\\n 换行
\\r 回车
\\t 制表
\\v 纵向制表

匹配\ 为了匹配反斜杠\本身,需要使用\\\

\或\?:多数正则表达式实现使用单个反斜杠转义特殊字符,以便能使用这些字符本身。但MySQL要求两个反斜杠(MySQL自己解释一个,正则表达式库解释一个)。

匹配字符类

存在找出经常使用的数字、所有字母字符或所有数字字母字符等的匹配。为更方便的工作,可以使用预定义的字符集,称为字符类。如下表所示:

说明
[:alnum:] 任意字母和数字(同[a-zA-Z0-9])
[:alpha:] 任意字符(同[a-zA-Z]
[:blank:] 空格和制表(同[\\t]
[:cntrl:] ASCII控制字符(ASCII 0到31和127)
[:digit:] 任意数字(同[0-9]
[:graph:] [:print:]相同,但不包括空格
[:lower:] 任意小写字母(同[a-z]
[:print:] 任意可打印字符
[:punct:] 既不在[:alnum:]又不在[:cntrl:]中的任意字符
[:space:] 包括空格在内的任意空白字符(同[\\f\\n\\r\\t\\v])
[:upper:] 任意大写字母(同[A-Z])
[:xdigit:] 任意十六进制数字(同[a-fA-F0-9]

匹配多个实例

有时需要对匹配的数目进行更强的控制。可用下列的正则表达式重复元字符来完成:

元字符 说明
* 0个或多个匹配
+ 1个或多个匹配(等于{1,}
? 0个或1个匹配(等于{0,1}
{n} 指定数目的匹配
{n,} 不少于指定数目的匹配
{n,m} 匹配数目的范围(m不超过255)

例子:

SELECT prod_name
FROM products
WHERE prod_name REGEXP '\\([0-9] sticks?\\)'
ORDER BY prod_name;

正则表达式\\([0-9] sticks?\\) 解释:\\(匹配([0-9]匹配任意数字,sticks?匹配stickstickss后的?s可选,因为?匹配它前面的任何字符的0次或1次出现,\\)匹配)。没有?,匹配stick和sticks会非常困难。

SELECT prod_name
FROM products
WHERE prod_name REGEXP '[[:digit:]]{4}'
ORDER BY prod_name;

[:digit:]匹配任意数字,因而它为数字的一个集合。{4}确切地要求它前面的额字符(任意数字)出现4次,所以[[:digit:]]{4}匹配连在一起的任意4位数字。

需要注意的是,在使用正则表达式时,编写某个特殊的表达式几乎总是不止有一种方法。上面的例子也可以如下编写:

SELECT prod_name
FROM products
WHERE prod_name REGEXP '[0-9][0-9][0-9][0-9]'
ORDER BY prod_name;

定位符

为了匹配特定位置的文本,需要使用下表列出的定位符:

元字符 说明
^ 文本的开始
$ 文本的结尾
[[:<:]] 词的开始
[[:>:]] 词的结尾

如果想找出一个数(包括以小数点开始的数)开始的所有产品,简单搜索[0-9\\.](或[[:digit:]\\.])不行,因为它将在文本内任意位置查找匹配。解决方法是使用^定位符,如下所示:

SELECT prod_name
FROM products
WHERE prod_name REGEXP '^[0-9\\.]'
ORDER BY prod_name;

^匹配串的开始。因此^[0-9\\.]只在.或任意数字为串中第一个字符时才匹配它们。没有^,则还要多检索出那些中间有数字的行。

^的双重用途^有两种用法。在集合中(用[]定义),用它来否定该集合,否则,用来指串的开始处。

是REGEXP起类似LIKE的作用:LIKE和REGEXP的不同在于,LIKE匹配整个串而REGEXP匹配子串。利用^开始每个表达式,用$结束每个表达式,可以使REGEXP的作用与LIKE一样。


创建计算字段

计算字段并不实际存在于数据库表中。计算字段是运行在SELECT语句内创建的。

字段:基本上与列的意思相同,经常互换使用,不过数据库一般称为列,而术语字段通常用在计算字段的连接上。

客户机与服务器的格式:可在SQL语句内完成的许多转换和格式化的工作都可以直接在客户机应用程序内完成。但一般来说,在数据库服务器上完成这些操作比在客户机中完成要快得多。

拼接字段

拼接:将值联结到一起构成单个值

在SELECT语句中,可使用Concat()函数来拼接两个列。

MySQL的不同之处:多数DBMS使用+||来实现拼接,MySQL则使用Concat()函数来实现。当把SQL语句转换成MySQL语句时一定要把这个区别铭记在心。

SELECT Concat(vend_name,'(',vend_country,')')
FROM vendors
ORDER BY vend_name;

Concat()拼接串:即把多个串连接起来形成一个较长的串。Concat()需要一个或多个指定的串,各个串之间用逗号分隔。

上面的SELECT语句连接以下4个元素:

  • 存储在vend_name列中的名字
  • 包含一个空格和一个左圆括号的串
  • 存储在vend_country列中的国家
  • 包含一个右圆括号的串。

删除空格

SELECT Concat(RTrim(vend_name),'(',RTrim(vend_country),')')
FROM vendors
ORDER BY vend_name;

RTrim()函数:去掉右边的所有空格。

Trim函数:MySQL除了支持RTrim()还支持LTrim()以及Trim()(去掉左右两边的空格)。

使用别名

通过计算后的列是没有名字的,只是一个值。但是未命名的列不能在客户机中使用,因此SQL支持列别名。别名(alias)是一个字段或值得替换名。别名用AS关键字赋予。

SELECT Concat(RTrim(vend_name),'(',RTrim(vend_country),')') AS
vend_title
FROM vendors
ORDER BY vend_name;

SELECT语句本身与以前使用的相同,只不过这里的语句中计算字段之后跟了文本AS vend_title。它指示SQL创建一个包含指定计算的名为vend_title的计算字段。任何客户机可以按名引用这个列,就像它是一个实际的表列一样。

别名的其他用途:别名还有其他的用途。常见的用途包括在实际的表列名包含不符合规定的字符时重新命名它,在原来的名字含混或容易误解时扩充它,等等。

导出列:别名有时也被称为导出列,不管称为什么,他们所表示的都是相同的东西。

执行算术运算

SELECT prod_id,quantity,item_price,
        quantity*item_price AS expand_price
FROM orderitems
WHERE order_num=20005;

输出中显示的expand_price列为一个计算字段,此计算为quantity*item_price。客户机现在可以使用这个新计算列,就像使用其他列一样。

MySQL支持下表中所列出的基本算术操作符。此外,圆括号可用来区分优先顺序。

操作符 说明
+
-
*
/

如何测试计算SELECT提供了测试和试验函数与计算的一个很好的办法。虽然SELECT通常用来从表中检索数据,但可以省略FROM子句以便简单地访问和处理表达式。例如,SELECT 3*2;将返回6, SELECT Trim('abc');将返回abc,而SELECT Now();利用Now()函数返回当前日期和时间。通过这些例子,可以明白如何根据需要使用SELECT进行试验。


使用数据处理函数

函数

文本处理函数

SELECT vend_name, Upper(vend_name) AS vend_name_upcase
FROM vendors
ORDER BY vend_name;

Upper()将文本转换为大写。

常用的文本处理函数:

函数 说明
Left() 返回串左边的字符
Length() 返回串的长度
Locate() 找出串中的一个子串
Lower() 将串转换为小写
LTrim() 去掉串左边的空格
Right() 返回串右边的字符
RTrim() 去掉串右边的空格
Soundex() 返回串的SOUNDEX值
SubString() 返回子串的字符
Upper() 将串转换为大写

SOUNDEX是一个将任何文本串转换为描述其语音表示的字母数字模式的算法。soundex考虑了类似的发音字符和音节,使得能够对串进行发音比较而不是字母比较。

日期和时间处理函数

日期和时间采用相应的数据类型和特殊的格式存储,以便能快速和有效地排序或过滤,并且节省物理存储空间。

常用的日期和时间处理函数

函数 说明
AddDate() 增加一个日期(天、周等)
AddTime() 增加一个时间(时、分等)
CurDate() 返回当前日期
CurTime() 返回当前时间
Date() 返回日期时间的日期部分
DateDiff() 计算两个日期之差
Date_Add() 高度灵活的日期运算函
Date_Format() 返回一个格式化的日期或时间串
Day() 返回一个日期的天数部
DayOfWeek() 对于一个日期,返回对应的星期
Hour() 返回一个时间的小时部
Minute() 返回一个时间的分钟部
Month() 返回一个日期的月份部
Now() 返回当前日期和时
Second() 返回一个时间的秒部
Time() 返回一个日期时间的时间部分
Year() 返回一个日期的年份部分

首先需要注意的是MySQL使用的日期格式。无论你什么时候指定一 个日期,不管是插入或更新表值还是用WHERE子句进行过滤,日期必须为格式yyyy-mm-dd。因此, 2005年9月1日,给出为2005-09-01。虽然其他的日期格式可能也行,但这是首选的日期格式,因为它排除了多义性(如,
04/05/06是2006年5月4日或2006年4月5日或2004年5月6日或……)。

SELECT cust_id,order_num
FROM orders
WHERE Date(order_data) = '2005-09-01';

查找下单日期为'2005-09-01'。

SELECT cust_id, order_num
FROM orders
WHERE Data(order_date) BETWEEN '2005-09-01' AND '2005-09-30';

其中, BETWEEN操作符用来把2005-09-01和2005-09-30定义为一个要匹配的日期范围。

SELECT cust_id, order_num
FROM orders
WHERE Year(order_date) = 2005 AND Month(order_date) = 9;

Year()是一个从日期(或日期时间)中返回年份的函数。类似,Month()从日期中返回月份。因此, WHERE Year(order_date) = 2005 AND Month(order_date) = 9检索出order_date为2005年9月的所有行 。

数值处理函数

数值处理函数仅处理数值数据。这些函数一般主要用于代数、三角或几何运算,因此没有串或日期—时间处理函数的使用那么频繁。

函数 说明
Abs() 返回一个数的绝对值
Cos() 返回一个角度的余
Exp() 返回一个数的指数值
Mod() 返回除操作的余数
Pi() 返回圆周率
Rand() 返回一个随机数
Sin() 返回一个角度的正弦
Sqrt() 返回一个数的平方根
Tan() 返回一个角度的正切

汇总数据

聚集数据

聚集函数:运行在行组上,计算和返回单个值得函数。

函数 说明
AVG() 返回某列的平均值
COUNT() 返回某列的行数
MAX() 返回某列的最大值
MIN() 返回某列的最小值
SUM() 返回某列值之和

AVG()函数

通过对表中行数计数并计算特定列值之和,求得该列的平均值。AVG()可用来返回所有列的平均值,也可用来返回特定列或行的平均值。

SELECT AVG(prod_price) AS avg_price
FROM products;

此SELECT语句返回值avg_price,它包含products表中所有产品的平均价格。

AVG()也可以用来确定特定列或行的平均值。如下:

SELECT AVG(prod_price) AS avg_price
FROM products
WHERE vend_id = 1003;

这条SELECT语句与之前一条的不同之处在于它包含了WHERE子句。此WHERE子句仅过滤出vend_id为1003的产品,因此avg_price返回的值只是该供应商的产品的平均值。

只用于单个列:AVG()函数只能用来确定特定数值列的平均值,而且列名必须作为函数参数给出。为了获得多个列的平均值,必须使用多个AVG()函数。

NULL值:AVG()函数忽略列值为NULL的行。

COUNT()函数

两种使用方式:

  • 使用COUNT(*)对表中行的数目进行计数,不管表列中包含的是空值(null)还是非空值
  • 使用COUNT(column)对特定列中具有值得行进行计数,忽略NULL值
SELECT COUNT(*) AS num_cust
FROM customers;

在此例中,利用COUNT(*)对所有行进行计数,不管行中各列有什么值。计数值在num_cust中返回。

SELECT COUNT(cust_email) AS num_cust
FROM customers;

这条SELECT语句使用COUNT(cust_email)对cust_email列中有值得行进行计数。

NULL值:如果指定列名,则指定列的值为空的行被COUNT()函数忽略,但如果COUNT()函数中用的是星号(*),则不忽略。

MAX()函数

MAX()函数返回的是指定列中的最大值。MAX()要求指定列名。

SELECT MAX(prod_price) AS max_price
FROM products;

这里,MAX()返回的是products表中最贵的物品的价格。

对非值数据使用MAX():虽然MAX()一般用来找出最大数值或日期值,但MySQL允许将它用来返回任意列中的最大值,包括返回文本列中的最大值。在用于文本数据时,如果数据按相应的列排序,则MAX()返回最后一行。

NULL值:MAX函数忽略列值为NULL的行。

MIN()函数

与MAX()函数相似。

SUM函数

用来返回指定列值得和。

SELECT SUM(quantity) AS items_ordered
FROM orderitems
WHERE order_num = 20005;

函数SUM(quantity)返回订单中所有物品数量之和,WHERE子句保证只统计某个物品订单中的物品。

SUM()也可以用来合计计算值。

SELECT SUM(item_price*quantity) AS total_price
FROM orderitems
WHERE order_num = 20005;

函数SUM(item_price*quantity)返回订单中所有物品价钱之和,WHERE子句同样保证只统计某个物品订单中的物品。

在多个列上进行计算:利用标准的算术操作符,所有聚集函数都可用来执行多个列上的计算。

聚集不同值

以上5个聚集函数都可以如下使用:

  • 对所有的列执行计算,指定ALL参数或不给参数(ALL默认)
  • 只包含不同的值,指定DISTINCT参数。
SELECT AVG(DISTINCT prod_price) AS avg_price
FROM products
WHERE vend_id = 1003;

注:对于COUNT()函数,DISTINCT不能用于COUNT(*),即COUNT(DISTINCT),这样会产生错误。类似地,DISTINCT必须使用列名吗,不能用于计算或表达式。

DISTINCT用于MIN或MAX无意义。

组合聚集函数

SELECT COUNT(*) AS num_items,
       MIN(prod_price) AS price_min,
       MAX(prod_price) AS price_max,
       AVG(prod_price) AS price_avg
FROM products;

这里用单挑SELECT语句执行了4个聚集计算,返回4个值。


分组数据

数据分组

SELECT COUNT(*) AS num_prods
FROM products
WHERE vend_id = 1003;

上面是返回每供应商1003提供的产品数量。

创建分组

分组是在SELECT语句的GROUP BY子句中建立的。如下:

SELECT vend_id, COUNT(*) AS num_prods
FROM products
GROUP BY vend_id;

上面的SELECT语句指定了两个列,vend_id包含产品供应商的ID,num_prods为计算字段(用COUNT(*)函数建立)。GROUP BY子句指示MySQL按vend_id排序并分组数据。这导致对每个vend_id而不是整个表计算num_prods一次。

因为使用了GROUP BY,就不必指定要计算和估值的每个组。系统会自动完成。GROUP BY 子句指示MySQL分组数据,然后对每个组而不是整个结果进行聚集。

使用GROUP BY子句前,需要知道如下重要的规定:

  • GROUP BY 子句可以包含任意数目的列。这使得能对分组进行嵌套,为数据分组提供更细致的控制。
  • 如果在GROUP BY子句中嵌套了分组,数据将在最后规定的分组上进行汇总。换句话说,在建立分组时,指定的所有列都一起计算(所以不能从个别列取回数据)。
  • GROUP BY子句中列出的每个列都必须是检索列或有效的表达式(但不能是聚集函数)。如果在SELECT中使用了表达式,则必须在GROUP BY子句中指定相同的表达式。不能使用别名。
  • 除聚集计算语句外,SELECT语句中的每个列都必须在GROUP BY子句中给出。
  • 如果分组列中具有NULL值,则NULL将作为一个分组返回。如果列中有多行的NULL值,他们将分为一组。
  • GROUP BY 子句必须出现在WHERE子句之后,ORDER BY子句之前

使用ROLLUP:使用WITH ROLLUP关键字,可以得到每个分组以及每个分组汇总级别(针对每个分组)的值,如下所示:

SELECT vend_id, COUNT(*) AS num_prods
FROM products
GROUP BY vend_id WITH ROLLUP;

过滤分组

除了使用GROUP BY分组数据外,MySQL还允许过滤分组,规定包括哪些分组,排除哪些分组。WHERE过滤指定的是行而不是分组。达成这个目标可以采用HAVING子句。与WHERE子句类似。唯一的差别是WHERE子句过滤行,HAVING子句过滤分组。

HAVING支持所有WHERE操作符:所有的WHERE子句的条件(包括通配符条件和带多个操作符的子句)。君可用于HAVING子句。

SELECT cust_id, COUNT(*) AS orders
FROM orders
GROUP BY cust_id
HAVING COUNT(*) >= 2;

这条SELECT语句的前3行类似于上面的语句。最后一行增加了HAVING子句,它过滤COUNT(*)>=2的那些分组。

这里WHERE子句不起作用,因为过滤是基于分组聚集值而不是特定行值。

HAVING和WHERE的差别:WHERE在数据分组前进行过滤,HAVING在数据分组后进行过滤。这是一个重要的区别,WHERE排除的行不包括在分组中。这可能会改变计算值,从而影响HAVING子句中基于这些值过滤掉的分组。

SELECT vend_id,COUNT(*) AS num_prods
FROM products
WHERE prod_price >= 10
GROUP BY vend_id
HAVING COUNT(*) >= 2;

第一行是使用了聚集函数的基本SELECT,它与前面的例子很相像。WHERE子句过滤所有prod_price至少为10的行。然后按vend_id分组数据。HAVING子句过滤计数为2或2以上的分组。

分组和排序

虽然GROUP BY和ORDER BY经常完成相同的工作,但他们是非常不同的。如下表所示:

ORDER BY GROUP BY
排序产生的输出 分组行。但输出可能不是分组的顺序
任意列都可以使用(甚至非选择的列也可以使用) 只可能使用选择列或表达式列,而且必须使用每个选择列表达式
不一定需要 如果与聚集函数一起使用列(或表达式),则必须使用

第一项差别极为重要。

不要忘记ORDER BY:一般在使用GROUP BY 子句时,应该也给出ORDER BY子句。这是保证数据正确排序的唯一方法。千万不要仅依赖GROUP BY排序数据。

SELECT order_num, SUM(quantity*item_price) AS ordertotal
FROM orderitems
GROUP BY order_num
HAVING SUM(quantity*item_price) >=50;

上面的输出可能是乱序的,如果要排序输出,应该如下:

SELECT order_num, SUM(quantity*item_price) AS ordertotal
FROM orderitems
GROUP BY order_num
HAVING SUM(quantity*item_price) >= 50
ORDER BY ordertotal;

这个例子,GROUP BY子句用来返回订单号(order_num列)分组数据,以便SUM(*)函数能够返回总计订单价格。HAVING子句过滤数据,使得只返回总计订单价格大于等于50的订单。最后,ORDER BY子句排序输出。

SELECT子句顺序

具体的子句有:

子句 说明 是否必须使用
SELECT 要返回的列或表达式
FROM 从中检索数据的表 仅在从表选择数据时使用
WHERE 行级过滤
GROUP BY 分组说明 仅在按组计算聚集时使用
HAVING 组级过滤
ORDER BY 输出排序顺序
LIMIT 要检索的行数

使用子查询

查询:任何SQL语句都是查询。但此术语一般指SELECT语句。

子查询:嵌套在其他查询中的查询。

利用子查询进行过滤

例子:现在需要列出订购物品TNT2的所有客户,步骤如下:

  • 检索所有包含物品TNT2的所有订单的编号
  • 检索具有前一步骤列出的订单编号的所有客户的ID
  • 检索前一步骤返回的所有客户ID的客户信息

第一条SELECT语句查找prod_id为TNT2的所有订单物品

SELECT order_num
FROM orderitems
WHERE prod_id = 'TNT2';

下一步,查询上条语句返回的订单号,假如orde_num为20005和20007。

SELECT cust_id
FROM orders
WHERE order_num IN (20005,20007);

得到对应的客户ID。

现在使用子查询,如下:

SELECT cust_id
FROM orders
WHERE order_num IN (SELECT order_num 
                    FROM orderitems
                   WHERE prod_id = 'TNT2');

在SELECT语句中,子查询总是从内向外处理。在处理上面的SELECT语句时,MySQL实际上执行了两个操作。

首先,它执行了下面的查询:

SELECT order_num FROM orderitems WHERE prod_id = 'TNT2';

此查询返回两个订单号20005和20007.然后这两个值以IN操作符要求的逗号分隔的格式传递给外部查询的WHERE子句。外部查询变成:

SELECT cust_id FROM orders WHERE  order_num IN (20005,20007);

这样的结果与硬编码的WHERE子句所返回的值相同。

格式化SQL:包含子查询的SELECT语句难以阅读和调试,特别是较为复杂时更是如此。如上所示把子查询分解为多行并且适当地进行缩进,能极大地简化子查询的使用 。

列必须匹配:在WHERE子句中使用子查询(如这里所示),应该保证SELECT语句具有与WHERE子句中相同数目的列。通常,子查询将返回单个列并且与单个列匹配,但如果需要也可以使用多个列 。

虽然子查询一般与IN操作符结合使用,但也可以用于测试等于(=)、不等于(<>)等。

子查询和性能:这里给出的代码有效并获得所需的结果。但是,使用子查询并不总是执行这种类型的数据检索的最有效的方法。

作为计算字段使用子查询

使用子查询的另一方法是创建计算字段。假如需要显示customers表中每个客户的订单总数。订单与相应的客户ID存储在orders表中。

为了执行这个操作,遵循下面的步骤

  • 从customers表中检索客户列表
  • 对于检索出的每个客户,统计其在orders表中的订单数目

如下,对客户10001的订单进行计数

SELECT COUNT(*) AS orders
FROM orders
WHERE cust_id = 10001;

对每个客户执行COUNT()计算,应该将COUNT()作为一个子查询。

SELECT cust_name,
       cust_state,
       (SELECT COUNT(*)
         FROM orders
         WHERE orders.cust_id = customers.cust_id) AS orders
FROM customers
ORDER BY cust_name;

这条SELECT语句对customers表中每个客户返回3列:cust_name、cust_state和orders。orders是一个计算字段,它是由圆括号中的子查询建立的。该子查询对检索出的每个客户执行一次。

子查询中的WHERE子句与前面使用的WHERE子句稍有不同,因为它使用了完全限定列名。下面的语句告诉SQL比较orders表中的cust_id与当前正从customers表中检索的cust_id:

WHERE orders.cust_id = customers.cust_id;

相关子查询:涉及外部查询的子查询

任何时候只要列名可能有多义性,就必须使用这种语法(表名和列名由一个句点分隔)。如果不使用完全限定的列名时,如下:

SELECT cust_name,
       cust_state,
       (
           SELECT COUNT(*)
           FROM orders
           WHERE cust_id = cust_id
       ) AS orders
FROM customers
ORDER BY cust_name;

这种语句是错误的。有两个cust_id列,一个在customers中,另一个在orders中,需要比较这两个列以正确地把订单与他们相应的顾客匹配。如果不完全限定列名,MySQL将假定你是对orders表中的cust_id进行自身比较。而SELECT COUNT(*) FROM orders WHERE cust_id = cust_id;总是返回orders表中的订单总数(因为MySQL查看每个订单的cust_id是否与自身匹配,当然,它们总是匹配的)。

逐渐增加子查询来建立查询:用子查询测试和调试查询很有技巧性,特别是在这些语句的复杂性不断增加的情况下更是如此。用子查询建立(和测试)查询的最可靠的方法是逐渐进行。首先,建立和测试最内层的查询。然后,用硬编码数据建立和测试外层查询,并且仅在确认它正常后才嵌入子查询。这时,再次测试它。对于要增加的每个查询,重复这些步骤。这样做仅给构造查询增加了一点点时间,但节省了以后(找出查询为什么不正常)的大量时间,并且极大地提高了查询一开始就正常工作的可能性。


联结表

SQL最强大的功能之一就是能在数据检索查询的执行中联结(join)表。联结是利用SQL的SELECT能执行的最重要的操作,很好地理解联结及其语法是学习SQL的一个极为重要的组成部分。

关系表

关系表的设计就是要保证把信息分解成多个表,一类数据一个表。各表通过某些常用的值(即关系设计中的关系)互相关联。

外键:外键为某个表中的一列,它包含另一个表的主键值,定义了两个表之间的关系。

这样做的好处如下:

  • 信息不重复,从而不浪费时间和空间
  • 如果某一个信息变动,可以只更新对应表的单个记录,相关表中的数据不用改动
  • 由于数据无重复,显然数据是一致的,这使得处理数据不用改动

可伸缩性:能够适应不断增加的工作量而不失败。设计良好的数据库或应用程序称之为可伸缩性好。

用联结的理由:如果数据存储在多个表中,用单条SELECT语句检索出数据,此时可以使用联结。联结是一种机制,用来在SELECT语句中关联表,因此称之为联结。使用特殊的语法,可以联结多个表返回一组输出,联结在运行时关联表中正确的行。

维护引用完整性:联结不是物理实体。它在实际的数据库表中不存在。联结由MySQL根据需要建立,它存在于查询的执行当中。在使用关系表时,仅在关系列中插入合法的数据非常重要。这就是维护引用完整性,它是通过在表的定义中指定主键和外键来实现的。

创建联结

联结的创建非常简单,规定要联结的所有表以及它们如何关联即可。如下:

SELECT vend_name, prod_name, prod_price
FROM vendors, products
WHERE vendors.vend_id = products.vend_id
ORDER BY vend_name,prod_name;

此处,SELECT语句与前面的所有语句一样指定要检索的列。这里,最大的差别是所指定的两个列(prod_name和prod_price)在一个表中,而另一个列(vend_name)在另一个表中。

这里的FROM子句列出了两个表,分别是vendors和products。它们就是这条SELECT语句联结的两个表的名字。这两个表用WHERE子句正确联结,WHERE子句指示MySQL匹配vendors表中的vend_id和products表中的vend_id。这里需要完全限定列名,因为如果只给出vend_id,则MySQL不知道指的是哪一个。

完全限定列名:在引用的列可能出现二义性时,必须使用完全限定列名(用一个点分隔的表名和列名)。如果引用一个没有用表名限制的具有二义性的列名,MySQL将返回错误。

WHERE子句的重要性

在一条SELECT语句中联结几个表时,相应的关系是在运行中构造的。在数据库表的定义中不存在能指示MySQL如何对表进行联结的东西。在联结两个表时,实际上做的是将第一个表中的每一行与第二个表中的每一行配对。WHERE子句作为过滤条件,它只包含匹配给定条件(这里是联结条件)的行。没有WHERE子句,第一个表中的每一行将与第二个表中的每个行配对,而不管它们逻辑上是否可以配在一起。

笛卡尔积:由没有哦联结条件的表关系返回的结果是笛卡尔积。检索出的行的数目将是第一个表中的行数乘以第二个表中的行数。

不要忘了WHERE子句:应该保证所有连接都有WHERE子句,否则MySQL将返回比想要的数据多得多的数据。同理,应该保证WHERE子句的正确性。不正确的过滤条件将导致MySQL返回不正确的数据。

叉联结:返回称为叉联结的笛卡尔积的联结类型。

内部联结

之前使用的联结称为等值联结,它基于两个表之间的相等测试。这种联结也称为内部联结。对于这种联结可以使用稍微不同的语法来明确指定联结的类型。

SELECT vend_name,prod_name, prod_price
FROM vendors INNER JOIN products
ON vendors.vend_id = products.vend_id;

这里,两个表之间的关系是FROM子句的组成部分,以INNER JOIN指定。在使用这种语法时,联结条件用特定的ON子句而不是WHERE子句给出。传递给ON的实际条件与传递给WHERE的相同。

使用哪种语法:ANSI SQL规范首选INNER JOIN语法。此外,尽管使用WHERE子句定义联结的确比较简单,但是使用明确的联结语法能够确保不会忘记联结条件,有时候这样做也能影响性能。

联结多个表

SQL对一条SELECT语句中可以联结的表的数目没有限制。创建联结的基本规则也相同。首先列出所有表,然后定义表之间的关系。例如:

SELECT prod_name,vend_name, prod_price,quantity
FROM orderitems,products,vendors
WHERE products.vend_id = vendors.vend_id 
AND orderitems.prod_id = products.prod_id
AND order_num = 20005;

此例子显示编号为20005的订单中的物品。订单物品存储在orderitems表中。每个产品按其产品ID存储,它引用products表中的产品。这些产品通过供应商ID联结到vendors表中相应的供应商,供应商ID存储在每个产品的记录中。这里的FROM子句列出了3个表,而WHERE子句定义了这两个联结条件,而第三个联结条件用来过滤出订单20005中的物品。

性能考虑:MySQL在运行时关联指定的每个表以处理联结。这种处理可能是非常耗费资源的,因此应该仔细,不要联结不必要的表。联结的表越多,性能下降越厉害。

之前的子查询例子:

SELECT cust_name, cust_contact
FROM customers
WHERE cust_id IN(SELECT cust_id
                 FROM orders
                 WHERE order_num IN(SELECT order_num
                                    FROM orderitems
                                    WHERE prod_id = 'TNT2'));

子查询并不总是执行复杂SELECT操作的最有效的方法,下面使用联结方式查询:

SELECT cust_name, cust_contact
FROM custorms, orders, orderitems
WHERE customers.cust_id = orders.cust_id
AND orderitems.order_num = orders.order_num
AND prod_id = 'TNT2';

这个查询中返回数据需要使用3个表。但这里不使用嵌套子查询,而是使用两个联结。前两个关联联结中的表,后一个过滤产品TNT2的数据。

联结是SQL中最重要最强大的特性,有效地使用联结需要对关系数据库设计有基本的了解。


创建高级联结

使用表别名

之前一个例子如下:

SELECT Concat(RTrim(vend_name),'(',RTrim(vend_country),')') AS
vend_title
FROM vendors
ORDER BY vend_name;

别名除了用于列名和计算字段外,SQL还允许给表名起别名。理由如下:

  • 缩短SQL语句
  • 允许在单条SELECT语句中多次使用相同的表

例子如下:

SELECT cust_name, cust_contact
FROM customers AS c, orders AS o, orderitems AS oi
WHERE c.cust_id = o.cust_id
AND oi.order_num = o.order_num
AND prod_id = 'TNT2';

可以看到,FROM子句中的3个表全都具有别名。customers AS c建立c作为customers的别名,等等。这使得能使用省写的c而不是全名customers。表别名不仅能用于WHERE子句,它还可以用于SELECT的列表、ORDER BY子句以及语句的其他部分。

表别名只在查询执行中使用。与列别名不一样,表别名不返回到客户机。

使用不同类型的联结

自联结

假如发现某物品(其ID为DTNTR)存在问题,因此想知道生产该物品的供应商生产的其他物品是否也存在这些问题。

SELECT prod_id, prod_name
FROM products
WHERE vend_id = (SELECT vend_id 
                 FROM products 
                 WHERE prod_id = 'DTNTR');

这使用了子查询。内部的SELECT语句做了一个简单的检索,返回生产ID为DTNTR的物品供应商的vend_id。该ID用于外部查询的WHERE子句中,以便检索出这个供应商生产的所有物品。

现在使用联结的相同查询:

SELECT p1.prod_id, p1.prod_name
FROM products AS p1, products AS p2
WHERE p1.vend_id = p2.vend_id
AND p2.prod_id = 'DTNTR';

此查询中需要的两个表实际上是相同的表,因此Products表在FROM子句中出现了两次。虽然这是完全合法的,但对products的引用具有二义性,因为MySQL不知道引用的是products表中的哪一个实例。

为解决此问题,使用了表别名。products的第一次出现为别名p1,第二次出现为别名p2.现在可以将这些别名作为表名。然后使用p1前缀明确地给出所需列的全名。如果不这样,MySQL将返回错误,因为分别存在两个名为prod_id、prod_name的列。MySQL不知道想要的是哪一个列(即使事实上是同一个列)。WHERE(通过匹配p1中的vend_id和p2中的vend_id)首先联结两个表,然后按第二个表中的prod_id过滤数据,返回所需的数据。

用自联结而不用子查询:自联结通常作为外部语句用来替代从相同表中检索数据时使用的子查询语句。虽然最终的结果是相同的,但有时候处理联结远比处理子查询快得多。应该试一下两种方法,以确定哪一种的性能更好。

自然联结

标准的联结(内部联结)返回所有数据,甚至相同的列多次出现。自然联结多次出现,使每一个列只返回一次。

系统不完成这项工作,由自己完成它。自然联结是这样一种联结,其中只能选择那些唯一的列。这一般是通过对使用通配符(SELECT *),对所有其他表的列明确使用的子集来完成的。

SELECT c.*, o.order_num, o.order_Date, oi.prod_id, oi.quantity, oi.item_price
FROM customers AS c, orders AS o, orderitems AS oi
WHERE c.cust_id = o.cust_id
AND oi.order_num = o.order_num
AND prod_id = 'FB';

在这个例子中,通配符只对第一个表使用。所有其他列明确列出,所以没有重复的列被检索出来。

外部联结

许多联结将一个表中的行与另一个表中的行相关联。但有时候会需要包含没有关联行的那些行。例如,可能需要使用联结来完成以下工作:

  • 对每个客户下了多少订单进行计数,包括那些至今尚未下订单的客户;
  • 列出所有产品及订购数量,包括没有人订购的产品;
  • 计算平均销售规模,包括那些至今尚未下订单的客户

在上述例子中,联结包含了那些在相关表中没有关联行的行。这种类型的联结称为外部联结。如下:

SELECT customers.cust_id, orders.order_num
FROM customers INNER JOIN orders
ON customers.cust_id = orders.cust_id;

外部联结类似。为了检索所有客户,包括那些没有订单的客户,可如下进行:

SELECT customers.cust_id, orders.order_num
FROM customers LEFT OUTER JOIN orders
ON customers.cust_id = orders.cust_id;

类似于上面的内部联结,这条SELECT语句使用了关键字OUTER JOIN来指定联结的类型(而不是在WHERE子句中指定)。但是,与内部联结关联两个表中的行不同的是,外部联结还包括没有关联行的行。在使用OUTER JOIN语法时,必须使用LEFT或者RIGHT关键字指定包括其所有行的表(RIGHT指出的是OUTER JOIN右边的表,而LEFT指出的是OUTER JOIN左边的表)。上面的例子使用LEFT OUTER JOIN从FROM子句的左边表中选择所有行。为了从右边的表中选择所有行,应该使用RIGHT OUTER JOIN,如下例所示:

SELECT customers.cust_id, orders.order_num
FROM customers RIGHT OUTER JOIN orders
ON orders.cust_id = customers.cust_id;

没有*=操作符:MySQL不支持简化字符*=和=*的使用。这两种操作符在其他DBMS中很流行。

外部联结的类型:存在两种基本的外部联结形式:左外部联结和右外部联结。它们之间的唯一差别是所关联的表的顺序不同。换句话说,左外部联结可通过颠倒FROM或WHERE子句中表的顺序转换为右外部联结。因此,两种类型的外部联结可互相转换,究竟使用哪一种纯粹是根据方便而定。

使用带聚集函数的联结

如下:

SELECT customers.cust_name,customers.cust_id, COUNT(orders.order_num) AS num_ord
FROM customers INNER JOIN orders
ON customers.cust_id = orders.cust_id
GROUP BY customers.cust_id;

此SELECT语句使用INNER JOIN将customers和orders表互相关联。GROUP BY子句按客户分组数据,因此,函数调用COUNT(orders.order_num)对每个客户的订单计数,将它作为num_ord返回。

聚集函数也可以与其他联结使用:

SELECT customers.cust_name,customers.cust_id,COUNT(orders.order_num) AS num_ord
FROM customers LEFT OUTER JOIN orders
ON customers.cust_id = orders.cust_id
GROUP BY customers.cust_id;

这个例子使用左外部联结来包含所有客户,甚至包含那些没有任何下订单的客户。

使用联结和联结条件

  • 注意所使用的联结类型。一般我们使用内部联结,但使用外部联结也是有效的。
  • 保证使用正确的联结条件,否则将返回不正确的数据。
  • 应该总是提供联结条件,否则会出现笛卡尔积。
  • 在一个联结中可以包含多个表,甚至对于每个联结可以采用不同的联结类型。虽然这样做事合法的,一般也很有用,但应该在一起测试它们前,分别测试每个联结。这将使故障排除更为简单。

组合查询

执行多个查询(多条SELECT语句),并将结果作为单个查询结果集返回。这些组合通常被称为并或复合查询。可能用到的情况如下:

  • 在单个查询中从不同的表返回类似结构的数据。
  • 对单个表执行多个查询,按单个查询返回数据。

组合查询和多个WHERE条件: 多数情况下,组合相同表的两个查询完成的工作与具有多个WHERE子句条件的单条查询完成的工作相同。换句话说,任何具有多个WHERE子句的SELECT语句都可以作为一个组合查询给出,在以下段落中可以看到这一点。这两种技术在不同的查询中性能也不同。因此,应该试一下这两种技术,以确定对特定的查询哪一种性能更好。

创建组合查询

用UNION操作符来组合数条SQL查询。

使用UNION

给出每条SELECT语句,在各条语句之间放上关键字UNION。

假如需要价格小于等于5的所有物品的一个列表,而且还想包括供应商1001和1002生产的所有物品。

SELECT vend_id, prod_id,prod_price
FROM products
WHERE prod_price <= 5;
SELECT vend_id, prod_id, prod_price
FROM products
WHERE vend_id IN(1001,1002);

第一条SELECT检索价格不高于5的所有物品。第二条SELECT使用IN找出供应商1001和1002生产的所有物品。组合查询如下:

SELECT vend_id, prod_id,prod_price
FROM products
WHERE prod_price <= 5
UNION 
SELECT vend_id, prod_id, prod_price
FROM products
WHERE vend_id IN (1001,1002);

UNION指示MySQL执行两条SELECT语句,并把输出合并成单个查询结果集。使用多条WHERE子句如下:

SELECT vend_id,prod_id,prod_price
FROM products
WHERE prod_price <= 5
OR vend_id IN (1001,1002);

在这里,使用UNION可能比使用WHERE子句更为复杂,但是更复杂的过滤条件,或者从多个表中检索数据时,使用UNIION可能会使处理更简单。

UNION规则

  • UNION必须由两条或两条以上的SELECT语句组成,语句之间用关键字UNION分隔(因此,如果组合4条SELECT语句,必须使用3个UNION关键字)。
  • UNION中的每个查询必须包含相同的列、表达式或聚集函数(不过各个列不需要以相同的次序列出)。
  • 列数据类型必须兼容:类型不必完全相同,但必须是DBMS可以隐含地转换的类型。

包含或取消重复的行

UNION从查询结果集中自动去除了重复的行。这是UNION默认行为,如果需要保留重复的行,可以使用UNION ALL。

SELECT vend_id,prod_id,prod_price
FROM products
WHERE prod_price <= 5
UNION ALL
SELECT vend_id, prod_id,prod_price
FROM products
WHERE vend_id IN (1001,1002);

UNION与WHERE:UNION几乎总是完成与多个WHERE条件相同的工作。UNION ALL为UNION的一种形式,它完成WHERE子句完成不了的工作。如果确实需要每个条件的匹配行全部出现(包括重复行),则必须使用UNION ALL而不是WHERE。

对组合查询结果进行排序

SELECT语句的输出用ORDER BY子句排序。在用UNION组合查询时,只能使用ORDER BY子句,它必须出现在最后一条SELECT语句之后。对于结果集,不存在用一种方式排序一部分,而又使用另一种方式排序另一部分的情况,因此不允许使用多条ORDER BY子句。

SELECT vend_id,prod_id,prod_price
FROM products
WHERE prod_price <= 5
UNION 
SELECT vend_id, prod_id,prod_price
FROM products
WHERE vend_id IN (1001,1002)
ORDER BY vend_id,prod_price;

MySQL将用最后一条ORDER BY子句对所有SELECT语句返回的结果进行排序。

UNION可以组合不同的表。


全文本搜索

并非所有引擎都支持全文本搜索:两个最常用的的引擎为MyISAM和InnDB,前者支持全文本搜索,而后者不支持。

虽然前面讲述了使用LIKE和正则表达式作进行文本搜索,但是存在几个限制,如下:

  • 性能:通配符和正则表达式匹配通常要求MySQL尝试匹配表中所有行(而且这些搜索极少使用表索引)。因此,由于被搜索行数不断增加,这些搜索可能非常耗时。
  • 明确控制:使用通配符和正则表达式匹配,很难(而且并不是总能)明确地控制匹配什么和不匹配什么。例如,指定一个词必须匹配,一个词必须不匹配,而一个词仅在第一个词确实匹配的情况下才可以匹配或者才可以不匹配。
  • 智能化的结果:虽然基于通配符和正则表达式的搜索提供了非常灵活的搜索,但它们都不能提供一种智能化的选择结果的方法。例如,一个特殊词的搜索将会返回包含该词的所有行,而不区分
    包含单个匹配的行和包含多个匹配的行(按照可能是更好的匹配来排列它们)。类似,一个特殊词的搜索将不会找出不包含该词但包含其他相关词的行。

使用全文本搜索

为了进行全文本搜索,必须索引被搜索的列,而且随着数据的改变不断地重新索引。在索引之后,SELECT和Match()和Against()一起使用以实际执行搜索。

启用全文本搜索支持

一般在创建表时启用全文本搜索。CREATE TABLE语句接受FULLTEXT子句,它给出被索引列的一个逗号分隔的列表。

CREATE TABLE productnotes
(
    note_id     int         NOT NULL AUTO_INCREMENT,
    prod_id     char(10)     NOT NULL,
    note_date     datetime    NOT NULL,
    note_text    text        NULL,
    PRIMARY KEY(note_id),
    FULLTEXT(note_text)
)ENGINE = MyISAM;

这条CREATE TABLE语句定义表productnotes并列出它所包含的列即可。这些列中有一个名为note_text的列,为了进行全文本搜索,MySQL根据子句FULLTEXT(note_text)的指示对它进行索引。这里只索引一个列,也可以索引多个列。

不要在导入数据时使用FULLTEXT:更新索引需要花点时间,虽然不是很多,但毕竟要时间。如果正在导入数据到一个新表,此时不应该启用FULLTEXT索引。应该首先导入所有数据,然后再修改表,定义FULLTEXT。这样有助于更快地导入数据(而且使索引数据的总时间小于在导入每行时分别进行索引所需的总时间。

进行全文本搜索

在索引之后,使用两个函数Match()和Against()执行全文本搜索,其中Match()指定被搜索的列,Against()指定要使用的搜索表达式。

SELECT note_text
FROM productnotes
WHERE Match(note_text) Against('rabbit');

此SELECT语句检索单个列note_text。由于WHERE子句,一个全文本搜索被执行。Match(note_text)指示MySQL针对指定的列进行搜索,Against(‘rabbit’)指定词rabbit作为搜索文本。

使用完整的Match()说明:传递给Match()的值必须与FULLTEXT()定义中的相同。如果指定多个列,则必须列出它们(而且次序正确)。

搜索不区分大小写:除非使用BINARY,否则全文本搜索不区分大小写。

上述语句可用LIKE达成同样的目的:

SELECT note_text
FROM productnotes
WHERE note_text LIKE '%rabbit%';

但是使用全文本搜索,将返回以文本匹配的良好程度排序的数据。全文本搜索的一个重要的部分就是对结果排序。具有较高等级的行先返回。

演示排序:

SELECT note_text
       Match(note_text) Against('rabbit') AS rank
FROM productnotes;

在SELECT而不是WHERE子句中使用Match()和Against()。这使所有行都被返回。

排序多个搜索项:如果指定多个搜索项,则包含多数匹配词的那些行将具有比包含较少词(或仅有一个匹配)的那些行高的等级值。

使用查询扩展

查询扩展用来设法放宽所返回的全文本搜索结果的范围。

在使用查询扩展时,MySQL对数据和索引进行两遍扫描来完成搜索。

  • 进行一个基本的全文本搜索,找出与搜索条件匹配的所有行;
  • 其次,MySQL检查这些匹配行并选择所有有用的词
  • 再其次,MySQL再次进行全文本搜索,这次不仅使用原来的条件,而且使用所有有用的词。
SELECT note_text
FROM productnotes
WHERE Match(note_text) Against('anvils' WITH QUERY EXPANSION);

布尔文本搜索

全文本搜索的一种形式:布尔方式

  • 要匹配的词
  • 要排斥的词(如果某行包含这个词,则不返回该行,即使它包含其他指定的词也是如此)
  • 排列提示(指定某些词比其他词更重要,更重要的词等级更高)
  • 表达式分组
  • 另外一些内容

即使没有FULLTEXT索引也可以使用:即使没有定义FULLTEXT索引,也可以使用它。但这是一种非常缓慢的操作。

SELECT note_text
FROM productnotes
WHERE Match(note_text) Against('heavy' IN BOOLEAN MODE);

此全文本搜索检索包含词heavy的所有行。其中使用关键字IN BOOLEAN MODE,但实际上没有指定布尔操作符,因此,其结果与没有指定布尔方式的结果相同。

为了匹配包含heavy但不包含任意以rope开始的词的行,使用以下查询:

SELECT note_text
FROM productnotes
WHERE Match(note_text) Against('heavy -rope*' IN BOOLEAN MODE);

其中-rope*明确地指示MySQL排除包含rope*(任何以rope开始的词)的行。

全文本搜索布尔操作符:

布尔操作符 说明
+ 包含,词必须存在
- 排除,词必须不出现
> 包含,而且增加等级值
< 包含,而且减少等级值
() 把词组成子表达式(允许这些子表达式作为一个组被包含、排除、排列等)
~ 取消一个词的排序值
* 词尾的通配符
“” 定义一个短语(与单个词的列表不一样,它匹配整个短语以便包含或排除这个短语
SELECT note_text
FROM productnotes
WHERE Match(note_text) Against('+rabbit +bait' IN BOOLEAN MODE);

这个搜索匹配包含词rabbit和bait的行

SELECT note_text
FROM productnotes
WHERE Match(note_text) Against('rabbit bait' IN BOOLEAN MODE);

没有指定操作符,这个搜索匹配rabbit和bait中的至少一个词。

SELECT note_text
FROM productnotes
WHERE Match(note_text) Against('"rabbit bait"' IN BOOLEAN MODE);

这个搜索匹配短语rabbit bait而不是匹配两个词rabbit和bait。

SELECT note_text
FROM productnotes
WHERE Match(note_text) Against('>rabbit <carrot' IN BOOLEAN MODE);

匹配rabbit和carrot,增加前者的等级,降低后者的等级。

SELECT note_text
FROM productnotes
WHERE Match(note_text) Against('+safe +(<combination)' IN BOOLEAN MODE);

这个搜索匹配词safe和combination,降低后者的等级。

排列而不排序:在布尔方式中,不按等级值降序排序返回的行。

全文本搜索的使用说明

  • 在索引全文本数据时,短词被忽略且从索引中排除。短词定义为那些具有3个或3个以下字符的词(如果需要,这个数目可以更改)。
  • MySQL带有一个内建的非用词(stopword)列表,这些词在索引全文本数据时总是被忽略的。如果需要,可以覆盖这个列表。
  • 许多词出现的频率很高,搜索它们没有用处(返回太多的结果)。因此,MySQL规定了一条50%规则,如果一个词出现在50%以上的行中,则将它作为一个非用词忽略。50%规则不用于IN BOOLEAN MODE。
  • 如果表中的行数少于3行,则全文本搜索不返回结果(因此每个词或者不出现,或者至少出现在50的行中)。
  • 忽略词中的单引号。例如,don’t索引dont
  • 不具有词分隔符(包括日语和汉语)的语言不能恰当地返回全文本搜索结果。
  • 仅在MyISAM数据库引擎中支持全文本搜索。

插入数据

INSERT是用来插入(或添加)行到数据库表的。插入可以用几种方式使用:

  • 插入完整的行
  • 插入行的一部分
  • 插入多行
  • 插入某些查询的结果

插入及系统安全:可针对每个表或每个用户,利用MySQL的安全机制禁止使用INSERT语句。

插入完整的行

把数据插入表中的最简单的方法是使用基本的INSERT语法,它要求指定表名和被插入的新行中的值。

INSERT INTO Customers
VALUES(NULL,
      'Pep E. LaPew',
      '100 Main Street',
      'LOS Angeles',
      'CA',
      '90046',
      'USA',
      NULL,
      NULL);

没有输出:INSERT语句一般不会产生输出

此例子插入一个新客户到customers表。存储到每个表列中的数据在VALUES子句中给出,对每个列必须提供一个值。如果某个列没有值,应该使用NULL值(假定表允许对该列指定空值)。各个列必须以它们在表定义中出现的次序填充。第一列为NULL值,这是因为每次插入一个新行时,该列有MySQL自动增量。不想给出一个值(这是MySQL的工作),但又不能省略该列,所以指定一个NULL值(它被MySQL忽略,MySQL在这里插入下一个可用的cust_id值)。

这种语法很简单,但是不安全,应该尽量避免使用。高度依赖表中列的定义次序,并且还依赖其次序容易获得的信息。更安全的方法如下:

INSERT INTO customers(
    cust_name,
    cust_address,
    cust_city,
    cust_state,
    cust_zip,
    cust_country,
    cust_contact,
    cust_email
)
VALUES(
    'Pep E. LaPew',
    '100 Main Street',
    'Los Angles',
    'CA',
    '90046',
    'USA',
    NULL,
    NULL
);

这个例子与上面的例子完全相同,但是在表名后的括号里明确地给出了列名。在插入行时,MySQL将用VALUES列表中的相应值填入列表中的对应项。VALUES中的第一个值对应于第一个指定的列名。

总是使用列的列表:一般不要使用没有明确给出列的列表的INSERT语句。使用列的列表能使SQL代码继续发挥作用,即使表结构发生了变化。

省略列:如果表的定义允许,则可以在INSERT操作中省略某些列。省略的列必须满足以下条件:

  • 该列定义为允许NULL值(无值或空值)。
  • 在表定义中给出默认值。这表示如果不给出值,将使用默认值。

提高整体性能:INSERT操作可能很耗时,而且可能降低等待处理的SELECT语句的性能。如果数据检索是最重要的,则可以通过在INSERT和INTO之间添加关键字LOW_PRIORITY,指示MySQL降低INSERT语句的优先级,如下示例

INSERT LOW_PRIORITY INTO

这也适用于UPDATE和DELETE语句。

插入多个行

使用多条INSERT语句,甚至一次提交,每个语句用一个分号结束。

INSERT INTO customers(
    cust_name,
    cust_address,
    cust_city,
    cust_state,
    cust_zip,
    cust_country,
    cust_contact,
    cust_email
)
VALUES(
    'Pep E. LaPew',
    '100 Main Street',
    'Los Angles',
    'CA',
    '90046',
    'USA',
    NULL,
    NULL
);
INSERT INTO customers(
    cust_name,
    cust_address,
    cust_city,
    cust_state,
    cust_zip,
    cust_country,
    cust_contact,
    cust_email
)
VALUES(
    'M. Martian',
    '42 Galaxy Way',
    'New York',
    'NY',
    '11213',
    'USA',
    NULL,
    NULL
);

或者,只要每条INSERT语句中的列名(和次序)相同,可以如下组合各语句:

INSERT INTO customers(
    cust_name,
    cust_address,
    cust_city,
    cust_state,
    cust_zip,
    cust_country,
    cust_contact,
    cust_email
)
VALUES(
    'Pep E. LaPew',
    '100 Main Street',
    'Los Angles',
    'CA',
    '90046',
    'USA',
    NULL,
    NULL
),
(
    'M. Martian',
    '42 Galaxy Way',
    'New York',
    'NY',
    '11213',
    'USA',
    NULL,
    NULL
);

其中单条INSERT语句有多组值,每组值用一对圆括号括起来,用逗号分隔。

提高INSERT的性能:单条INSERT语句处理多个插入比使用多条INSERT语句快。

插入检索出的数据

INSERT SELECT:将一条SELECT语句的结果插入表中。在插入时,不应该出现重复的主键值,如果主键值重复,后续的INSERT操作将会失败,或仅省略这列值让MySQL在导入数据的过程中产生新值。

INSERT INTO customers(
    cust_name,
    cust_address,
    cust_city,
    cust_state,
    cust_zip,
    cust_country,
    cust_contact,
    cust_email
)
SELECT cust_name,
    cust_address,
    cust_city,
    cust_state,
    cust_zip,
    cust_country,
    cust_contact,
    cust_email
FROM custnew;    

这个例子使用INSERT SELECT从custnew中将所有数据导入customers。SELECT语句从custnew检索出要插入的值,而不是列出它们。SELECT中列出的每个列对应于custnew表名后所跟的列表中的每个列。

INSERT SELECT中的列名:不一定要求列名匹配。MySQL不关心SELECT返回的列名。它使用的是列的位置,因此SELECT中的第一列将用来填充表列中的第一列。

INSERT SELECT中的SELECT语句可包含WHERE子句以过滤插入的数据。


更新和删除数据

更新数据

  • 更新表中特定行
  • 更新表中的所有行

不要省略WHERE子句:在使用UPDATE时,一定要注意细心。因为稍不注意,就会更新表中的所有行。

UPDATE语句由三部分组成:

  • 要更新的表
  • 列名和他们的新值
  • 确定要更新行的过滤条件
UPDATE customers
SET cust_email = 'elmer@fudd.com'
WHERE cust_id = 10005;

UPDATE语句总是以要更新的表的名字开始。SET命令用来将新值赋给被更新的列。

更新多个列的语法:

UPDATE customers
SET cust_name = 'The fudds',
    cust_email = 'elmer@fuss.com'
WHERE cust_id = 10005;

在更新多个列时,只需要使用单个SET命令,每个“列=值”对之间用逗号分隔(最后一列不用逗号)。

在UPDATE语句中使用子查询:UPDATE语句中可以使用子查询,使得能用SELECT语句检索出的数据更新列数据。

IGNORE关键字:如果用UPDATE语句更新多行,并且在更新这些行中的一行或多行时发生一个错误,则整个UPDATE操作被取消(错误发生前更新的行被恢复到它们原来的值)。为即使是发生错误,也继续进行更新,可使用IGNORE关键字。UPDATE IGNORE customers…。

为了删除某个列的值,可设置它为NULL值。(加入表定义允许NULL值)。

UPDATE customers
SET cust_email = NULL
WHERE cust_id = 10005;

删除数据

  • 从表中删除特定的行
  • 从表中删除所有行
DELETE FROM customers WHERE cust_id = 10006;

DELETE FROM要求指定从中删除数据的表名。WHERE子句过滤要删除的行。DELETE不需要列名或通配符。DELETE删除整行而不是删除列。为了删除指定的列,使用UPDATE。

删除表的内容而不是删除表:DELETE语句从表中删除行,甚至是删除表中的所有行。但是,DELETE不删除表本身。

更快的删除:如果想删除所有行,不要使用DELETE。使用TRUNCATE TABLE语句,它完成相同的工作,但速度更快(TRUNCATE实际是删除原来的表并重新创建一个新表,而不是逐行删除表中的数据)。

更新和删除的指导原则

  • 除非确实打算更新和删除每一行,否则绝对不要使用不带WHERE子句的UPDATE或DELETE语句。
  • 保证每个表都有主键,尽可能像WHERE子句那样使用它(可以指定各主键、多个值或值的范围)。
  • 在对UPDATE或DELETE语句使用WHERE子句前,应该先用SELECT进行测试,保证它过滤的是正确的记录,以防编写的WHERE子句不正确。
  • 使用强制实施引用完整性的数据库,这样MySQL将不允许删除具有与其他表相关联的数据的行。

创建和操纵表

创建表

表创建基础

为利用CREATE TABLE创建表,必须给出下列信息:

  • 新表的名字,在关键字CREATE TABLE之后给出
  • 表列的名字和定义,用逗号分隔
CREATE TABLE customers
(
    cust_id            int            NOT NULL AUTO_INCREMENT,
    cust_name        char(50)    NOT NULL,
    cust_address    char(50)    NULL,
    cust_city        char(50)    NULL,
    cust_state        char(50)    NULL,
    cust_zip        char(50)    NULL,
    cust_country    char(50)    NULL,
    cust_email        char(255)    NULL,
    PRIMARY KEY(cust_id)
)ENGINE=InnoDB;

表名紧跟在CREATE TABLE关键字之后。实际的表定义(所有列)括在圆括号之中。各列之间用逗号分隔。这个表由8列组成。每列的定义及列名(它在表中必须是唯一的)开始,后跟列的数据类型。表的主键可以在创建表时使用PRIMARY KEY关键字指定。这里,指定列cust_id为主键列。整条语句由右圆括号后的分号结束。

处理现有的表:在创建新表时,指定的表名必须不存在,否则将会出错。如果要防止意外覆盖已有的表,SQL要去首先手工删除该表,然后再重建它,而不是简单用创建表语句覆盖它。如果仅想在表不存在时创建它,应该在表名后给出IF NOT EXISTS。

CREATE TABLE IF NOT EXISTS customers
(
    cust_id            int            NOT NULL AUTO_INCREMENT,
    PRIMARY KEY(cust_id)
)ENGINE=InnoDB;

这样就会提示已经创建表了。

使用NULL值

NULL值就是没有值或者缺值。允许NULL值得列也允许在插入行时不给出该列的值。不允许NULL值得列不接受该列没有值的行,换句话说,在插入行或更新行时,该列必须有值。

每个表列或者是NULL列,或者是NOT NULL列,这种状态在创建时由表的定义规定。

CREATE TABLE orders
(
    order_num        int            NOT NULL    AUTO_INCREMENT,
    order_date        datetime    NOT NULL,
    cust_id            int            NOT NULL,
    PRIMARY KEY(order_num)
)ENGINE=InnoDB;

这条语句创建本书中所用的orders表。orders表包含3个列,分别是订单号、订单日期和客户ID。所有3个列都需要,因此每个列的定义都含有关键字NOT NULL。这将会阻止插入没有值得列。如果试图插入没有值的列,将返回错误,且插入失败。

CREATE TABLE vendors
(
    vend_id            int            NOT NULL AUTO_INCREMENT,
    vend_name        char(50)    NOT NULL,
    vend_address    char(50)    NULL,
    vend_city        char(50)    NULL,
    PRIMARY KEY(vend_id)
)ENGINE=InnoDB;

理解NULL:NULL值不是空串。NULL值时没有值,它不是空串。如果指定‘ ’(两个单引号,其间没有字符),这在NOT NULL列中是允许的。空串是一个有效的值,它不是无值。NULL值用关键字NULL而不是空串指定。

主键再介绍

主键值必须是唯一的。表中的每一行必须具有唯一的主键值。如果主键使用单个列,则它的值必须唯一。如果使用多个列,则这些的列的组合值必须是唯一的。

多个列作为主键;

CREATE TABLE orderitems
(
    order_num         int            NOT NULL,
    order_item        int         NOT NULL,
    prod_id            char(10)    NOT NULL,
    PRIMARY KEY(order_num,order_item)
)ENGINE=InnoDB;

主键可以在创建表时定义,或者在创建表之后定义。

主键和NULL值:主键只能使用不允许NULL值得列。允许NULL值得列不能作为唯一标识。

使用AUTO_INCREMENT

AUTO_INCREMENT告诉MySQL,本列每当增加一行时自动增量。每次执行一个INSERT操作时,MySQL自动对该列增量(从而才有这个关键字AUTO_INCREMENT),给该列赋予下一个可用的值。这样给每个行分配一个唯一的值,从而可以用作主键值。

每个表只允许一个AUTO_INCREMENT列,而且它必须被索引。

覆盖AUTO_INCREMENT:如果一个列被指定为AUTO_INCREMENT,可以简单地在INSERT语句中指定一个值,只要它是唯一的(至今尚未使用过)即可,该值将被用来替代自动生成的值。后续的增量将开始使用该手工插入的值。

确定AUTO_INCREMENT的值:使用AUTO_INCREMENT列时,获得最新的值,可以使用last_insert_id()函数,示例:

SELECT last_insert_id();

指定默认值

如果在插入行时没有给出值,MySQL允许指定此时使用的默认值。默认值用CREATE TABLE语句的列定义中的DEFAULT关键字指定。

CREATE TABLE orderitems
(
    order_num        int         NOT NULL,
    order_item        char(50)    NOT NULL DEFAULT '1';
    PRIMARY KEY(order_num)
)ENGINE=InnoDB;

这条语句创建一个orderitems表。其中列order_item有默认值‘1’。

不允许函数:MySQL不允许使用函数作为默认值,它只支持常量。

使用默认值而不是NULL值

引擎类型

  • InnoDB是一个可靠的事务处理引擎,它不支持全文本搜索;
  • MEMORY在功能等同于MyISAM,但由于数据存储在内存(不是磁盘)中,速度很快(特别适合于临时表);
  • MyISAM是一个性能极高的引擎,它支持全文本搜索,但不支持事务处理。

引擎类型可以混用。

外键不能跨引擎:混用引擎有一个大缺陷。外键(用于强制实施引用完整性)不能跨引擎,即使用一个引擎的表不能引用具有使用不同引擎的表的外键。

更新表

为更新表定义,可使用ALTER TABLE语句。但是,在理想状态下,当表中存储数据以后,该表不应该再被更新。

  • 在ALTER TABLE之后给出要更改的表名(该表必须存在,否则将会出错);
  • 所做更改的列表
ALTER TABLE vendors
ADD vend_phone CHAR(20);

这条语句给vendors表增加一个名为vend_phone的列,必须明确其数据类型。

删除刚刚的列,可以这样做:

ALTER TABLE vendors
DROP COLUMN vend_phone;

ALTER TABLE的一种常见用途是定义外键。如下:

ALTER TABLE orderitems
ADD CONSTRAINT fk_orderitems_orders
FOREIGN KEY (order_num) REFERENCES orders (order_num);

ALTER TABLE orderitems
ADD CONSTRAINT fk_orderitems_products
FOREIGN KEY (prod_id) REFERENCES orders (prod_id);

这里,对同一个表使用了两次ALTER TABLE语句进行两次更改,但是可以使用单条ALTER TABLE语句,每个更改用逗号分隔。

ALTER TABLE productnotes add prod_phone char(50), add cust_id char(50);

复杂的表结构更改一般需要手动删除过程,它涉及以下步骤:

  • 用新的列布局创建一个新表;
  • 使用INSERT SELECT语句从旧表复制数据到新表。如果有必要,可使用转换函数和计算字段;
  • 检验包含所需数据的新表;
  • 重命名旧表(如果确定,可以删除它);
  • 用旧表原来的名字重命名新表;
  • 根据需要,重新创建触发器、存储过程、索引和外键。

删除表

删除整个表。使用DROP TABLE语句即可。

DROP TABLE customers2;

这条语句删除customers2表。删除表没有确认,也不能撤销,执行这条语句将永久删除该表。

重命名表

使用RENAME TABLE语句可以重命名一个表

RENAME TABLE customer2 TO customers;

将表customers2重命名为customers。

RENAME TABLE backup_customers TO customers,
            backup_vendors   TO vendors;

使用视图

视图

视图是虚拟的表。只包含使用时动态检索数据的查询。

视图的常见应用:

  • 重用SQL语句
  • 简化复杂的SQL操作。在编写查询后,可以方便地重用它而不必知道它的基本查询细节
  • 使用表的组成部分而不是整个表
  • 保护数据。可以给用户授予表的特定部分的访问权限而不是整个表的访问权限
  • 更改数据格式和表示。视图可返回与底层表的表示和格式不同的数据。

视图仅仅是用来查看存储在别处的数据的一种设施。视图本身不包含数据,因此它们返回的数据是从其他表中检索出来的。在添加或更改这些表中的数据时,视图将返回改变过的数据。

性能问题:因为视图不包含数据,所以每次检索数据时,都必须处理查询执行时所需的任一个检索。如果你用多个联结和过滤创建了复杂的视图或者嵌套了视图,可能会发现性能下降得很厉害。

视图的规则和限制

  • 与表一样,视图必须唯一命名
  • 对于可以创建的视图数目没有限制
  • 为了创建视图,必须具有足够的访问权限。
  • 视图可以嵌套,即可以利用从其他视图中检索数据的查询来构造一个视图。
  • ORDER BY可以用在视图中,但如果从该视图检索数据SELECT中也含有ORDER BY,那么该视图中的ORDER BY 将被覆盖。
  • 视图不能索引,也不能有关联的触发器或默认值
  • 视图可以和表一起使用。

使用视图的方法

  • 视图用CREATE VEIW语句来创建
  • 使用SHOW CREATE VIEW viewname; 来查看创建视图的语句
  • 用DROP删除视图,其语法为DROP VIEW viewname;
  • 更新视图时,可以先用DROP再用CREATE,也可以直接用CREATE OR REPLACE VIEW。如果要更新的视图不存在,则第2条更新语句会创建一个视图;如果要更新的视图存在,则第2条更新语句会替换原有视图。

利用视图简化复杂的联结

CREATE VIEW productcustomers AS 
SELECT cust_name,cust_contact,prod_id
FROM customers,orders,ordeitems
WHERE customers.cust_id = orders.cust_id
AND orderitems.order_num = orders.order_num;

这条语句创建一个名为productcustomers的视图,它联结三个表,以返回已订购了任意客产品的所有客户的列表。如果执行SELECT * FROM productcustomers;将列出订购了任意产品的客户。

SELECT cust_name,cust_contact
FROM productcustomers
WHERE prod_id = 'TNT2';

这条语句通过WHERE子句从视图中检索特定数据。在MySQL处理此查询时,它将指定的WHERE子句添加到视图查询中的已有WHERE子句中,以便正确过滤数据。

用视图重新格式化检索出的数据

SELECT Concat(RTrim(vend_name),'(',RTrim(vend_country),')')
AS vedn_title
FROM vendors
ORDER BY vend_name;

假如经常需要重用这个格式的结果。不必在每次需要时执行廉洁,创建一个视图,每次需要它时使用它即可:

CREATE VIEW vendorlocations AS
SELECT Concat(RTrim(vend_name),'(',RTrim(vend_country),')')
AS vedn_title
FROM vendors
ORDER BY vend_name;

如果想达成目的,如下:

SELECT * FROM vendorlocations;

用视图过滤不想要的数据

CREATE VIEW customeremaillist AS
SELECT cust_id,cust_name,cust_email
FROM customers
WHERE cust_email IS NOT NULL;
SELECT * FROM customeremaillist;

WHERE子句与WHERE子句:如果从视图检索数据时使用了一条WHERE子句,则两组子句(一组在视图中,另一组是传递给视图的)将自动组合。

使用视图与计算字段

SELECT     prod_id,
        quantity,
        item_price,
        quantity*item_price AS enpanded_price
FROM     orderitems
WHERE    order_num=20005;

将其转换为视图:

CREATE VIEW orderitemsexpanded AS
SELECT     prod_id,
        quantity,
        item_price,
        quantity*item_price AS enpanded_price
FROM     orderitems;

为检索订单20005的详细内容,如下;

SELECT *
FROM orderitemsexpanded
WHERE order_num = 20005;

更新视图

如果视图中有以下操作,则不能进行视图的更新:

  • 分组(使用GROUP BY和HAVING)
  • 联结
  • 子查询
  • 聚集函数
  • DISTINCT
  • 导出列

视图主要是数据检索。


使用存储过程

存储过程

好处:简单、安全和高性能。

使用存储过程

称存储过程的执行为调用,因此MySQL执行存储过程的语句称为CALL。CALL接受存储过程的名字以及需要传递给它的任意参数。

CALL productpricing(
    @pricelow,
    @pricehigh,
    @priceaverage
);

创建存储过程

CREATE PROCEDURE productpricing()
BEGIN
    SELECT Avg(prod_price) AS priceaverage
    FROM products;
END;

用CREATE PROCEDURE productpricing()语句定义。如果存储过程接受参数,它们将在()列举出来。此存储过程没有参数,但后跟的()仍然需要。BEGIN和END语句用来限定存储过程体,过程体本身是一个简单的SELECT语句。

MySQL命令行客户机的分隔符:由于默认的MySQL语句的分隔符为;。MySQL命令行实用程序也使用;作为语句分隔符。如果命令行实用程序要解释存储过程自身内的;字符,则他们最终不会成为存储过程的成分,这会使存储过程中的SQL出现句法错误。

解决办法是临时更改命令行实用程序的语句分隔符,如下所示:

DELIMITER //
CREATE PROCEDURE productpricing()
BEGIN
    SELECT Avg(prod_price) AS priceaverage
    FROM products;
END //
DELIMITER ;

其中,DELIMITER // 告诉实用程序使用//作为新的语句分隔符。除 \ 符号外,任何字符都可以用作语句分隔符。

使用上述的存储过程:

CALL productpricing();

执行刚创建的存储过程并显示返回的结果。

删除存储过程

DROP PROCEDURE productpricing;

仅当存在时删除:如果指定的过程不存在,则DROP PROCEDURE将 产生一个错误。当过程存在想删除它时(如果过程不存在也不产生错误)可使用DROP PROCEDURE IF EXISTS。

使用参数

一般存储过程不显示结果,而是把结果返回给指定的变量。

CREATE PROCEDURE productpricing(
    OUT    pl    DECIMAL(8,2),
    OUT    ph    DECIMAL(8,2),
    OUT    pa    DECIMAL(8,2)
)
BEGIN
    SELECT Min(prod_price)
    INTO pl
    FROM products;
    SELECT Max(prod_price)
    INTO ph
    FROM products;
    SELECT Avg(prod_price)
    INTO pa
    FROM products;
END;

此存储过程接受3个参数:pl存储产品最低价格,ph接受产品最高价格,pa存储产品平均价格。每个参数必须具有指定的类型,这里使用十进制。关键字OUT指出相应的参数用来从存储过程传出一个值(返回给调用者)。MySQL支持IN(传递给存储过程)、OUT(从存储过程传出)和INOUT(对存储过程传入和传出)类型的参数。

为调用此修改的存储过程,必须指定3个变量名,如下所示:

CALL productpricing(
    @pricelow,
    @pricehigh,
    @priceaverage
)

变量名:所有MySQL变量都必须以@开始。

为了显示检索出的产品平均价格,可如下:

SELECT @priceaverage;

为了获得3个值,可使用如下语句:

SELECT @pricehigh, @pricelow, @priceaverage;

下面使用IN和OUT参数:

CREATE PROCEDURE ordertotal(
    IN onumber INT,
    OUT ototal DECIMAL(8,2)
)
BEGIN
    SELECT Sum(item_price * quantity)
    FROM orderitems
    WHERE order_num = onumber
    INTO ototal;
END

onumber定义为IN,因为订单号被传入存储过程。ototal定义为OUT,因为要从存储过程中返回。

CALL ordertotal(20005,@total);

建立智能存储过程

-- Name:ordertotal
-- Parameters:        onumber = order number
--                    taxable = 0 if not taxable, 1 if taxable
--                    ototal  = order total variable
cREATE PROCEDURE ordertotal(
        IN         onumber INT,
        IN      taxable BOOLEAN,
        OUT     ototal    DECIMAL(8,2)
) COMMENT 'Obtain order total, optionally adding tax'
BEGIN
    -- Declare variable for total
    DECLARE total DECIMAL(8,2);
    -- Declare tax percentage
    DECLARE taxrate INT DEFAULT 6;

    -- Get the order total
    SELECT Sum(item_price * quantity)
    FROM orderitems
    WHERE order_num = onumber
    INTO total;

    -- Is this taxable?
    IF taxable THEN
        -- Yes, so add taxrate to the total
        SELECT total+(total/100*taxrate) INTO total;
    END IF;
    -- And finally, save to out variable
    SELECT total INTO ototal;
END

此存储过程发生了很大变化。首先,增加了注释(前面放置了- -)。在存储过程体中,用DECLARE语句定义了两个局部变量。DECLARE要求指定变量名和数据类型,它也支持可选的默认值,这里taxrate默认是6。

COMMENT关键字:COMMENT不是必需的,但如果给出,将在SHOW PROCEDURE STATUS的结果中显示。

IF语句:IF语句还支持ELSEIF和ELSE子句(前者还使用THEN子句,后者不使用)。

检查存储过程

为了显示用来创建一个存储过程的CREATE语句,使用SHOW CREATE PROCEDURE语句:

SHOW CREATE PROCEDURE ordertotal;

限制过程状态结果:SHOW PROCEDURE STATUS列出所有存储过程。为限制其输出,可使用LIKE指定一个过滤模式,例如:

SHOW PROCEDURE STATUS LIKE 'ordertotal';

使用游标

游标是一个存储在MySQL服务器上的数据库查询,它不是一条SELECT语句,而是被该语句检索出来的结果集。在存储了游标之后,应用程序可以根据需要滚动或浏览其中的数据。

游标主要用于交互式应用,其中用户需要滚动屏幕上的数据,并对数据进行浏览或更改。

只能用于存储过程:MySQL游标只能用于存储过程(和函数)。

使用游标

  • 在能够使用游标前,必须声明(定义)它。这个过程实际上没有检索数据,它只是定义要使用的SELECT语句。
  • 一旦声明后,必须打开游标以供使用。这个过程用前面的SELECT语句把数据实际检索出来。
  • 对于填有数据的游标,根据需要取出(检索)各行。
  • 在结束游标使用时,必须关闭游标。

创建游标

游标用DECLARE语句创建。DECLARE命名游标,并定义相应的SELECT语句,根据需要带WHERE和其他子句。

CREATE PROCEDURE processorders()
BEGIN
    DECLARE ordernumers CURSOR
    FOR
    SELECT order_num FROM orders;
END;

DECLARE语句用来定义和命名游标,这里为ordernumbers。存储过程处理完成后,游标就消失(因为它局限于存储过程)。

打开和关闭游标

游标用OPEN CURSOR语句来打开;

OPEN ordernumbers;

在处理OPEN语句时执行查询,存储检索出的数据以供浏览和滚动。

游标处理完成后,应当使用如下语句关闭游标:

CLOSE ordernumbers;

CLOSE释放游标使用的所有内部内存和资源,因此在每个游标不再需要时都应该关闭。

隐含关闭:如果不明确关闭游标,MySQL将会在到达END语句时自动关闭它。

CREATE PROCEDURE processorders()
BEGIN
    -- Declare the cursor
    DECLARE ordernumbers CURSOR
    FOR
    SELECT order_num FROM orders;

    -- Open the cursor
    OPEN ordernumbers;

    -- Close the cursor
    CLOSE ordernumbers;
END

使用游标数据

在一个游标被打开后,可以使用FETCH语句分别访问它的每一行。FETCH指定检索什么数据(所需的列),检索出来的数据存储在什么地方。它还向前移动游标中的内部行指针,使下一条FETCH语句检索下一行(不重复读取每一行)。

CREATE PROCEDURE processorders()
BEGIN
    -- Declare local variable
    DECLARE o INT;

    -- Declare the cursor
    DECLARE ordernumbers CURSOR
    FOR
    SELECT order_num FROM orders;

    -- Open the cursor
    OPEN ordernumbers;

    -- Get order number
    FETCH ordernumbers INTO o;

    -- Close the cursor
    CLOSE ordernumbers;
END;

其中,FETCH用来检索当前行的order_num列(将自动从第一行开始)到一个名为o的局部声明的变量中。对检索出来的数据不做任何处理。

在下一个例子中,循环检索数据,从第一行到最后一行:

CREATE PROCEDURE processorders()
BEGIN
    -- Declare local variable
    DECLARE done BOOLEAN DEFAULT 0;
    DECLARE o INT;

    -- Declare the cursor
    DECLARE ordernumbers CURSOR
    FOR 
    SELECT order_num FROM orders;

    -- Declare continue handler
    DECLARE CONTINUE HANDLER FOR SQLSTATE '02000' SET done = 1;

    -- Open the cursor
    OPEN ordernumbers;

    -- Loop through all rows
    REPEAT

        -- Get order number
        FETCH ordernumbers INTO o;

    -- End of loop
    UNTIL done END REPEAT;

    -- Close the cursor
    CLOSE ordernumbers;
END;

这个例子使用FETCH检索当前order_num到声明的名为o的变量中。但与前一个例子不一样的是,这个例子中的FETCH是在REPEAT内,因此它反复执行直到done为真(由UNTIL done END REPEAT;规定)。为使它起作用,用一个DEFAULT 0定义变量done。那么根据下条语句:

DECLARE CONTINUE HANDLER FOR SQLSTATE '02000' SET done=1;

这条语句定义了一个CONTINUE HANDLER,它是在条件出现时被执行的代码。这里,它指出当SQLSTATE ‘02000’出现时,SET done=1。SQLSTATE ‘02000’是一个未找到条件,当REPEAT由于没有更多的行供循环而不能继续时,出现这个条件。

DECLARE语句的顺序:DECLARE语句的发布存在特定的次序。用DECLARE语句定义的局部变量必须在定义任意游标或句柄之前定义,而句柄必须在游标之后定义。不遵守此顺序将发生错误信息。

更进一步修改的版本:

CREATE PROCEDURE processorders()
BEGIN
    -- Declare local variable
    DECLARE done BOOLEAN DEFAULT 0;
    DECLARE o INT;
    DECLARE t DECIMAL(8,2);

    -- Declare the cursor
    DECLARE ordernumbers CURSOR
    FOR
    SELECT order_num FROM orders;
    -- Declare continue handler
    DECLARE CONTINUE HANDLER FOR SQLSTATE '02000' SET done = 1;

    -- Create a table to store the results
    CREATE TABLE IF NOT EXISTS ordertotals
    (order_num INT, total DECIMAL(8,2));

    -- Open the cursor
    OPEN ordernumbers;

    -- Loop through all rows
    REPEAT 
        -- Get order number
        FETCH ordernumbers INTO o;
        -- Get the total for this order
        CALL ordertotal(o,1,t);
        -- Insert order and total into ordertotals
        INSERT INTO ordertotals(order_num,total)
        VALUES(o,t);
    -- End of loop
    UNTIL done END REPEAT;
    -- Close the cursor
    CLOSE ordernumbers;
END;

此存储过程不返回数据,但它能够创建和填充另一个表,,可以用一条简单的SELECT语句查看该表。


触发器

触发器:在某个表发生更改时自动处理。触发器时MySQL响应以下任意一条语句自动执行的一条MySQL语句(或位于BEGIN和END语句之间的一组语句)。

  • DELETE
  • INSERT
  • UPDATE

其他MySQL语句不支持触发器

创建触发器

在创建触发器时,需要给出4条信息:

  • 唯一的触发器名
  • 触发器关联的表
  • 触发器应该响应的活动
  • 触发器何时执行

保持每个数据库的触发器名唯一:在MySQL中触发器名必须在每个表中唯一,但不是在每个数据库中唯一。这表示同一个数据库中的两个表可具有相同名字的触发器。这在其他每个数据库触发器名必须唯一的DBMS中是不允许的,而且以后的MySQL版本很可能会使命名规则更为严格。因此,最好在数据库范围内使用唯一的触发器名。

触发器用CREATE TRIGGER语句创建。

CREATE TRIGGER newproduct AFTER INSERT ON products
FOR EACH ROW SELECT 'Product added';

CREATE TRIGGER用来创建名为newproduct的新触发器。触发器可在一个操作发生之前或之后执行,这里给出了AFTER INSERT,所以此触发器将在INSERT语句成功执行后执行。这个触发器还指定FOR EACH ROW,因此代码对每个插入行执行。在这个例子中,文本Product added将对每个插入的行显示一次。

仅支持表:只有表才支持触发器,视图不支持(临时表也不支持)。

触发器按每个表每个事件每次地定义,每个表每个事件每次只允许一个触发器。因此,每个表最多支持6个触发器(每条INSERT、UPDATE和DELETE的之前和之后)。单一触发器不能与多个事件或多个表关联。

触发器失败:如果BEFORE触发器失败,则MySQL将不执行请求的操作。此外,如果BEFORE触发器或语句本身失败,MySQL将不执行AFTER触发器。

删除触发器

使用DROP TRIGGER语句;

DROP TRIGGER newproduct;

触发器不能更新或覆盖。为了修改一个触发器,必须先删除它,然后再重新创建。

使用触发器

INSERT触发器

INSERT触发器在INSERT语句执行之前湖之后执行。需要注以下几点:

  • 在INSERT触发器代码内,可引用一个名为NEW的虚拟表,访问被插入的行;
  • 在BEFORE INSERT触发器中,NEW的值也可以被更新(允许更改被插入的值);
  • 对于AUTO_INCREMENT列,NEW在INSERT执行之前包含0,在INSERT执行之后包含新的自动生成值。

如下:

CREATE TRIGGER neworder AFTER INSERT ON orders
FOR EACH ROW SELECT NEW.order_num;

此代码创建一个名为neworder的触发器,它按照AFTER INSERT ON orders执行。在插入一个新订单到orders表时,MySQL生成一个新订单号并保存到order_num中。触发器从NEW.order_num取得这个值并返回它。此触发器必须按照AFTER INSERT执行,因为在BEFORE INSERT语句执行之前,新order_num还没有生成。对于orders的每次插入使用这个触发器将总是返回新的订单号。

测试这个触发器,试着插入新行,如下所示:

INSERT INTO orders(order_date,cust_id)
VALUES(Now(),10001);

在插入完成后,将自动返回新的order_num。

BEFORE或AFTER:通常将BEFORE用于数据验证和净化(目的是保证插入表中的数据确实是需要的数据)。

DELETE触发器

  • 在DELETE触发器代码内,可以引用一个名为OLD的虚拟表,访问被删除的行。
  • OLD中的值全都是只读的,不能更新
CREATE TRIGGER deleteorder BEFORE DELETE ON orders
FOR EACH ROW
BEGIN
    INSERT INTO archive_orders(order_num,order_date,cust_id)
    VALUES(OLD.order_num,OLD.order_date,OLD.cust_id);
END;

在任意订单被删除前将执行此触发器。它使用一条INSERT语句将OLD中的值(要被删除的订单)保存到一个名为archive_orders的存档表中。

使用BEFORE DELETE触发器的优点(相对于AFTER DELETE触发器来说)为,如果由于某种原因,订单不能存档,DELETE本身将被抛弃。

UPDATE触发器

  • 在UPDATE触发器代码中,可以引用一个名为OLD的虚拟表访问以前(UPDATE语句前)的值,引用一个名为NEW的虚拟表访问新更新的值
  • 在BEFORE UPDATE触发器中,NEW中的值也能被更新(允许更改将要用于UPDATE语句中的值)
  • OLD中的值全都是只读的,不能更细。
CREATE TRIGGER updatevendor BEFORE UPDATE ON vendors
FOR EACH ROW SET NEW.vend_state = Upper(NEW.vend_state);

显然,任何数据净化都需要在UPDATE语句之前进行。每次更新一个行时,NEW.vend_state中的值都用Upper(NEW.vend_state)替换。

关于触发器

  • 创建触发器可能需要特殊的安全访问权限,但是触发器的执行是自动的。如果INSERT、UPDATE和DELETE语句能够执行,则相关的触发器也能执行。
  • 应该用触发器来保证数据的一致性(大小写、格式等)。在触发器中执行这种类型的处理的优点是它总是进行这种处理,而且是透明地进行,与客户无关。
  • 触发器的一种非常有意义的使用时创建审计跟踪。使用触发器把更改记录到另一表中非常容易。
  • MySQL触发器不支持CALL语句。这表示不能从触发器内调用存储过程。所需的存储过程需要复制到触发器内。

管理事务处理

事务处理

用来维护数据库的完整性,它保证成批的MySQL操作要么完全执行,要么完全不执行。

利用事务处理,可以保证一组操作不会中途停止,它们或者作为整体执行,或者完全不执行。如果没有错误发生,整组语句提交到(写到)数据库表。如果发生错误,则进行回退(撤销)以恢复数据库到某个已知且安全的状态。

事务(transaction):指一组SQL语句

回退(rollback):指撤销指定的SQL语句的过程

提交(commit):指将未存储的SQL语句结果写入数据库表

保留点(savepoint):指事务处理中设置的临时占位符,可以对它发布回退(与回退整个事务处理不同)。

控制事务处理

MySQL使用下面的语句来标识事务的开始:

START TRANSACTION

使用ROLLBACK:MySQL的ROLLBACK命令用来回退(撤销)MySQL语句,如下;

SELECT * FROM ordertotals;
START TRANSACTION;
DELETE FROM ordertotals;
SELECT * FROM ordertotals;
ROLLBACK;
SELECT * FROM ordertotals;

首先执行一条SELECT语句显示该表不为空。然后开始一个事务处理,用一条DELETE语句删除ordertotals中的所有行。另一条SELECT语句验证ordertotals确实为空。这是用一条ROLLBACK语句回退START TRANSACTION之后的所有语句,最后一条SELECT用户显示该表不为空。

显然,ROLLBACK只能在一个事务处理内使用(在执行START TRANSACTION命令之后)。

哪些语句可以回退:事务处理用来管理INSERT、UPDATE和DELETE语句。不能回退SELECT、CREATE或DROP语句。事务处理块中可以使用这两条语句,但如果执行回退,不会被撤销。

使用COMMIT

一般的MySQL语句都是直接针对数据库表执行和编写的,这就是所谓的隐含提交(implicit commit),即提交(写或保存)操作是自动进行的。

但是,在事务处理块中,提交不会隐含地进行。为进行明确的提交,使用COMMIT语句:

START TRANSACTION;
DELETE FROM orderitems WHERE order_num = 20010;
DELETE FROM orders WHERE order_num = 20010;
COMMIT;

最后的COMMIT语句仅在不出错时写出更改。如果第一条DELETE起作用,但第二条失败,则DELETE不会提交(实际上,会自动撤销)。

隐含事务关闭:当COMMIT或ROLLBACK语句执行后,事务会自动关闭(将来的更改会隐含提交)。

使用保留点

为了支持回退部分事务处理,必须能在事务处理块中合适的位置放置占位符。这样,如果需要回退,可以回退到某个占位符。这些占位符称为保留点。可以使用SAVEPOINT语句创建;

SAVEPOINT delete1;

每个保留点都取标识它的唯一名字,以便在回退时,MySQL知道要回退到何处。为了回退到本例给出的保留点,可如下进行:

ROLLBACK TO delete1;

释放保留点:保留点在事务处理完成(执行一条ROLLBACK或COMMIT)后自动释放。也可以用RELEASE SAVEPOINT明确地释放保留点。

更改默认的提交行为

默认的MySQL行为是自动提交所有更改。为指示MySQL不自动提交更改,需要使用以下语句;l

SET autocommit=0;

autocommit标志决定是否自动提交更改,不管有没有COMMIT语句。设置autocommit=0指示MySQL不自动提交更改(直到autocommit被设置为真位置)。

autocommit标志是针对每个联结而不是服务器。