0、 我的MYSQL学习心得(四) 数据类型(系列文章)
1、MySql数据库学习--存储过程(1)
0、[转]MySQL存储过程调试工具-dbForge Studio for MySQL
①、存储过程的优缺点
1 为什么要用存储过程 2 3 几个去 IBM 面试的兄弟回来抱怨:去了好几个不同的 IBM 项目组,几乎每个面试官问到数据库的时候都要问用没用过存储过程,烦人不?大家去面的程序员,又不是 DBA,以前的项目都没有用到存储,不照样运行的好好的? 4 5 存储过程真的那么重要吗,它到底有什么好处呢? 6 7 笔者认为,存储过程说白了就是一堆 SQL 的合并。中间加了点逻辑控制。 8 9 但是存储过程处理比较复杂的业务时比较实用。 10 比如说,一个复杂的数据操作。如果你在前台处理的话。可能会涉及到多次数据库连接。但如果你用存储过程的话。就只有一次。从响应时间上来说有优势。 11 也就是说存储过程可以给我们带来运行效率提高的好处。 12 另外,程序容易出现 BUG 不稳定,而存储过程,只要数据库不出现问题,基本上是不会出现什么问题的。也就是说从安全上讲,使用了存储过程的系统更加稳定。 13 14 数据量小的,或者和钱没关系的项目不用存储过程也可以正常运作。mysql 的存储过程还有待实际测试。如果是正式项目,建议你用 sql server 或 oracle 的存储过程。数据与数据之间打交道的话,过程会比程序来的快的多。面试官问有没有用存储,实际上就是想知道前来面试的程序员到底做过数据量大的项目没。如果是培训出来的,或者小项目小公司出来的,对存储肯定接触的少了。 15 16 所以,要想进大公司,没有丰富存储过程经验,是不行的。 17 18 那么什么时候才可以用存储?对于数据量不是很大以及业务处理不是很复杂的小项目就无需要了么? 19 错。存储过程不仅仅适用于大型项目,对于中小型项目,使用存储过程也是非常有必要的。其威力和优势主要体现在: 20 1.存储过程只在创造时进行编译,以后每次执行存储过程都不需再重新编译,而一般 SQL 语句每执行一次就编译一次,所以使用存储过程可提高数据库执行速度。 21 2.当对数据库进行复杂操作时(如对多个表进行 Update,Insert,Query,Delete 时),可将此复杂操作用存储过程封装起来与数据库提供的事务处理结合一起使用。这些操作,如果用程序来完成,就变成了一条条的 SQL 语句,可能要多次连接数据库。而换成存储,只需要连接一次数据库就可以了。 22 3.存储过程可以重复使用,可减少数据库开发人员的工作量。 23 4.安全性高,可设定只有某此用户才具有对指定存储过程的使用权。 24 25 26 本文来自CSDN博客,转载请标明出处:http://blog.csdn.net/defonds/archive/2009/07/15/4349922.aspx 27 28 存储过程的使用,好像一直是一个争论。 29 我不倾向于尽可能使用存储过程,是这么认为的: 30 1. 运行速度: 大多数高级的数据库系统都有statement cache的,所以编译sql的花费没什么影响。但是执行存储过程要比直接执行sql花费更多(检查权限等),所以对于很简单的sql,存储过程没有什么优势。 31 2. 网络负荷:如果在存储过程中没有多次数据交互,那么实际上网络传输量和直接sql是一样的。 32 3. 团队开发:很遗憾,比起成熟的IDE,没有什么很好存储过程的IDE工具来支持,也就是说,这些必须手工完成。 33 4. 安全机制:对于传统的C/S结构,连接数据库的用户可以不同,所以安全机制有用;但是在web的三层架构中,数据库用户不是给用户用的,所以基本上,只有一个用户,拥有所有权限(最多还有一个开发用户)。这个时候,安全机制有点多余。 34 5. 用户满意:实际上这个只是要将访问数据库的接口统一,是用存储过程,还是EJB,没太大关系,也就是说,在三层结构中,单独设计出一个数据访问层,同样能实现这个目标。 35 6. 开发调试:一样由于IDE的问题,存储过程的开发调试要比一般程序困难(老版本DB2还只能用C写存储过程,更是一个灾难)。 36 7. 移植性:算了,这个不用提,反正一般的应用总是绑定某个数据库的,不然就无法靠优化数据库访问来提高性能了。 37 8. 维护性:的确,存储过程有些时候比程序容易维护,这是因为可以实时更新DB端的存储过程,但是在3层结构下,更新server端的数据访问层一样能实现这个目标,可惜现在很多平台不支持实时更新而已。 38 39 从上面可知道,存储过程的使用不能有死规定(全用,或全不用),以前Terminal - Server, Client-DB的方式已经过时了,存储过程很多优势已经不明显。 40 现在,我认为的原则是:所有数据访问在应用层封装为数据访问层,在那里,如果SQL简单的话,直接用SQL;如果SQL复杂,或者数据交互多且中间数据最后不会用到,使用存储过程。其他凭经验吧。 41 42 本文来自CSDN博客,转载请标明出处:http://blog.csdn.net/zy1691/archive/2009/01/09/3742780.aspx 43 44 存储过程是由一些SQL语句和控制语句组成的被封装起来的过程,它驻留在数据库中,可以被客户应用程序调用,也可以从另一个过程或触发器调用。它的参数可以被传递和返回。与应用程序中的函数过程类似,存储过程可以通过名字来调用,而且它们同样有输入参数和输出参数。 45 46 根据返回值类型的不同,我们可以将存储过程分为三类:返回记录集的存储过程,返回数值的存储过程(也可以称为标量存储过程),以及行为存储过程。顾名思义,返回记录集的存储过程的执行结果是一个记录集,典型的例子是从数据库中检索出符合某一个或几个条件的记录;返回数值的存储过程执行完以后返回一个值,例如在数据库中执行一个有返回值的函数或命令;最后,行为存储过程仅仅是用来实现数据库的某个功能,而没有返回值,例如在数据库中的更新和删除操作。 47 48 使用存储过程的好处 49 50 相对于直接使用SQL语句,在应用程序中直接调用存储过程有以下好处: 51 52 (1)减少网络通信量。调用一个行数不多的存储过程与直接调用SQL语句的网络通信量可能不会有很大的差别,可是如果存储过程包含上百行SQL语句,那么其性能绝对比一条一条的调用SQL语句要高得多。 53 54 (2)执行速度更快。有两个原因:首先,在存储过程创建的时候,数据库已经对其进行了一次解析和优化。其次,存储过程一旦执行,在内存中就会保留一份这个存储过程,这样下次再执行同样的存储过程时,可以从内存中直接调用。 55 56 (3)更强的适应性:由于存储过程对数据库的访问是通过存储过程来进行的,因此数据库开发人员可以在不改动存储过程接口的情况下对数据库进行任何改动,而这些改动不会对应用程序造成影响。 57 58 (4) 布式工作:应用程序和数据库的编码工作可以分别独立进行,而不会相互压制。 59 60 优点: 61 1.由于应用程序随着时间推移会不断更改,增删功能,T-SQL过程代码会变得更复杂,StoredProcedure为封装此代码提供了一个替换位置。 62 63 2.执行计划(存储过程在首次运行时将被编译,这将产生一个执行计划-- 实际上是 Microsoft SQL Server为在存储过程中获取由 T-SQL 指定的结果而必须采取的步骤的记录。)缓存改善性能。 64 ........但sql server新版本,执行计划已针对所有 T-SQL 批处理进行了缓存,而不管它们是否在存储过程中,所以没比较优势了。 65 66 3.存储过程可以用于降低网络流量,存储过程代码直接存储于数据库中,所以不会产生大量T-sql语句的代码流量。 67 68 4.使用存储过程使您能够增强对执行计划的重复使用,由此可以通过使用远程过程调用 (RPC) 处理服务器上的存储过程而提高性能。RPC 封装参数和调用服务器端过程的方式使引擎能够轻松地找到匹配的执行计划,并只需插入更新的参数值。 69 70 5.可维护性高,更新存储过程通常比更改、测试以及重新部署程序集需要较少的时间和精力。 71 72 6.代码精简一致,一个存储过程可以用于应用程序代码的不同位置。 73 74 7.更好的版本控制,通过使用 Microsoft Visual SourceSafe 或某个其他源代码控制工具,您可以轻松地恢复到或引用旧版本的存储过程。 75 76 8.增强安全性: 77 a、通过向用户授予对存储过程(而不是基于表)的访问权限,它们可以提供对特定数据的访问; 78 b、提高代码安全,防止 SQL注入(但未彻底解决,例如,将数据操作语言--DML,附加到输入参数); 79 c、SqlParameter 类指定存储过程参数的数据类型,作为深层次防御性策略的一部分,可以验证用户提供的值类型(但也不是万无一失,还是应该传递至数据库前得到附加验证)。 80 81 82 83 缺点: 84 85 1.如果更改范围大到需要对输入存储过程的参数进行更改,或者要更改由其返回的数据,则您仍需要更新程序集中的代码以添加参数、更新 GetValue() 调用,等等,这时候估计比较繁琐了。 86 87 2.可移植性差 88 89 由于存储过程将应用程序绑定到 SQL Server,因此使用存储过程封装业务逻辑将限制应用程序的可移植性。如果应用程序的可移植性在您的环境中非常重要,则将业务逻辑封装在不特定于 RDBMS 的中间层中可能是一个更佳的选择。 90 91 92 区别一,存储过程保存在数据库里面,存储过程可以被连接此数据库的所有程序设计语言和程序使用,自定义函数不能。 93 94 区别二,存储过程可以有数据库管理软件修改,使得多层结构程序调整系统逻辑时,并不需要编译和分发程序。 95 96 区别三,存储过程执行中,不会引起网络流量,不占用程序服务器的内存和CPU资源。
②、mysql游标循环的使用
③、mysql游标使用
1 mysql被oracle收购后,从mysql-5.5开始,将InnoDB作为默认存储引擎,是一次比较重大的突破。InnoDB作为支持事务的存储引擎,拥有相关的 2 RDBMS特性:包括ACID事务支持,数据完整性(外健),灾难恢复能力等特性。 3 使用mysql做为数据库的话,将来程序员肯定要写很多,存储过程,function等。在写些东东的时候,游标肯定是少不了的。下面简单简介一下。 4 一,什么是游标(cursor) 5 个人觉得就是一个cursor,就是一个标识,用来标识数据取到什么地方了。你也可以把它理解成数组中的下标。 6 二,游标(cursor)的特性 7 1,只读的,不能更新的。 8 2,不滚动的 9 3,不敏感的,不敏感意为服务器可以活不可以复制它的结果表 10 游标(cursor)必须在声明处理程序之前被声明,并且变量和条件必须在声明游标或处理程序之前被声明。 11 三,使用游标(cursor) 12 1.声明游标 13 DECLARE cursor_name CURSOR FOR select_statement 14 这个语句声明一个游标。也可以在子程序中定义多个游标,但是一个块中的每一个游标必须有唯一的名字。声明游标后也是单条操作的,但是不 15 能用SELECT语句不能有INTO子句。 16 2. 游标OPEN语句 17 OPEN cursor_name 18 这个语句打开先前声明的游标。 19 20 3. 游标FETCH语句 21 FETCH cursor_name INTO var_name [, var_name] ... 22 这个语句用指定的打开游标读取下一行(如果有下一行的话),并且前进游标指针。 23 Mysql是通过一个Error handler的声明来进行判断的指针是否到尾, 24 declare continue handler for Not found (do some action); 25 在Mysql里当游标遍历溢出时,会出现一个预定义的NOT FOUND的Error,我们处理这个Error并定义一个continue的handler就可以了。 26 定义一个flag,在NOT FOUND,标示Flag,在Loop里以这个flag为结束循环的判断就可以了。 27 28 4. 游标CLOSE语句 29 CLOSE cursor_name 30 这个语句关闭先前打开的游标。 31 32 注意:MySQL的游标是向前只读的,也就是说,你只能顺序地从开始往后读取结果集,不能从后往前,也不能直接跳到中间的记录. 33 34 1、 无返回结果语句,如:INSERT,UPDATE,DROP, DELETE等 35 36 2、 select语句返回单行变量并可传给本地变量(select ..into) 37 38 3、 返回多行结果集的select语句,并可使用MySQL游标循环处理
1 CREATE TABLE `users` 2 ( 3 `ID` bigint(20) unsigned NOT NULL AUTO_INCREMENT, 4 `user_name` varchar(60), 5 `user_pass` varchar(64), 6 PRIMARY KEY (`ID`) 7 ) ENGINE=InnoDB AUTO_INCREME 8 9 10 insert into users values(null,'tank','tank'), 11 (null,'zhang','zhang'), 12 (null,'ying','ying'); 13 14 create procedure test_cursor (in param int(10),out result varchar(90)) 15 begin 16 declare name varchar(20); 17 declare pass varchar(20); 18 declare done int; 19 declare cur_test CURSOR for select user_name,user_pass from users; 20 declare continue handler FOR SQLSTATE '02000' SET done = 1; 21 if param then 22 select concat_ws(',',user_name,user_pass) into result from users where id=param; 23 else 24 open cur_test; 25 repeat 26 fetch cur_test into name, pass; 27 select concat_ws(',',result,name,pass) into result;#返回给本地变量输出 28 until done 29 end repeat; 30 close cur_test; 31 end if; 32 end; 33 34 call test_cursor(3,@test); 35 select @test
将上面表users的名字全部变为king
1 create PROCEDURE pro_users() 2 begin 3 DECLARE result varchar(128); 4 DECLARE no int; 5 DECLARE cur_1 CURSOR FOR SELECT user_name FROM users; #把select出来的数据全部保存到游标cur_1中 6 DECLARE CONTINUE HANDLER FOR NOT FOUND SET no=1; #当读到数据的最后一条时,设置no变量为1 7 SET no=0; #初始化变量no为0 8 OPEN cur_1; #打开游标 9 while no=0 do #判断是不是到了最后一条数据 10 fetch cur_1 into result; #读取游标中的数据一一复给变量result。 11 update users set user_name='king' where user_name=result;#这个不用解释了吧 12 end while; 13 close cur_1;#最后关闭游标.游标里面存放了那么多数据总是要清理掉的吧。 14 end 15 16 call pro_users()
④、mysql游标使用
⑤、mysql游标循环的使用
2、Mysql数据库——触发器
①、MySQL 触发器简单实例
②、mysql触发器的作用及语法
③、mysql之触发器trigger
④、mysql 触发器学习