如何从具有现有记录的表中安全地创建一对多表?

问题描述 投票:0回答:1

我有一个包含1000个订单记录的表,一个订单可以包含1个或多个产品,但是所有数据都在单个表中。 (SQL Fiddle

相反,订单数据应该在orders表中,并且该订单中的产品应该在映射表中

我现有的订单表

+-----------------+----------+--------------+
| orders_auto_inc | order_id | product_code |
+-----------------+----------+--------------+
|               1 |      333 | abcd         |
|               2 |      334 | same product |
|               3 |      334 | same product |
|               4 |      334 | same product |
|               5 |      337 | hij          |
+-----------------+----------+--------------+

我要创建的新映射表

CREATE TABLE orders_prod_map (
    orders_auto_inc int(10),
    product_code varchar(100)
);

我希望决赛桌看起来像这样

mysql> SELECT * FROM orders;
+-----------------+----------+
| orders_auto_inc | order_id |
+-----------------+----------+
|               1 |      333 |
|               2 |      334 |
|               5 |      337 |
+-----------------+----------+

mysql> SELECT * FROM orders_prod_map;
+-----------------+--------------+
| orders_auto_inc | product_code |
+-----------------+--------------+
|               5 | hij          |
|               1 | abcd         |
|               2 | same product |
|               2 | same product |
|               2 | same product |
+-----------------+--------------+

我可以编写一个Python脚本来做到这一点,但是总有可以仅通过MySQL查询将数据导入到映射表中吗?

mysql normalization
1个回答
1
投票

我个人将quantity列添加到orders_prod_map以在将来加快查询速度(只计算一次COUNT(*)而不是每个SELECT)。您可以使用这些查询生成必要的数据。

Create `orders_prod_map`:
INSERT INTO orders_prod_map
SELECT m.orders_auto_inc, COUNT(*), o.product_code
FROM orders o
JOIN (SELECT order_id, MIN(orders_auto_inc) AS orders_auto_inc
      FROM orders
      GROUP BY order_id) m ON m.order_id = o.order_id
GROUP BY o.order_id, o.product_code

orders中删除重复的行:

DELETE o1
FROM orders o1
JOIN orders o2 ON o2.order_id = o1.order_id
              AND o2.orders_auto_inc < o1.orders_auto_inc

并从product_code中删除orders列:

ALTER TABLE orders DROP COLUMN product_code

然后您可以从每个结果表中SELECT *(请注意,我对数据进行了一些修改,以将不同的乘积添加到334号订单以使结果更有用)]

orders

orders_auto_inc     order_id
1                   333
2                   334
5                   337

orders_prod_map

orders_auto_inc     quantity    product_code
1                   1           abcd
2                   1           not same product
2                   2           same product
5                   1           hij

Demo on dbfiddle

如果不需要数量列,请将INSERT查询更改为:

INSERT INTO orders_prod_map
SELECT m.orders_auto_inc, o.product_code
FROM orders o
JOIN (SELECT order_id, MIN(orders_auto_inc) AS orders_auto_inc
      FROM orders
      GROUP BY order_id) m ON m.order_id = o.order_id
ORDER BY o.order_id, product_code

然后orders_prod_map的内容是:

orders_auto_inc     product_code
1                   abcd
2                   not same product
2                   same product
2                   same product
5                   hij

Demo on dbfiddle

© www.soinside.com 2019 - 2024. All rights reserved.