【转】关于用不用外键

内容纲要

今天听了一个企业技术总监的宣讲,结果听说在他开发系统的过程中,都没有用到外键,这让我很惊讶,赶紧上网搜索了一些资料看了看,终于明白了不用外键的原因。

这是一篇关于是否使用外键的讨论,讲的很有道理:

对于主/外键/索引来说,在一些开发团队中被认为是处理数据库关系的利器,也被某些开发团队认为是处理某些具体业务的魔鬼,您的观点呢?在实际应用中您会采取哪种方式?

大家共同观点:

主键和索引是不可少的,不仅可以优化数据检索速度,开发人员还省不其它的工作,

矛盾焦点:数据库设计是否需要外键。这里有两个问题:一个是如何保证数据库数据的完整性和一致性;二是第一条对性能的影响。

正方观点:

1,由数据库自身保证数据一致性,完整性,更可靠,因为程序很难100%保证数据的完整性,而用外键即使在数据库服务器当机或者出现其他问题的时候,也能够最大限度的保证数据的一致性和完整性。

eg:数据库和应用是一对多的关系,A应用会维护他那部分数据的完整性,系统一变大时,增加了B应用,A和B两个应用也许是不同的开发团队来做的。他们如何协调保证数据的完整性,而且一年以后如果又增加了C应用呢?

2,有主外键的数据库设计可以增加ER图的可读性,这点在数据库设计时非常重要。

3,外键在一定程度上说明的业务逻辑,会使设计周到具体全面。

反方观点:

1,可以用触发器或应用程序保证数据的完整性

2,过分强调或者说使用主键/外键会平添开发难度,导致表过多等问题

3,不用外键时数据管理简单,操作方便,性能高(导入导出等操作,在insert,  update,   delete  数据的时候更快)

eg:在海量的数据库中想都不要去想外键,试想,一个程序每天要insert数百万条记录,当存在外键约束的时候,每次要去扫描此记录是否合格,一般还不止一个字段有外键,这样扫描的数量是成级数的增长!我的一个程序入库在3个小时做完,如果加上外键,需要28个小时! 

结论:

1,在大型系统中(性能要求不高,安全要求高),使用外键;在大型系统中(性能要求高,安全自己控制),不用外键;小系统随便,最好用外键。

2,用外键要适当,不能过分追求

3,不用外键而用程序控制数据一致性和完整性时,应该写一层来保证,然后个个应用通过这个层来访问数据库。

文章出处:http://www.diybl.com/course/7_databases/database_other/200838/103356.html

上面提到了触发器的概念,我这里用MYSQL举例

触发器的概念:“在数据库中为响应一个特殊表格中的某些事件而自动执行的程序代码。”(Wikipedia)说得简单一些,它是在一个特殊的数据库事件,如INSERTDELETE发生时,自动激活的一段代码。触发器可方便地用于日志记录、对单个表格到其他链接式表格进行自动的“层叠式”更改、或保证对表格关系进行自动更新。当一个新整数值增加到数据库域中时,自动更新运行的总数的代码段是一个触发器。自动记录对一个特殊数据库表格所作更改的SQL命令块也是一个触发器实例。

触发器是MySQL 5.x的新功能,随着5.x代码树新版本的出现,这一功能也逐渐得到改善。在本文中,我将简单介绍如何定义并使用触发器,查看触发器状态,并如何在使用完毕后删除触发器。我还将为你展示一个触发器在现实世界中的应用实例,并检验它对数据库记录的改变。

例子

通过简单(虽然是人为的)实例来说明是了解MySQL触发器应用的最佳办法。首先我们建立两个单域的表格。一个表格中为姓名列表(表格名:data),另一个表格中是所插入字符的字符数(表格名:chars)。我希望在data表格中定义一个触发器,每次在其中插入一个新姓名时,chars表格中运行的总数就会根据新插入记录的字符数目进行自动更新。

以下为引用的内容:

