记录一条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

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

相关推荐
程序员的世界你不懂1 小时前
【Flask】测试平台开发,新增说明书编写和展示功能 第二十三篇
java·前端·数据库
自学也学好编程1 小时前
【数据库】Redis详解:内存数据库与缓存之王
数据库·redis
JAVA不会写2 小时前
在Mybatis plus中如何使用自定义Sql
数据库·sql
IT 小阿姨(数据库)2 小时前
PgSQL监控死元组和自动清理状态的SQL语句执行报错ERROR: division by zero原因分析和解决方法
linux·运维·数据库·sql·postgresql·centos
ChinaRainbowSea2 小时前
7. LangChain4j + 记忆缓存详细说明
java·数据库·redis·后端·缓存·langchain·ai编程
小马学嵌入式~3 小时前
嵌入式 SQLite 数据库开发笔记
linux·c语言·数据库·笔记·sql·学习·sqlite
Java小白程序员4 小时前
MyBatis基础到高级实践:全方位指南(中)
数据库·mybatis
Monly214 小时前
人大金仓:merge sql error, dbType null, druid-1.2.20
数据库·sql
不宕机的小马达4 小时前
【Mysql|第一篇】Mysql的安装与卸载、Navicat工具的使用
数据库·mysql
float_六七4 小时前
数据库连接池:性能优化的秘密武器
数据库·oracle·性能优化