本篇文章主要介绍mysql中FIND_IN_SET函数用法,用来精确查询字段中以逗号分隔的数据
以及其与 like 和 in 的区别
1.问题发现
之前在做工作任务时有这么一个需求:需要用接口所传的服务商编号和所开通的产品类型查询这张表中是否有此信息来做返回结果。
但公司的产品类型有多个,每个服务商可能开通了多个不同的产品类型,存入产品类型时用的是一个字段,用逗号分隔开存储。
这种场景下就需要精确查找其产品类型,我一开始想的是用 in 和 like 来实现,但实际使用后并不是这种效果...请看下文讲解~
2.解决问题
使用 in 实现
sql语句如下:
SELECT
agentNum
FROM
p4_agent_limit_merc_access a
WHERE
agentNum = #{agentNum} (传入的服务商编号)
AND #{productType} (传入的产品类型)
IN(a.productType);
但实际上这样写是查不到数据的,只有a.productType 字段的值等于传入的产品类型时(和IN前面的字符串完全匹配),这时查询才有效,否则是查不到结果的,即使a.productType 里面包含传入的产品类型。
使用 like 实现
sql语句如下:
SELECT
agentNum
FROM
p4_agent_limit_merc_access a
WHERE
agentNum = #{agentNum}
AND a.productType
LIKE concat('%',#{productType}, '%');
表面看上去这样写是没问题的,但你细品一下,就知道问题很大,因为like是广泛的模糊查询,但一个字符串里包含你要传入的值时,也能查出来,比如我们的产品类型productType有QPOS和POS,然而此次查询的服务商只开通了QPOS产品,我却传入了POS这个类型,用上述语句查询后也能查出这条结果,这就是问题所在,所以用like查询出的范围会更广,这样明显不合理,不是我们想要的结果...
使用FIND_IN_SET函数实现
首先介绍下 FIND_IN_SET函数:
FIND_IN_SET(str,strlist)
str 要查询的字符串
strlist 字段名 参数以”,”分隔 如 (1,2,6,8)
查询字段(strlist)中包含(str)的结果,返回结果为null或记录
假如字符串str在由N个子链组成的字符串列表strlist 中,则返回值的范围在 1 到 N 之间。 一个字符串列表就是一个由一些被 ‘,’ 符号分开的子链组成的字符串。如果第一个参数是一个常数字符串,而第二个是type SET列,则FIND_IN_SET() 函数被优化,使用比特计算。 如果str不在strlist 或strlist 为空字符串,则返回值为 0 。如任意一个参数为NULL,则返回值为 NULL。这个函数在第一个参数包含一个逗号(‘,’)时将无法正常运行。
介绍简单了解下就好,具体看怎么用,我的sql如下:
SELECT
agentNum
FROM
p4_agent_limit_merc_access a
WHERE
agentNum = #{agentNum}
AND FIND_IN_SET(#{productType},a.productType);
使用上述语句就可以精确查询出数据,实现需求。
再来看看FIND_IN_SET函数具体使用例子:
SELECT FIND_IN_SET('b', 'a,b,c,d'); 返回2
因为b 在strlist集合中放在2的位置 从1开始
select FIND_IN_SET('1', '1'); 返回 就是1 这时候的strlist集合有点特殊 只有一个字符串 其实就是要求前一个字符串 一定要在后一个字符串集合中 才返回 大于0的数
select FIND_IN_SET('2', '1,2'); 返回2
select FIND_IN_SET('6', '1'); 返回0
3.总结
当 a.productType 字段是常量时,则可以用 in 来实现。
当其为变量时,则必须要用FIND_IN_SET函数来实现了。
更多精彩功能请关注我的个人博客网站:https://liujian.cool
欢迎关注我的个人公众号:程序猿刘川枫