mysql> CREATE TABLE data (nameVARCHAR(255));

Query OK, 0 rows affected (0.09 sec)

mysql> CREATE TABLE chars (countINT(10));

Query OK, 0 rows affected (0.07 sec)

mysql> INSERT INTO chars (count) VALUES(0);

Query OK, 1 row affected (0.00 sec)

mysql> CREATE TRIGGER t1 AFTER INSERT ON

data FOR EACH ROW UPDATE chars SET count

= count + CHAR_LENGTH(NEW.name);

Query OK, 0 rows affected (0.01 sec)

理解上面代码的关键在于CREATETRIGGER命令,它被用来定义一个新触发器。这个命令建立一个新触发器,假定的名称为t1,每次有一个新记录插入到data表格中时,t1就被激活。

在这个触发器中有两个重要的子句:

AFTER INSERT子句表明触发器在新记录插入data表格后激活。

UPDATE chars SET count = count +CHAR_LENGTH(NEW.name)子句表示触发器激活后执行的SQL命令。在本例中,该命令表明用新插入的data.name域的字符数来更新chars.count栏。这一信息可通过内置的MySQL函数CHAR_LENGTH()获得。

放在源表格域名前面的NEW关键字也值得注意。这个关键字表明触发器应考虑域的new值(也就是说,刚被插入到域中的值)。MySQL还支持相应的OLD前缀,可用它来指域以前的值。

你可以通过调用SHOWTRIGGER命令来检查触发器是否被激活。

以下为引用的内容:

mysql> SHOW TRIGGERSG

*************************** 1. row******************

?Trigger: t1

?Event: INSERT

?Table: data

Statement: UPDATE chars SET count = count +CHAR_LENGTH(NEW.name)

Timing: AFTER

?Created: NULL

ql_mode:

1 row in set (0.01 sec)

激活触发器后,开始对它进行测试。试着在data表格中插入几个记录:

以下为引用的内容:

mysql> INSERT INTO data (name) VALUES('Sue'), ('Jane');

Query OK, 2 rows affected (0.00 sec)

Records: 2?Duplicates: 0?Warnings: 0

然后检查chars表格看触发器是否完成它该完成的任务:

以下为引用的内容:

mysql> SELECT * FROM chars;

+-------+

| count |

+-------+

| 7|

+-------+

1 row in set (0.00 sec)

如你所见,data表格中的INSERT命令激活触发器,它计算插入记录的字符数,并将结果存储在chars表格中。如果你往data表格中增加另外的记录,chars.count值也会相应增加。

触发器应用完毕后,可有DROPTRIGGER命令轻松删除它。

以下为引用的内容:

mysql> DROP TRIGGER t1;

Query OK, 0 rows affected (0.00 sec)

注意:理想情况下,你还需要一个倒转触发器,每当一个记录从源表格中删除时,它从字符总数中减去记录的字符数。这很容易做到,你可以把它当作练习来完成。提示:应用BEFOREDELETE ON子句是其中一种方法。

现在,我想建立一个审计记录来追踪对这个表格所做的改变。这个记录将反映表格的每项改变,并向用户说明由谁做出改变以及改变的时间。我需要建立一个新表格来存储这一信息(表格名:audit),如下所示。

以下为引用的内容:

mysql> CREATE TABLE audit (id INT(7),

 balance FLOAT, user VARCHAR(50)

NOT NULL, time TIMESTAMP NOT NULL);

Query OK, 0 rows affected (0.09 sec)

接下来,我将在accounts表格中定义一个触发器。

以下为引用的内容:

mysql> CREATE TRIGGER t1 AFTER UPDATEONaccounts

FOR EACH ROW INSERT INTO audit (id,balance, user, time)

VALUES (OLD.id, NEW.balance,CURRENT_USER(), NOW());

Query OK, 0 rows affected (0.04 sec)

注释:accounts表格每经历一次UPDATE,触发器插入(INSERT)对应记录的id、新的余额、当前时间和登录audit表格的用户的名称。

