SQL Query 또는 Laravel SQL Query Builder를 사용하여 테이블 / 열 조합 생성
Nov 15 2020
제품 변형에 대한 기존 계획이 있습니다.
각 생산 시간, 수량 및 변형 옵션의 조합을 만들고 싶습니다.
제품의 수량, 생산 시간, 변형 및 변형 옵션에 액세스하여 선택 양식을 만듭니다.
table_groups
+------------+
| id | title |
+----+-------+
| 1 | rug |
+----+-------+
table_days
+----+----------+------+
| id | group_id | day |
+----+----------+------+
| 1 | 1 | 1 |
| 2 | 1 | 2 |
| 3 | 1 | 3 |
+----+----------+------+
table_quantities
+----+----------+-----------+
| id | group_id | quantity |
+----+----------+-----------+
| 1 | 1 | 100 |
| 2 | 1 | 200 |
| 3 | 1 | 300 |
| 4 | 1 | 400 |
+----+----------+-----------+
table_attributes
+----+----------+-----------+
| id | group_id | title |
+----+----------+-----------+
| 1 | 1 | Color |
| 2 | 1 | Size |
+----+----------+-----------+
table_attribute_values
+----+----------+--------------+--------+
| id | group_id | attribute_id | title |
+----+----------+--------------+--------+
| 1 | 1 | 1 | Red |
| 2 | 1 | 1 | Yellow |
| 3 | 1 | 1 | Black |
| 4 | 1 | 2 | Small |
| 5 | 1 | 2 | Medium |
+----+----------+--------------+--------+
예제 스키마를 준비했습니다. 그러나 원하는 결과를 얻지 못했습니다.
SQL 바이올린
내가 한 이만큼 :
SELECT
GROUP_CONCAT(DISTINCT days_group) as days_list,
GROUP_CONCAT(DISTINCT quantities_group SEPARATOR ',') as quantities_list,
GROUP_CONCAT(DISTINCT attribute_values_group SEPARATOR ',') as attribute_values_list
FROM
table_groups
LEFT JOIN (
SELECT days.day, days.group_id,
GROUP_CONCAT(days.day) as days_group
FROM table_days days GROUP BY days.id
) joindays ON joindays.group_id = table_groups.id
LEFT JOIN (
SELECT quantities.quantity, quantities.group_id,
GROUP_CONCAT(quantities.quantity) as quantities_group
FROM table_quantities quantities GROUP BY quantities.id
) joinquantities ON joinquantities.group_id = table_groups.id
LEFT JOIN table_attributes attributes ON attributes.group_id = table_groups.id
LEFT JOIN (
SELECT attribute_id, group_id,
GROUP_CONCAT(attribute_values.title) as attribute_values_group
FROM table_attribute_values attribute_values
GROUP BY attribute_values.attribute_id, attribute_values.id
) joinattributevalues ON joinattributevalues.attribute_id = attributes.id
GROUP BY joinattributevalues.attribute_id;
쿼리 결과 :
+---------------+-----------+-----------------+-----------------------+
| group_id | days_list | quantities_list | attribute_values_list |
+---------------+-----------+-----------------+-----------------------+
| 1 | 1,2,3 | 100,200,300,400 | Red,Yellow,Black |
| 2 | 1,2,3 | 100,200,300,400 | Small,Medium |
+---------------+-----------+-----------------+-----------------------+
내가 원하는 올바른 결과는 다음과 같습니다. 도와 주실 수 있나요?
+-----------+---------------------+--------+
| group_id | combinations | price |
+-----------+---------------------+--------+
| 1 | 1-100-Red-Small | |
+-----------+---------------------+--------+
| 1 | 1-100-Red-Medium | |
+-----------+---------------------+--------+
| 1 | 1-100-Yellow-Small | |
+-----------+---------------------+--------+
| 1 | 1-100-Yellow-Medium | |
+-----------+---------------------+--------+
| 1 | 1-100-Black-Small | |
+-----------+---------------------+--------+
| 1 | 1-100-Black-Medium | |
+-----------+---------------------+--------+
| 1 | 1-200-Red-Small | |
+-----------+---------------------+--------+
| 1 | 1-200-Red-Medium | |
+-----------+---------------------+--------+
| 1 | 1-200-Yellow-Small | |
+-----------+---------------------+--------+
| 1 | 1-200-Yellow-Medium | |
+-----------+---------------------+--------+
| 1 | 1-200-Black-Small | |
+-----------+---------------------+--------+
| 1 | 1-200-Black-Medium | |
+-----------+---------------------+--------+
| 1 | 1-300-Red-Small | |
+-----------+---------------------+--------+
| 1 | 1-300-Red-Medium | |
+-----------+---------------------+--------+
| 1 | 1-300-Yellow-Small | |
+-----------+---------------------+--------+
| 1 | 1-300-Yellow-Medium | |
+-----------+---------------------+--------+
| 1 | 1-300-Black-Small | |
+-----------+---------------------+--------+
| 1 | 1-300-Black-Medium | |
+-----------+---------------------+--------+
| 1 | 1-400-Red-Small | |
+-----------+---------------------+--------+
| 1 | 1-400-Red-Medium | |
+-----------+---------------------+--------+
| 1 | 1-400-Yellow-Small | |
+-----------+---------------------+--------+
| 1 | 1-400-Yellow-Medium | |
+-----------+---------------------+--------+
| 1 | 1-400-Black-Small | |
+-----------+---------------------+--------+
| 1 | 1-400-Black-Medium | |
+-----------+---------------------+--------+
| 1 | 2-100-Red-Small | |
+-----------+---------------------+--------+
| 1 | 2-100-Red-Medium | |
+-----------+---------------------+--------+
| 1 | 2-100-Yellow-Small | |
+-----------+---------------------+--------+
| 1 | 2-100-Yellow-Medium | |
+-----------+---------------------+--------+
| 1 | 2-100-Black-Small | |
+-----------+---------------------+--------+
| 1 | 2-100-Black-Medium | |
+-----------+---------------------+--------+
| 1 | 2-200-Red-Small | |
+-----------+---------------------+--------+
| 1 | 2-200-Red-Medium | |
+-----------+---------------------+--------+
| 1 | 2-200-Yellow-Small | |
+-----------+---------------------+--------+
| 1 | 2-200-Yellow-Medium | |
+-----------+---------------------+--------+
| 1 | 2-200-Black-Small | |
+-----------+---------------------+--------+
| 1 | 2-200-Black-Medium | |
+-----------+---------------------+--------+
| 1 | 2-300-Red-Small | |
+-----------+---------------------+--------+
| 1 | 2-300-Red-Medium | |
+-----------+---------------------+--------+
| 1 | 2-300-Yellow-Small | |
+-----------+---------------------+--------+
| 1 | 2-300-Yellow-Medium | |
+-----------+---------------------+--------+
| 1 | 2-300-Black-Small | |
+-----------+---------------------+--------+
| 1 | 2-300-Black-Medium | |
+-----------+---------------------+--------+
| 1 | 2-400-Red-Small | |
+-----------+---------------------+--------+
| 1 | 2-400-Red-Medium | |
+-----------+---------------------+--------+
| 1 | 2-400-Yellow-Small | |
+-----------+---------------------+--------+
| 1 | 2-400-Yellow-Medium | |
+-----------+---------------------+--------+
| 1 | 2-400-Black-Small | |
+-----------+---------------------+--------+
| 1 | 2-400-Black-Medium | |
+-----------+---------------------+--------+
| 1 | 3-100-Red-Small | |
+-----------+---------------------+--------+
| 1 | 3-100-Red-Medium | |
+-----------+---------------------+--------+
| 1 | 3-100-Yellow-Small | |
+-----------+---------------------+--------+
| 1 | 3-100-Yellow-Medium | |
+-----------+---------------------+--------+
| 1 | 3-100-Black-Small | |
+-----------+---------------------+--------+
| 1 | 3-100-Black-Medium | |
+-----------+---------------------+--------+
| 1 | 3-200-Red-Small | |
+-----------+---------------------+--------+
| 1 | 3-200-Red-Medium | |
+-----------+---------------------+--------+
| 1 | 3-200-Yellow-Small | |
+-----------+---------------------+--------+
| 1 | 3-200-Yellow-Medium | |
+-----------+---------------------+--------+
| 1 | 3-200-Black-Small | |
+-----------+---------------------+--------+
| 1 | 3-200-Black-Medium | |
+-----------+---------------------+--------+
| 1 | 3-300-Red-Small | |
+-----------+---------------------+--------+
| 1 | 3-300-Red-Medium | |
+-----------+---------------------+--------+
| 1 | 3-300-Yellow-Small | |
+-----------+---------------------+--------+
| 1 | 3-300-Yellow-Medium | |
+-----------+---------------------+--------+
| 1 | 3-300-Black-Small | |
+-----------+---------------------+--------+
| 1 | 3-300-Black-Medium | |
+-----------+---------------------+--------+
| 1 | 3-400-Red-Small | |
+-----------+---------------------+--------+
| 1 | 3-400-Red-Medium | |
+-----------+---------------------+--------+
| 1 | 3-400-Yellow-Small | |
+-----------+---------------------+--------+
| 1 | 3-400-Yellow-Medium | |
+-----------+---------------------+--------+
| 1 | 3-400-Black-Small | |
+-----------+---------------------+--------+
| 1 | 3-400-Black-Medium | |
+-----------+---------------------+--------+
참고 : 그룹, 속성 및 속성 값의 수에는 제한이 없습니다. 예제 결과는 다음과 같을 수 있습니다.
Attributes:
+-------+------+-------+--------+
| Color | Size | Model | Gender |
+-------+------+-------+--------+
Combinations:
+------------------------------+
| 1-100-Red-Small-Model 1-Male |
+------------------------------+
| 1-100-Red-Small-Model 2-Male |
+------------------------------+
SQL 쿼리로 수행 할 필요는 없습니다. Laravel Query Builder 메서드로도이를 수행 할 수 있습니다.
도움을 주셔서 미리 감사드립니다.
샘플 SQL Fiddle 확인
답변
3 CoffeeNeedCoffee Nov 17 2020 at 06:17
그렇게해야합니다. 재귀 CTE가 나중에 포함되지 않았기 때문에 mysql 5.7에서는 작동하지 않지만 group_id 당 다양한 수의 속성과 attribute_value를 가질 수 있습니다. 바이올린이 여기 있습니다 .
with recursive allAtts as (
/* Get our attribute list, and format it if we want; concat(a.title, ':', v.title) looks quite nice */
SELECT
att.group_id,
att.id,
CONCAT(v.title) as attDesc,
dense_rank() over (partition by att.group_id order by att.id) as attRank
FROM table_attributes att
INNER JOIN table_attribute_values v
ON v.group_id = att.group_id
AND v.attribute_id = att.id
),
cte as (
/* Recursively build our attribute list, assuming ranks are sequential and we properly linked our group_ids */
select group_id, id, attDesc, attRank from allAtts WHERE attRank = 1
union all
select
allAtts.group_id,
allAtts.id,
concat_ws('-', cte.attDesc, allAtts.attDesc) as attDesc,
allAtts.attRank
from cte
join allAtts ON allAtts.attRank = cte.attRank +1
AND cte.group_id = allAtts.group_id
)
/* Our actual select statement, which RIGHT JOINs against the table_groups
so we don't lose entries w/o attributes */
select
grp.id,
concat_ws('-', d.day, qty.quantity, cte.attDesc) as combinations
from cte
inner join (select group_id, max(attRank) as attID
from cte
group by group_id) m on cte.group_id = m.group_id and m.attID = cte.attrank
RIGHT JOIN table_groups grp ON grp.id = cte.group_id
LEFT JOIN table_days d on grp.id = d.group_id
LEFT JOIN table_quantities qty on grp.id = qty.group_id;