mysql存储过程使用心得

2018-04-11 04:23:40

由于经常跟MYSQL数据库打交道,在工作中因为各种需要写了一些存储过程,

这里把平时使用存储过的心得体会写一下:
1.MYSQL存储函数
mysql存储过程以是SQL语言开发的,SQL语言平时使用中最常用的就是 UPDATE,INSERT/REPLACE,DELETE
等了,但是,由于MYSQL 提供了很多内部函数,因此,结合SQL语言,我们可以实现很多功能。
比如,有的时候,我们需要对某个字段进行使用某个HASH函当选,一般来说我们都是通过外部程序编写代码实现
hash函数,处理好后再交给MYSQL进行操作。
当然如果对性能要求不是那么高的话,我们可以使用MYSQL存储函数来实现,可以简化对数据的操作,直接使用SQL就可以了,
省去了外部脚本的开发。
unsigned long ELFHash(const char *key)
{
unsigned long h = 0;
unsigned long x = 0;
while (*key)
{
h = (h << 4) + (*key++);
if ((x = h & 0xF0000000L) != 0)
{
h ^= (x >> 24);
h &= ~x;
}
}
return (h & 0x7FFFFFFF);
}
这里是 ELFHash 的SQL实现:
CREATE DEFINER=`test`@`127.0.0.1` FUNCTION `elfhash`(strkey varchar(256)) RETURNS int(11)
NO SQL
BEGIN

DECLARE h INT UNSIGNED DEFAULT 0;
DECLARE x INT UNSIGNED DEFAULT 0;
DECLARE k INT UNSIGNED DEFAULT 0;
DECLARE strLen INT UNSIGNED DEFAULT 0;
DECLARE pos INT UNSIGNED DEFAULT 0;
DECLARE cc char(1) DEFAULT NULL;

SELECT LENGTH(strkey)+1 INTO strLen ;

WHILE (pos<strLen) DO
set k = ASCII(MID(strkey,pos,1));
set h = (h << 4) + k;
SET x = h & 0xF0000000;
IF (x!= 0) THEN
set h = h^(x >> 24);
set h = h&(~x);
END IF;

SET pos = pos +1;
END WHILE;

SET h = h & 0x7FFFFFFF;
RETURN h;
END

2. 在SQL里使用一些内部函数,可以实现一些比较有用的功能
比如,MYSQL提供的BIT类型长度有限,最大只能是64位,
如果需要操作大于64位的BITARRY,MYSQL提供的类型就不能满足了。
这时我们可以使用存储函数来模拟一个大于64位的BITARRY

# 取得某一位的值
CREATE DEFINER=`ttuser`@`116.228.188.74` FUNCTION `getbitarry`(aStatus varchar(22), aTaskID int) RETURNS int(11)
NO SQL
BEGIN
#Routine body goes here...

RETURN MID(aStatus,aTaskID,1);
END


# 给BIT位赋值
CREATE DEFINER=`ttuser`@`116.228.188.74` FUNCTION `setbitarray`(aStatus varchar(22), aTaskID int, aflag int) RETURNS varchar(22) CHARSET utf8
NO SQL
BEGIN
#Routine body goes here...
SET aStatus = CONCAT(MID(aStatus, IF(aflag,1,0), aTaskID-1), aflag, MID(aStatus, aTaskID+1, LENGTH(aStatus)-aTaskID));
RETURN aStatus;
END


CREATE TABLE `t_test` (
`Fid` int(11) NOT NULL,
`Fstatus` varchar(22) NOT NULL DEFAULT '',
PRIMARY KEY (`Fid`,`Fstatus`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8;

UPDATE t_test SET Fstatus = setbitarray(Fstatus, 4, 1) ;
SELECT *, getbit(Fstatus,1), getbit(Fstatus,2) FROM t_test;



3. mysql存储过程里的拼SQL问题:

很多时候由于数据库被分库分表,在写存储过程的时候,库表名就需要动态生成, 简单的用set或者declare语句定义变量,然后直接作为sql的表名是不行的,mysql会把变量名当作表名。解决方法是将整条sql语句作为变量,其中穿插变量作为表名,
然后用 prepare stmt 语句执行。

SET @sqlStr = CONCAT('SELECT COUNT(DISTINCT TABLE_SCHEMA) FROM information_schema.TABLES WHERE TABLE_SCHEMA=\'',
@strDBname, '\' AND TABLE_NAME=\'', @strTBname,'\' INTO @nCount');
SELECT @sqlStr;
prepare stmt from @sqlStr;EXECUTE stmt;DEALLOCATE prepare stmt;

预处理语句的SQL语法可以在存储过程中使用,但是不能在存储函数或触发程序中使用。

 

4. clickhouse使用case when实现区间查询

select user.age_range as age,count(distinct user.cuid) as count
from (
select
cuid,
case
when age>='18' and age<='24' then '18-24岁'
when age>='25' and age<='29' then '25-29岁'
when age>='30' and age<='34' then '30-34岁'
when age>='35' then '35-100岁'
end as age_range
from
person
where
has(professional, 'student')
and
has(sex, '男')
) user group by age_range