每次我想要演示实际代码时,我会对mysql客户端的屏幕就出现的代码进行调整,将字体改成Courier,使他们看起来与普通文本不一样(让大家区别程序代码和正文)。在这里举个例子:

 

mysql> DROP FUNCTION f;

Query OK, 0 rows affected (0.00 sec)

  如果实例比较大,则需要在某些行和段落间加注释,同时我会用将"<--"符号放在页面的右边以表示强调。例如:

 

mysql> CREATE PROCEDURE p ()

-> BEGIN

-> /* This procedure does nothing */<--

-> END;//

Query OK, 0 rows affected (0.00 sec)

  有时候我会将例子中的"mysql>""->"这些系统显示去掉,你可以直接将代码复制到mysql客户端程序中(如果你现在所读的不是电子版的,可以在mysql.com网站下载相关脚本)

  所以的例子都已经在Suse 9.2LinuxMysql 5.0.3公共版上测试通过。在您阅读本书的时候,Mysql已经有更高的版本,同时能支持更多OS了,包括WindowsSparcHP-UX。因此这里的例子将能正常的运行在您的电脑上。但如果运行仍然出现故障,可以咨询你认识的资深Mysql用户,这样就能得到比较好的支持和帮助。

Why Triggers 为什么要用触发器

  我们在MySQL 5.0中包含对触发器的支持是由于以下原因:

MySQL早期版本的用户长期有需要触发器的要求。

  我们曾经许诺支持所有ANSI标准的特性。

  您可以使用它来检查或预防坏的数据进入数据库。

  您可以改变或者取消INSERT, UPDATE以及DELETE语句。

  您可以在一个会话中监视数据改变的动作。

  在这里我假定大家都读过"MySQL新特性"丛书的第一集--"MySQL存储过程",那么大家都应该知道MySQL至此存储过程和函数,那是很重要的知识,因为在触发器中你可以使用在函数中使用的语句。特别举个例子:

  复合语句(BEGIN /END)是合法的.

  流控制(Flow-of-control)语句(IF, CASE, WHILE, LOOP, WHILE, REPEAT,LEAVE,ITERATE)也是合法的.

  变量声明(DECLARE)以及指派(SET)是合法的.

  允许条件声明.

  异常处理声明也是允许的.

  但是在这里要记住函数有受限条件:不能在函数中访问表.因此在函数中使用以下语句是非法的。

 

ALTER 'CACHE INDEX' CALL COMMIT CREATEDELETE

DROP 'FLUSH PRIVILEGES' GRANT INSERT KILL

LOCK OPTIMIZE REPAIR REPLACE REVOKE

ROLLBACK SAVEPOINT 'SELECT FROM table'

'SET system variable' 'SET TRANSACTION'

SHOW 'START TRANSACTION' TRUNCATE UPDATE

  在触发器中也有完全一样的限制.

  触发器相对而言比较新,因此会有(bugs)缺陷.所以我在这里给大家警告,就像我在存储过程书中所说那样.不要在含有重要数据的数据库中使用这个触发器,如果需要的话在一些以测试为目的的数据库上使用,同时在你对表创建触发器时确认这些数据库是默认的。

Syntax 语法

1. Syntax: Name 语法:命名规则

 

CREATE TRIGGER <触发器名称><--

{ BEFORE | AFTER }

{ INSERT | UPDATE | DELETE }

ON <表名称>

FOR EACH ROW

<触发器SQL语句>

  触发器必须有名字,最多64个字符,可能后面会附有分隔符.它和MySQL中其他对象的命名方式基本相象.

  这里我有个习惯:就是用表的名字+'_'+触发器类型的缩写.因此如果是表t26,触发器是在事件UPDATE(参考下面的点(2)和(3))之前(BEFORE)的,那么它的名字就是t26_bu

 

2. Syntax: Time 语法:触发时间

 

