问题背景
如一个老师A有很多个学生,此时是老师A关联学生,也就是在老师A数据表中学生id字段由多个id拼接而成,如(‘4564654531,459634531,4543654531,3975654531,9521654531’)。如果此时要随机抽取几位学生,在抽取的学生中筛选出属于老师A的学生 ,此时in无法满足,find_in_set则可以很好的处理这个问题
in与FIND_IN_SET的区别
需要从一个字段中查找出该字段值中的某个数据(字段类型为字符串,存储的格式为(1,2,3,4))
我需要的是查出该字段中包含6的所有数据,那么我们就用到了find_in_set 这个函数
select * from 表名 where find_in_set(6, 字段名)
接下来详细给大家介绍一下 find_in_set 和 in 之间的区别
测试代码:
CREATE TABLE `test` (
`id` int(8) NOT NULL auto_increment,
`name` varchar(255) NOT NULL,
`list` varchar(255) NOT NULL,
PRIMARY KEY (`id`)
)
INSERT INTO `test` VALUES (1, 'name', 'daodao,xiaohu,xiaoqin');
INSERT INTO `test` VALUES (2, 'name2', 'xiaohu,daodao,xiaoqin');
INSERT INTO `test` VALUES (3, 'name3', 'xiaoqin,daodao,xiaohu');
test1:sql = select * from `test` where 'daodao' IN (`list`);
得到结果空值.
test2:sql = select * from `test` where FIND_IN_SET('daodao',`list`);
得到三条数据。
拿上面的实验数据说话,test1得到的结果为空,为什么呢?因为,mysql中In是比较等不等,此处‘list’是表中的一个字段,也就是变量,除非它的值刚好和name的值一样,否则返回的结果都为空。拿test1来说,也即把‘daodao’改为‘daodao,xiaohu,xiaoqin’才会匹配到第一条记。
test2返回三条数据,可能是我们刚好需要的。mysql中FIND_IN_SET函数用来比较是不是包含,不管‘list’字段是变量或给定的字符串常量都能很好的工作。MySQL中原型为:FIND_IN_SET(str,strlist)。 假如字符串str 在由N 子链组成的字符串列表strlist 中,则返回值的范围在 1 到 N 之间。
一个字符串列表就是一个由一些被‘,‘符号分开的子链组成的字符串。如果第一个参数是一个常数字符串,而第二个是type SET列,则 FIND_IN_SET() 函数被优化,使用比特计算。 如果str不在strlist 或strlist 为空字符串,则返回值为 0 。如任意一个参数为NULL,则返回值为 NULL。这个函数在第一个参数包含一个逗号(‘,’)时将无法正常运行。str也可以是变量,比如表中的一个字段。
当然,这不是我们项目中需要将FIND_IN_SET替换为IN的原因,因为在我们项目中两者都可以实现功能。只是IN比FIND_IN_SET性能高。我们要查询的字段是主键,使用IN时会使用索引,只会查询表中部分数据。FIND_IN_SET则会查询表中全部数据,由于数据量比较大,性能肯定不高,所以替换为IN。
空格与,(逗号)的使用问题
我们使用SQL在MySQL数据中查询,当查询条件为一个数据集合时会经常会使用到空格与逗号将其数据分隔。如下图