而MySQL另外一个最流行的存储引擎之一Innodb存储数据的策略是分为两种的,一种是共享表空间存储方式,还有一种是独享表空间存储方式。
当使用共享表空间存储方式的时候,Innodb的所有数据保存在一个单独的表空间里面,而这个表空间可以由很多个文件组成,一个表可以跨多个文件存在,所 以其大小限制不再是文件大小的限制,而是其自身的限制。从Innodb的官方文档中可以看到,其表空间的最大限制为64TB,也就是说,Innodb的单 表限制基本上也在64TB左右了,当然这个大小是包括这个表的所有索引等其他相关数据。
而当使用独享表空间来存放Innodb的表的时候,每个表的数据以一个单独的文件来存放,这个时候的单表限制,又变成文件系统的大小限制了。
--传入id,输出name和sex的存储过程,这里同个id有多条数据,所以需要用到游标。
DELIMITER //
CREATE PROCEDURE p5(IN v_id INT)
BEGIN
DECLARE nodata INT DEFAULT 0#注意:这个变量声明必须放在游标声明前面
DECLARE v_name VARCHAR(30)
DECLARE v_sex CHAR(3)
DECLARE c_ns CURSOR FOR SELECT NAME,sex FROM t WHERE id = v_id
DECLARE EXIT HANDLER FOR NOT FOUND SET nodata = 1#当读到数据的最后一条时,设置变量为1
OPEN c_ns
WHILE nodata = 0 DO#判断是不是到了最后一条数据
FETCH c_ns INTO v_name,v_sex
SELECT v_name,v_sex,nodata
END WHILE
CLOSE c_ns
END
//
--执行该存储过程
mysql>call p5(2)
+--------+-------+--------+| v_name | v_sex | nodata |
+--------+-------+--------+
| song | 女 | 0 |
+--------+-------+--------+
1 row in set (0.04 sec)
+--------+-------+--------+
| v_name | v_sex | nodata |
+--------+-------+--------+
| dan | 男 | 0 |
+--------+-------+--------+
1 row in set (0.05 sec)
Query OK, 0 rows affected, 1 warning (0.05 sec)
--注意:CONTINUE 会继续当前的block 中的语句, 它在set done=1 后继续执行下一个语句。EXIT 则在 set done=1 后离开当前的语句块
所以这里用了EXIT(黄色阴影部分).
如果是continue,结果如下:
mysql> call p5(2)-> //
+--------+-------+
| v_name | v_sex |
+--------+-------+
| song | 女 |
+--------+-------+
1 row in set (0.00 sec)
+--------+-------+
| v_name | v_sex |
+--------+-------+
| dan | 男 |
+--------+-------+
1 row in set (0.01 sec)
+--------+-------+
| v_name | v_sex |
+--------+-------+
| dan | 男 |
+--------+-------+
1 row in set (0.02 sec)
Query OK, 0 rows affected, 1 warning (0.03 sec)
如果没有则插入数据,如果有则更新的方法:insert into `test` values (a,b) ON DUPLICATE KEY UPDATE `a`=c
[sql] view plain copy
CREATE PROCEDURE `test`.`new_procedure` ()
BEGIN
-- 需要定义接收游标数据的变量
DECLARE a CHAR(16)
-- 游标
DECLARE cur CURSOR FOR SELECT i FROM test.t
-- 遍历数据结束标志
DECLARE done INT DEFAULT FALSE
-- 将结束标志绑定到游标
DECLARE CONTINUE HANDLER FOR NOT FOUND SET done = TRUE
-- 打开游标
OPEN cur
-- 开始循环
read_loop: LOOP
-- 提取游标里的数据,这里只有一个,多个的话也一样;
FETCH cur INTO a
-- 声明结束的时候
IF done THEN
LEAVE read_loop
END IF
-- 这里做你想做的循环的事件
INSERT INTO test.t VALUES (a)
END LOOP
-- 关闭游标
CLOSE cur
END
实例2
[sql] view plain copy
BEGIN
DECLARE Done INT DEFAULT 0
DECLARE CurrentLingQi INT
DECLARE ShizuName VARCHAR(30)
/* 声明游标 */
DECLARE rs CURSOR FOR SELECT NodeName, LingQi FROM socialrelation
/* 异常处理 */
DECLARE CONTINUE HANDLER FOR SQLSTATE '02000' SET Done = 1
/* 打开游标 */
OPEN rs
/* 逐个取出当前记录LingQi字段的值,需要进行最大值的判断 */
FETCH NEXT FROM rs INTO ShizuName, CurrentLingQi
/* 遍历数据表 */
REPEAT
IF NOT Done THEN
SET CurrentLingQi = CurrentLingQi + 60
/* 如果更新后灵气值大于允许的最大值,则就设置为最大值 */
IF CurrentLingQi >= 1800 THEN
UPDATE socialrelation SET LingQi = 1800 WHERE NodeName = ShizuName
ELSE
/* 否则,正常更新 */
UPDATE socialrelation SET LingQi = LingQi + 60 WHERE NodeName = ShizuName
END IF
END IF
FETCH NEXT FROM rs INTO ShizuName, CurrentLingQi
UNTIL Done END REPEAT
/* 关闭游标 */
CLOSE rs
END
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)