CREATE TRIGGER <触发器名称>

{ BEFORE | AFTER } <--

{ INSERT | UPDATE | DELETE }

ON <表名称>

FOR EACH ROW

<触发的SQL语句>

  触发器有执行的时间设置:可以设置为事件发生前或后。

3. Syntax: Event语法:事件

 

CREATE TRIGGER <触发器名称>

{ BEFORE | AFTER }

{ INSERT | UPDATE | DELETE } <--

ON <表名称>

FOR EACH ROW

<触发的SQL语句>

  同样也能设定触发的事件:它们可以在执行insertupdatedelete的过程中触发。

4. Syntax: Table 语法:表

 

CREATE TRIGGER <触发器名称>

{ BEFORE | AFTER }

{ INSERT | UPDATE | DELETE }

ON <表名称> <--

FOR EACH ROW

<触发的SQL语句>

  触发器是属于某一个表的:当在这个表上执行插入、更新或删除操作的时候就导致触发器的激活.

我们不能给同一张表的同一个事件安排两个触发器。

5. Syntax: Granularity语法:(:( 步长)触发间隔

 

CREATE TRIGGER <触发器名称>

{ BEFORE | AFTER }

{ INSERT | UPDATE | DELETE }

ON <表名称>

FOR EACH ROW <--

<触发的SQL语句>

  触发器的执行间隔:FOR EACHROW子句通知触发器每隔一行执行一次动作,而不是对整个表执行一次。

6. Syntax: Statement 语法:语句

 

CREATE TRIGGER <触发器名称>

{ BEFORE | AFTER }

{ INSERT | UPDATE | DELETE }

ON <表名称>

FOR EACH ROW

<触发的SQL语句> <--

  触发器包含所要触发的SQL语句:这里的语句可以是任何合法的语句,包括复合语句,但是这里的语句受的限制和函数的一样。

Privileges权限

  你必须拥有相当大的权限才能创建触发器(CREATETRIGGER)。如果你已经是Root用户,那么就足够了。这跟SQL的标准有所不同,我也希望能尽快改成标准的。

  因此在下一个版本的MySQL中,你完全有可能看到有一种叫做CREATE TRIGGER的新权限。然后通过这样的方法赋予:

 

GRANT CREATE TRIGGER ON <表名称> TO<用户或用户列表>;

  也可以通过这样收回权限:

 

REVOKE CREATE TRIGGER ON <表名称> FROM<用户或用户列表>;

Referring to OLD and NEW columns 关于旧的和新创建的列的标识

  在触发器的SQL语句中,你可以关联表中的任意列。但你不能仅使用列的名称去标识,那会使系统混淆,因为那里可能会有列的新名(这可能正是你要修改的,你的动作可能正是要修改列名),还有列的旧名存在。因此你必须用这样的语法来标识:

"NEW .column_name"或者"OLD.column_name".这样在技术上处理(NEW | OLD .column_name)新和旧的列名属于创建了过渡变量("transitionvariables")。

  对于INSERT语句,只有NEW是合法的;对于DELETE语句,只有OLD才合法;而UPDATE语句可以在和NEW以及OLD同时使用。下面是一个UPDATE中同时使用NEWOLD的例子。

 

CREATE TRIGGER t21_au

BEFORE UPDATE ON t22

FOR EACH ROW

BEGIN

SET @old = OLD . s1;

SET @new = NEW.s1;

END;//

  现在如果t21表中的s1列的值是55,那么执行了"UPDATE t21 SET s1 = s1 + 1"之后@old的值会变成55,而@new的值将会变成56

Example of CREATE and INSERT CREATEINSERT的例子

CREATE table withtrigger创建有触发器的表

  这里所有的例程中我都假定大家的分隔符已经设置成//DELIMITER //)。

 

CREATE TABLE t22 (s1 INTEGER)//

CREATE TRIGGER t22_bi

BEFORE INSERT ON t22

FOR EACH ROW

BEGIN

SET @x = 'Trigger was activated!';

SET NEW.s1 = 55;

END;//

  在最开始我创建了一个名字为t22的表,然后在表t22上创建了一个触发器t22_bi,当我们要向表中的行插入时,触发器就会被激活,执行将s1列的值改为55的动作。

INSERT on table w itha trigger使用触发器执行插入动作

 

mysql> INSERT INTO t22 VALUES (1)//

  让我们看如果向表t2中插入一行数据触发器对应的表会怎么样?

  这里的插入的动作是很常见的,我们不需要触发器的权限来执行它。甚至不需要知道是否有触发器关联。

 

mysql> SELECT @x, t22.* FROM t22//

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

| @x | s1 |

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

| Trigger was activated! | 55 |

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

1 row in set (0.00 sec)

  大家可以看到INSERT动作之后的结果,和我们预期的一样,x标记被改动了,同时这里插入的数据不是我们开始输入的插入数据,而是触发器自己的数据。

Example of a "check" constraint

"check"完整性约束例子

What's a"check" constraint 什么是"check"约束

  在标准的SQL语言中,我们可以在(CREATE TABLE)创建表的过程中使用"CHECK(condition)"

例如:

 

CREATE TABLE t25

(s1 INT, s2 CHAR(5), PRIMARY KEY (s1),

CHECK (LEFT(s2,1)='A'))

ENGINE=INNODB;

  这里CHECK的意思是"s2列的最左边的字符不是'A'时,insertupdate语句都会非法"MySQL的视图不支持CHECK,我个人是很希望它能支持的。但如果你很需要在表中使用这样的功能,我建议大家使用触发器来实现。

 

CREATE TABLE t25

(s1 INT, s2 CHAR(5),

PRIMARY KEY (s1))

ENGINE=INNODB//

CREATE TRIGGER t25_bi

BEFORE INSERT ON t25

FOR EACH ROW

IF LEFT(NEW.s2,1)<>'A' THEN SETNEW.s1=0; END IF;//

CREATE TRIGGER t25_bu

BEFORE UPDATE ON t25

FOR EACH ROW

IF LEFT(NEW.s2,1)<>'A' THEN SETNEW.s1=0; END IF;//

  我只需要使用BEFOREINSERTBEFORE UPDATE语句就行了,删除了触发器不会对表有影响,同时AFTER的触发器也不能修改NEW的过程变量(transitionvariables)。为了激活触发器,我执行了向表中的行插入s10的数据,之后只要执行符合LEFT(s2,1)<> 'A'条件的动作都会失败:

 

INSERT INTO t25 VALUES (0,'a') /* primingthe pump */ //

INSERT INTO t25 VALUES (5,'b') /* gets error'23000' */ //

Don't Believe The Old MySQL Manual

该抛弃旧的MySQL的手册了

  我在这里警告大家不要相信过去的MySQL手册中所说的了。我们已经去掉了关于触发器的错误的语句,但是仍旧有很多旧版本的手册在网上,举个例子,这是一个德国的Url上的:

 

http://dev.mysql.com/doc/mysql/de/ANSI_diff_Triggers.html.

  这个手册上说触发器就是存储过程,忘掉吧,你也已经看见了,触发器就是触发器,而存储过程还是存储过程。

  手册上还说触发器可以从其他表上来删除,或者是当你删除一个事务的时候激发,无论他说的是什么意思,忘掉吧,MySQL不会去实现这些的。

  最后关于说使用触发器会对查询速度产生影响的说法也是错的,触发器不会对查询产生任何影响。

Bugs

  (不好的东西就不翻译了)

On December 14 2004, I did an "AdvancedSearch" inhttp://bugs.mysql.com for 'trigger' or

'triggers', I found that there were 17 activebugs as of that date.Of course they might disappear

before you read this, but just in case theyhaven't, I'll mention theimportant ones. If they're still

there, you'll have to work around them whenyou're trying triggers.

 

Bug#5859 DROP TABLE does not drop triggers.

  (删除表的时候没有自动删除触发器)

When you drop a table, dropping the table'striggers should beautomatic.

 

Bug#5892 Triggers have the wrong namespace.

  (触发器的命名空间有错,你必须在前面加上表的名字才能删除触发器,下面是例子)

You have to say "DROP TRIGGER <tablename> . <triggername>".

The correct way is "DROP TRIGGER<trigger name>".

 

Bug#5894 Triggers with altered tables causecorrupt databases.

  (触发器对表的改变可能会造成数据库数据被破坏)

Do not alter a table that has a trigger on it,until you know this isfixed.

另外,再讲述一下索引的概念

一、索引的概念

索引就是加快检索表中数据的方法。数据库的索引类似于书籍的索引。在书籍中,索引允许用户不必翻阅完整个书就能迅速地找到所需要的信息。在数据库中,索引也允许数据库程序迅速地找到表中的数据,而不必扫描整个数据库。

二、索引的特点

1.索引可以加快数据库的检索速度

2.索引降低了数据库插入、修改、删除等维护任务的速度

3.索引创建在表上,不能创建在视图上

4.索引既可以直接创建,也可以间接创建

5.可以在优化隐藏中,使用索引

6.使用查询处理器执行SQL语句,在一个表上,一次只能使用一个索引

7.其他

三、索引的优点

1.创建唯一性索引,保证数据库表中每一行数据的唯一性

2.大大加快数据的检索速度,这也是创建索引的最主要的原因

3.加速表和表之间的连接,特别是在实现数据的参考完整性方面特别有意义。

4.在使用分组和排序子句进行数据检索时,同样可以显著减少查询中分组和排序的时间。

5.通过使用索引,可以在查询的过程中使用优化隐藏器,提高系统的性能。

四、索引的缺点

1.创建索引和维护索引要耗费时间,这种时间随着数据量的增加而增加

2.索引需要占物理空间,除了数据表占数据空间之外,每一个索引还要占一定的物理空间,如果要建立聚簇索引,那么需要的空间就会更大

3.当对表中的数据进行增加、删除和修改的时候,索引也要动态的维护,降低了数据的维护速度

五、索引分类

1.直接创建索引和间接创建索引

直接创建索引: CREATEINDEX mycolumn_index ON mytable (myclumn)

间接创建索引:定义主键约束或者唯一性键约束,可以间接创建索引

2.普通索引和唯一性索引

普通索引:

CREATE INDEX mycolumn_index ONmytable(myclumn)

唯一性索引:保证在索引列中的全部数据是唯一的,对聚簇索引和非聚簇索引都可以使用

 

CREATE UNIQUE COUSTERED INDEXmyclumn_cindex ON mytable(mycolumn)

3.单个索引和复合索引

单个索引:即非复合索引

复合索引:又叫组合索引,在索引建立语句中同时包含多个字段名,最多16个字段

 

CREATE INDEX name_index ONusername(firstname,lastname)

4.聚簇索引和非聚簇索引(聚集索引,群集索引)

聚簇索引:物理索引,与基表的物理顺序相同,数据值的顺序总是按照顺序排列

CREATE CLUSTERED INDEX mycolumn_cindexONmytable(mycolumn) WITH

ALLOW_DUP_ROW(允许有重复记录的聚簇索引)

非聚簇索引:

 

CREATE UNCLUSTERED INDEX mycolumn_cindex ONmytable(mycolumn)

六、索引的使用

1.当字段数据更新频率较低,查询使用频率较高并且存在大量重复值是建议使用聚簇索引

2.经常同时存取多列,且每列都含有重复值可考虑建立组合索引

3.复合索引的前导列一定好控制好,否则无法起到索引的效果。如果查询时前导列不在查询条件中则该复合索引不会被使用。前导列一定是使用最频繁的列

4.多表操作在被实际执行前,查询优化器会根据连接条件,列出几组可能的连接方案并从中找出系统开销最小的最佳方案。连接条件要充份考虑带有索引的表、行数多的表;内外表的选择可由公式:外层表中的匹配行数*内层表中每一次查找的次数确定,乘积最小为最佳方案

5.where子句中对列的任何操作结果都是在sql运行时逐列计算得到的,因此它不得不进行表搜索,而没有使用该列上面的索引;如果这些结果在查询编译时就能得到,那么就可以被sql优化器优化,使用索引,避免表搜索。

例:

 

select * from record wheresubstring(card_no,1,4)=’5378’

           && select * from record where card_no like ’5378%’

任何对列的操作都将导致表扫描,它包括数据库函数、计算表达式等等,查询时要尽可能将操作移至等号右边

6.where条件中的’in’在逻辑上相当于’or’,所以语法分析器会将in('0','1')转化为column='0'orcolumn='1'来执行。我们期望它会根据每个or子句分别查找,再将结果相加,这样可以利用column上的索引;但实际上它却采用了"or策略",即先取出满足每个or子句的行,存入临时数据库的工作表中,再建立唯一索引以去掉重复行,最后从这个临时表中计算结果。因此,实际过程没有利用column上索引,并且完成时间还要受tempdb数据库性能的影响。inor子句常会使用工作表,使索引失效;如果不产生大量重复值,可以考虑把子句拆开;拆开的子句中应该包含索引

7.要善于使用存储过程,它使sql变得更加灵活和高效

最后是视图,再面向接口编程的思想和尽可能重用的思想,我们需要借助视图,这在李老师语录中我就写到,所以了解视图时很必要的.

http://book.51cto.com/art/200710/57708.htm

存储过程: http://book.51cto.com/art/200710/57711.htm

类似于JDBC的预处理语句,可以填充参数决定执行的结果。一向技术提出,总有解决某方便的问题才会被使用,所以为什么要用存储过程?

由于存储过程对于MySQL来说是新的功能,很自然的在使用时你需要更加注意。

  毕竟,在此之前没有任何人使用过,也没有很多大量的有经验的用户来带你走他们走过的路。然而你应该开始考虑把现有程序(可能在服务器应用程序中,用户自定义函数(UDF)中,或是脚本中)转移到存储过程中来。这样做不需要原因,你不得不去做。

 

  因为存储过程是已经被认证的技术!虽然在Mysql中它是新的,但是相同功能的函数在其他DBMS中早已存在,而它们的语法往是相同的。因此你可以从其他人那里获得这些概念,也有很多你可以咨询或者雇用的经验用户,还有许多第三方的文档可供你阅读。

 

  存储过程会使系统运行更快!虽然我们暂时不能在Mysql上证明这个优势,用户得到的体验也不一样。我们可以说的就是Mysql服务器在缓存机制上做了改进,就像Preparedstatements(预处理语句)所做的那样。由于没有编译器,因此SQL存储过程不会像外部语言(如C)编写的程序运行起来那么快。但是提升速度的主要方法却在于能否降低网络信息流量。如果你需要处理的是需要检查、循环、多语句但没有用户交互的重复性任务,你就可以使用保存在服务器上的存储过程来完成。这样在执行任务的每一步时服务器和客户端之间就没那么多的信息来往了。

所以存储过程是可复用的组件!想象一下如果你改变了主机的语言,这对存储过程不会产生影响,因为它是数据库逻辑而不是应用程序。存储过程是可以移植的!当你用SQL编写存储过程时,你就知道它可以运行在Mysql支持的任何平台上,不需要你额外添加运行环境包,也不需要为程序在操作系统中执行设置许可,或者为你的不同型号的电脑存储过程将被保存!如果你编写好了一个程序,例如显示银行事物处理中的支票撤消,那想要了解支票的人就可以找到你的程序。

 

  它会以源代码的形式保存在数据库中。这将使数据和处理数据的进程有意义的关联这可能跟你在课上听到的规划论中说的一样。存储过程可以迁移!

 

Mysql完全支持SQL 2003标准。某些数据库(如DB2Mimer)同样支持。但也有部分不支持的,如OracleSQL Server不支持。我们将会给予足够帮助和工具,使为其他DBMS编写的代码能更容易转移到Mysql上。

其他存储过程的链接:http://www.zhiweinet.com/jiaocheng/2009-05/10342.htm

http://tech.163.com/05/0906/16/1SVT978N00091589.html

http://www.jb51.net/article/14070.htm

这里记录存储过程的常用方法和语法

delimiter //

create procedure 过程名(in idint,out f varchar(20))

begin

end;//

call 过程名(参数列表);

 

CREATE PROCEDURE p6 (OUT p INT)

SET p = -5

CALL p6(@y)//

SELECT @y//

if .. then

...

else

...

end if

while .. do

...

end while

repeat

... until...

end repeat

declare a int [default 8];

set a = 1;

set @a = 1;

BEGIN END  == {}

CASE 变量

WHEN 数值1 THEN ...;

WHEN 数值2 THEN ...;

ELSE ...;

END CASE;

标号:

leave 标号;

end 标号;

DECLARE

{ EXIT | CONTINUE }

HANDLER FOR

{ error-number | { SQLSTATE error-string }| condition }

SQL statement

START TRANSACTION;

COMMIT;

ROLLBACK;

DECLARE cursor-name CURSOR FOR SELECT ...;

OPEN cursor-name;

FETCH cursor-name INTO variable [,variable];

CLOSE cursor-name;

(例子:

CREATE PROCEDURE p25 (OUT return_val INT)

BEGIN

DECLARE a,b INT;

DECLARE cur_1 CURSOR FOR SELECT s1 FROM t;

DECLARE CONTINUE HANDLER FOR NOT FOUND

SET b = 1;

OPEN cur_1;

REPEAT

FETCH cur_1 INTO a; <--

UNTIL b = 1

END REPEAT;

CLOSE cur_1;

SET return_val = a;

END;//

)

函数🙁例子)

