记录一条sql查询:以逗号隔开的id字符串的查询

目录

前言

在一个项目中有两张表,一张是商品码表,一张是记录出库单明细的出库记录表,记录表中有一个字段保存了以逗号隔开的商品码表的id字符串,需要根据出库明细id查找到对应出库的商品码。

表结构

goods_detail 商品码表结构

sql 复制代码
CREATE TABLE `goods_detail` (
  `id` bigint(20) NOT NULL COMMENT '主键',
  `goods_id` bigint(20) NOT NULL COMMENT '商品id',
  `bar_code` varchar(20) DEFAULT NULL COMMENT '箱码',
  `bottle_code` varchar(20) DEFAULT NULL COMMENT '瓶码'
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 ROW_FORMAT=DYNAMIC COMMENT='商品码表';

out_goods_record 出库记录表结构

sql 复制代码
CREATE TABLE `out_goods_record` (
  `id` bigint(20) NOT NULL COMMENT 'ID',
  `order_detail_id` bigint(20) NOT NULL COMMENT '出库单明细id',
  `goods_detail_id` longtext COMMENT '商品码id,多个用'',''分隔'
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 ROW_FORMAT=DYNAMIC COMMENT='出库记录';

sql语句

  • 使用查询函数FIND_IN_SET和函数GROUP_CONCAT
sql 复制代码
SELECT
	id,
	bar_code,
	bottle_code
FROM
	goods_detail
WHERE
	FIND_IN_SET(
		id,(
		SELECT
			GROUP_CONCAT( goods_detail_id )
		FROM
			out_good_record
		WHERE
			del_flag = '0'
			AND order_detail_id = 1734492741712220161
		GROUP BY
			order_detail_id
		))>0;

该sql查询效率非常低,需要147秒,前端会提示请求接口超时,不推荐。

  • 将逗号隔开的商品码id转成多行再查询
    将逗号隔开的商品码id转成多行的sql
sql 复制代码
SELECT
			substring_index( substring_index( r.goods_detail_id, ',', s.id + 1 ), ',',- 1 ) AS goods_detail_id
			FROM
			out_good_record r
			CROSS JOIN auto_add_seq s ON s.id <= LENGTH( r.goods_detail_id ) - LENGTH(
			REPLACE ( r.goods_detail_id, ',', '' ))
			WHERE
			r.order_detail_id = 1734492741712220161

auto_add_seq 这张表为自定义表,参考 mysql中字符串截取与拆分链接中的 二、分割成多行

  • 使用EXISTS
sql 复制代码
SELECT
	id,
	bar_code,
	bottle_code 
FROM
	goods_detail d 
WHERE
	EXISTS (
	SELECT
		1 
	FROM
		(
		SELECT
			substring_index( substring_index( r.goods_detail_id, ',', s.id + 1 ), ',',- 1 ) AS goods_detail_id 
		FROM
			out_good_record r
			CROSS JOIN incr_sequence s ON s.id <= LENGTH( r.goods_detail_id ) - LENGTH(
			REPLACE ( r.goods_detail_id, ',', '' )) 
		WHERE
			r.order_detail_id = 1734492741712220161 
		) t 
	WHERE
		t.goods_detail_id = d.id 
	);

由于数据量太大,使用EXISTS花费的时间太多了,EXISTS比较适合内部是大表的情况,不推荐使用

  • 使用IN查询
sql 复制代码
SELECT
	id,
	bar_code,
	bottle_code 
FROM
	goods_detail d 
WHERE
	d.id IN (
	SELECT
		substring_index( substring_index( r.goods_detail_id, ',', s.id + 1 ), ',',- 1 ) AS goods_detail_id 
	FROM
		out_good_record r
		CROSS JOIN incr_sequence s ON s.id < LENGTH( r.goods_detail_id ) - LENGTH( REPLACE ( r.goods_detail_id, ',', '' ) + 1 ) 
	WHERE
		r.order_detail_id = 1734492741712220161 
	);

使用IN查询,要效率要高多了

  • 关联查询
sql 复制代码
SELECT
	id,
	bar_code,
	bottle_code
FROM
	goods_detail d,
	(
	SELECT
	substring_index( substring_index( r.goods_detail_id, ',', s.id + 1 ), ',',- 1 ) AS goods_detail_id
	FROM
	out_good_record r
	CROSS JOIN incr_sequence s ON s.id <= LENGTH( r.goods_detail_id ) - LENGTH(
	REPLACE ( r.goods_detail_id, ',', '' ))
	WHERE
	r.order_detail_id = 1734492741712220161
	) t
WHERE
	d.id = t.goods_detail_id

关联查询效率最高,推荐使用

相关推荐
KmSH8umpK5 分钟前
Redis分布式锁从原生手写到Redisson高阶落地,附线上死锁复盘优化方案进阶第九篇
数据库·redis·分布式
悠悠121385 分钟前
一条 SQL 从敲下回车,到在 MySQL 里“跑完一生”,中间到底经历了啥?
数据库·sql·mysql
秋916 分钟前
MySQL 9.7.0 使用详解:新特性、实战与避坑指南
android·数据库·mysql
S1998_1997111609•X18 分钟前
恶意注入污染蜜罐HDMI进程函数值进行封禁垃圾蠕虫仓蟲的轮系依据行为戆直㞢仓shell token
数据库·爬虫·网络协议·百度·开闭原则
Yupureki1 小时前
《Redis数据库》1.初识Redis
数据库·redis·缓存
Lyyaoo.1 小时前
Redis实现分布式锁
数据库·redis·分布式
张~颜1 小时前
autovacuum
数据库·postgresql
山峰哥1 小时前
SQL优化从入门到精通:20个案例破解性能密码
数据库·sql·oracle·性能优化·深度优先
努力努力再努力wz1 小时前
【MySQL进阶系列】拒绝冗余SQL:带你透彻理解视图的底层逻辑
android·c语言·数据结构·数据库·c++·sql·mysql
历程里程碑1 小时前
MySQL数据类型全解析 + 代码实操讲解
大数据·开发语言·数据库·sql·mysql·elasticsearch·搜索引擎