CREATE FUNCTION factorial (n DECIMAL(3,0))RETURNSDECIMAL(20,0)

DETERMINISTIC

BEGIN

DECLARE factorial DECIMAL(20,0) DEFAULT 1;

DECLARE counter DECIMAL(3,0);

SET counter = n;

factorial_loop: REPEAT

SET factorial = factorial * counter;

SET counter = counter - 1;

UNTIL counter = 1

END REPEAT;

RETURN factorial;

END //

selectlinuxecho类似,相当于显示**的结果,后面可以跟查询语句,可以跟函数等等

下面是测试SQL程序

drop database if exists testview;

create database testview;

use testview;

drop table if exists student;

drop table if exists teacher;

create table student(id varchar(20) primarykey, namevarchar(20), teacherId varchar(20));

create table teacher(teacherId varchar(20),teacherNamevarchar(20));

create view getTeacherName as selects.name, t.teacherNamefrom student as s, teacher as t

where s.teacherId = t.teacherId;

insert into student values('1', 'student','1');

insert into teacher values('1', 'teacher');

select name, teacherName fromgetTeacherName;

delimiter //

create trigger updateTrigger after updateon teacher foreach row

begin

IF teacherId = OLD.teacherId THEN

update student set teacherId =NEW.teacherId;

END IF;

end;//

update teacher set teacherId ='2' whereteacherId= '1';

create procedure save(in id varchar(20), innamevarchar(20), in teacherId varchar(20), in

teacherName varchar(20))

begin

 insert into student values(id, name,teacherId);

 insert into teacher values(teacherId,teacherName);

end;//

call save('3', 'student name','teacherid','teachername');

select * from getTeacherName;

quit

One thought on “【转】关于用不用外键”

发表回复