- 1 1. Introduction
- 2 2. Basics of the WITH Clause (Common Table Expressions)
- 3 3. Basic Usage of the MySQL WITH Clause
- 4 4. ตัวอย่างการใช้งานจริงของ Recursive WITH Clauses
- 5 5. หมายเหตุและข้อพิจารณาเมื่อใช้ WITH Clause
- 6 6. กรณีการใช้งานจริงในสถานการณ์โลกจริง
- 7 7. คำถามที่พบบ่อย (FAQ)
- 7.1 Q1: เมื่อใดควรใช้ WITH clause?
- 7.2 Q2: เมื่อใดที่ recursive CTE มีประโยชน์?
- 7.3 Q3: คำสั่ง WITH มีประสิทธิภาพมากกว่าการใช้ view หรือไม่?
- 7.4 Q4: สาเหตุใดที่ทำให้ประสิทธิภาพลดลงเมื่อใช้ WITH clause?
- 7.5 Q5: มีทางเลือกใดสำหรับ MySQL เวอร์ชันที่ไม่รองรับ WITH clause?
- 7.6 Q6: แนวปฏิบัติที่ดีที่สุดเมื่อใช้ WITH clause คืออะไร?
- 8 8. Conclusion
1. Introduction
MySQL เป็นระบบจัดการฐานข้อมูลที่ได้รับความนิยมอย่างกว้างขวางจากนักพัฒนาและผู้ดูแลฐานข้อมูล โดยให้ฟังก์ชัน SQL ที่ทรงพลังและยืดหยุ่น ในคุณสมบัติต่าง ๆ ของมัน WITH clause (Common Table Expression: CTE) ที่ถูกแนะนำใน MySQL 8.0 เป็นเครื่องมือที่ช่วยให้คำสั่ง SQL อ่านง่ายขึ้นและเพิ่มความสามารถในการบำรุงรักษา
บทความนี้อธิบาย WITH clause ตั้งแต่พื้นฐานจนถึงการใช้งานขั้นสูง โดยมุ่งเป้าไปที่ผู้เริ่มต้นถึงผู้ใช้ระดับกลาง โดยเฉพาะเราจะครอบคลุมหัวข้อเชิงปฏิบัติเช่น การแทนที่ subqueries และ การทำงานของ recursive queries
สำหรับผู้ที่กำลังเรียนรู้ SQL หรือประสบปัญหาในการปรับประสิทธิภาพคำสั่งในงานประจำวัน บทความนี้มุ่งให้แนวทางแก้ไขที่เป็นรูปธรรม ติดตามเนื้อหาต่อไปเพื่อทำความเข้าใจพื้นฐานของ WITH clause และนำไปใช้ในสถานการณ์จริง
2. Basics of the WITH Clause (Common Table Expressions)
What Is the WITH Clause?
WITH clause คือไวยากรณ์ที่ใช้กำหนดชุดผลลัพธ์ชั่วคราว (Common Table Expression, CTE) ภายในคำสั่ง SQL และอ้างอิงมันในคำสั่งต่อ ๆ ไป รองรับตั้งแต่ MySQL 8.0 ขึ้นไป ทำให้การเขียน subquery ที่ซับซ้อนสามารถแปลงเป็นรูปแบบที่ชัดเจนและกระชับยิ่งขึ้น
ตัวอย่างเช่น เมื่อเขียน subquery โดยตรง ความอ่านง่ายอาจลดลงและคำสั่งโดยรวมอาจยาวเกินไป โดยการใช้ WITH clause คุณสามารถแบ่งคำสั่งออกเป็นบล็อกเชิงตรรกะ ทำให้เข้าใจได้ง่ายขึ้น
Basic Syntax of the WITH Clause
ด้านล่างเป็นไวยากรณ์พื้นฐานของ WITH clause:
WITH table_name AS (
SELECT column1, column2
FROM original_table
WHERE condition
)
SELECT column1, column2
FROM table_name;
ในไวยากรณ์นี้ ตารางเสมือน (Common Table Expression) จะถูกกำหนดหลัง WITH แล้วนำไปใช้ในคำสั่งหลัก ทำให้การใช้ subquery ที่ต้องเรียกซ้ำหลายครั้งสามารถเขียนได้อย่างกระชับ
Differences from Subqueries and Views
WITH clause สร้างชุดผลลัพธ์ที่มีอยู่ชั่วคราวและแตกต่างจาก subquery และ view ในหลายด้าน
| Feature | WITH Clause | Subquery | View |
|---|---|---|---|
| Scope | Valid only within the query | Usable only where defined | Reusable across the entire database |
| Persistence | Temporary | Temporary | Permanent |
| Purpose | Simplifies complex queries | Temporary data extraction | Frequently reused data extraction |
WITH clause อ่านง่ายกว่า subquery และเหมาะสำหรับกรณีที่ไม่จำเป็นต้องสร้างวัตถุถาวรเช่น view
Benefits of Using the WITH Clause
- Improved Query Readability แม้จะมีหลาย subquery อยู่ก็ตาม การจัดระเบียบด้วย WITH clause จะทำให้โครงสร้างชัดเจนขึ้น
- Improved Reusability ด้วยการกำหนดชุดผลลัพธ์ชั่วคราว สามารถอ้างอิงได้หลายครั้งภายในคำสั่งเดียว
- Improved Maintainability เนื่องจากคำสั่งสามารถแบ่งเป็นส่วนเชิงตรรกะ การแก้ไขและขยายจึงทำได้ง่ายขึ้น
3. Basic Usage of the MySQL WITH Clause
Replacing Subqueries
WITH clause เป็นเครื่องมือที่ทรงพลังสำหรับทำให้ subquery ที่ซับซ้อนง่ายขึ้น การฝัง subquery โดยตรงอาจทำให้คำสั่งทั้งหมดซับซ้อนและอ่านยาก แต่การใช้ WITH clause จะช่วยเพิ่มความอ่านง่าย
ด้านล่างเป็นตัวอย่างพื้นฐานของการแทนที่ subquery ด้วย WITH clause
Using a Subquery:
SELECT AVG(sales.total) AS average_sales
FROM (
SELECT SUM(amount) AS total
FROM orders
GROUP BY customer_id
) AS sales;
Using the WITH Clause:
WITH sales AS (
SELECT SUM(amount) AS total
FROM orders
GROUP BY customer_id
)
SELECT AVG(sales.total) AS average_sales
FROM sales;
ในตัวอย่างนี้ เรากำหนดชุดผลลัพธ์ชั่วคราวชื่อ sales ด้วย WITH clause แล้วอ้างอิงในคำสั่งหลัก ทำให้คำสั่งทั้งหมดอ่านง่ายและจัดระเบียบได้ดียิ่งขึ้น
Defining Multiple Common Table Expressions (CTEs)
WITH clause อนุญาตให้กำหนด CTE หลาย ๆ ตัวได้ ซึ่งช่วยให้การแยกโมดูลของคำสั่งที่ซับซ้อนทำได้มากขึ้น
Example:
WITH
sales_per_customer AS (
SELECT customer_id, SUM(amount) AS total_sales
FROM orders
GROUP BY customer_id
),
high_value_customers AS (
SELECT customer_id
FROM sales_per_customer
WHERE total_sales > 10000
)
SELECT customer_id
FROM high_value_customers;
In this example, sales_per_customer คำนวณยอดขายรวมต่อแต่ละลูกค้า, และจากผลลัพธ์นั้น, high_value_customers ดึงข้อมูลลูกค้าที่มีจำนวนการซื้อสูง. โดยการใช้ CTE หลายตัวต่อเนื่องกัน, คำสั่ง query สามารถสร้างขึ้นเป็นขั้นตอนได้.
การใช้ Nested CTEs
โดยการใช้ nested CTEs, คุณสามารถทำการดำเนินการข้อมูลที่ซับซ้อนมากขึ้นได้.
ตัวอย่าง:
WITH
sales_data AS (
SELECT product_id, SUM(amount) AS total_sales
FROM orders
GROUP BY product_id
),
ranked_sales AS (
SELECT product_id, total_sales,
RANK() OVER (ORDER BY total_sales DESC) AS rank
FROM sales_data
)
SELECT product_id, total_sales
FROM ranked_sales
WHERE rank <= 5;
ใน query นี้, sales_data รวมยอดขายต่อผลิตภัณฑ์, และ ranked_sales กำหนดอันดับตามปริมาณการขาย. สุดท้าย, ผลิตภัณฑ์ห้าตัวแรกจะถูกดึงออก.
จุดสำคัญสำหรับการใช้งานจริง
- คิดเป็นขั้นตอนเชิงตรรกะ สร้าง CTEs ทีละขั้นตอนเพื่อปรับปรุงความอ่านง่ายและทำให้การดีบักง่ายขึ้น.
- เก็บผลลัพธ์การคำนวณระหว่างขั้นตอน รวมผลลัพธ์การคำนวณหรือเงื่อนไขการกรองที่ใช้หลายครั้งไว้ใน CTEs เพื่อลดการทำซ้ำของโค้ด.
- ระมัดระวังกับชุดข้อมูลขนาดใหญ่ เนื่องจาก CTEs สร้างชุดผลลัพธ์ชั่วคราว, ควรพิจารณาผลกระทบต่อประสิทธิภาพเมื่อทำงานกับข้อมูลจำนวนมาก.
4. ตัวอย่างการใช้งานจริงของ Recursive WITH Clauses
Recursive WITH Clause คืออะไร?
Recursive WITH clause (recursive CTE) คือวิธีที่ใช้ Common Table Expression เพื่อทำการเรียก query ที่อ้างอิงตัวเองซ้ำ ๆ, ทำให้คุณสามารถประมวลผลข้อมูลแบบลำดับชั้นและการคำนวณแบบวนซ้ำได้. Recursive CTEs รองรับใน MySQL 8.0 ขึ้นไปและมีประโยชน์อย่างยิ่งเมื่อทำงานกับความสัมพันธ์แบบพ่อแม่‑ลูกและโครงสร้างแบบลำดับชั้น.
ไวยากรณ์พื้นฐานของ Recursive CTE
เพื่อกำหนด Recursive CTE, ใช้คีย์เวิร์ด WITH RECURSIVE. ไวยากรณ์พื้นฐานเป็นดังต่อไปนี้:
WITH RECURSIVE recursive_table_name AS (
initial_query -- starting point of the recursion
UNION ALL
recursive_query -- query called recursively
)
SELECT * FROM recursive_table_name;
- Initial query : ดึงชุดข้อมูลแรกสำหรับกระบวนการ recursive.
- Recursive query : สร้างแถวใหม่โดยอิงจาก initial query หรือผลลัพธ์ของการวนซ้ำก่อนหน้า.
- UNION ALL : รวมผลลัพธ์ของ initial query และ recursive query.
ตัวอย่าง: การประมวลผลข้อมูลแบบลำดับชั้น
Recursive CTEs มักใช้เพื่อขยายข้อมูลที่มีโครงสร้างแบบลำดับชั้น (เช่น ต้นไม้ขององค์กรหรือหมวดหมู่).
ตัวอย่าง: การขยายโครงสร้างการจัดการพนักงาน
พิจารณาตาราง employees ต่อไปนี้:
| employee_id | name | manager_id |
|---|---|---|
| 1 | Alice | NULL |
| 2 | Bob | 1 |
| 3 | Charlie | 1 |
| 4 | David | 2 |
โดยใช้ข้อมูลนี้, เราสามารถสร้าง query ที่ดึงโครงสร้างลำดับชั้นทั้งหมดเริ่มจากพนักงานที่กำหนด.
WITH RECURSIVE employee_hierarchy AS (
-- Initial query: get top-level employees
SELECT employee_id, name, manager_id, 1 AS level
FROM employees
WHERE manager_id IS NULL
UNION ALL
-- Recursive query: get direct reports
SELECT e.employee_id, e.name, e.manager_id, eh.level + 1
FROM employees e
INNER JOIN employee_hierarchy eh
ON e.manager_id = eh.employee_id
)
SELECT * FROM employee_hierarchy;
ผลลัพธ์:
| employee_id | name | manager_id | level |
|---|---|---|---|
| 1 | Alice | NULL | 1 |
| 2 | Bob | 1 | 2 |
| 3 | Charlie | 1 | 2 |
| 4 | David | 2 | 3 |
ใน query นี้, พนักงานที่อยู่ภายใต้จะถูกค้นหาแบบ recursive ตาม manager_id, ขยายโครงสร้างลำดับชั้นทั้งหมด.
ข้อจำกัดและข้อควรระวังสำหรับ Recursive CTEs
- ต้องมีเงื่อนไขสิ้นสุด หาก recursive query ไม่ตรงกับเงื่อนไขสิ้นสุด, อาจเกิดลูปไม่สิ้นสุด. ควรใส่เงื่อนไขที่เหมาะสมเสมอเพื่อป้องกันการ recursion ไม่สิ้นสุด.
- ผลกระทบต่อประสิทธิภาพ Recursive CTEs อาจต้องคำนวณหลายครั้งบนชุดข้อมูลขนาดใหญ่, ซึ่งอาจทำให้เวลาในการดำเนินการเพิ่มขึ้น. ใช้เงื่อนไข
LIMITและการกรองเพื่อเพิ่มประสิทธิภาพ. - ขีดจำกัดความลึกของ Recursion MySQL มีขีดจำกัดความลึกของ recursion, ดังนั้นควรระมัดระวังเมื่อทำการประมวลผลที่ลึกมาก. ขีดจำกัดนี้สามารถตั้งค่าได้ด้วยพารามิเตอร์
max_recursive_iterations.
สถานการณ์ที่ CTE แบบ Recursive มีประโยชน์
- การสำรวจโครงสร้างโฟลเดอร์ : ค้นหาโฟลเดอร์และโฟลเดอร์ย่อยอย่างเป็นลำดับชั้น.
- การสร้างแผนผังองค์กร : แสดงภาพลำดับชั้นจากผู้จัดการถึงผู้ใต้บังคับบัญชา.
- การแสดงต้นไม้ของหมวดหมู่ : ดึงข้อมูลหมวดหมู่ผลิตภัณฑ์หรือโครงสร้างแท็กแบบลำดับชั้น.
CTE แบบ Recursive เป็นวิธีที่ทรงพลังในการเขียนคำสั่ง SQL ที่กระชับสำหรับสถานการณ์เหล่านี้พร้อมกับเพิ่มความอ่านง่าย.
5. หมายเหตุและข้อพิจารณาเมื่อใช้ WITH Clause
ผลกระทบต่อประสิทธิภาพและการปรับแต่ง
- การคำนวณใหม่ของ CTE โดยทั่วไป CTE ที่กำหนดด้วย WITH clause จะถูกคำนวณใหม่ทุกครั้งที่มีการอ้างอิง ดังนั้นการใช้ CTE เดียวกันหลายครั้งอาจทำให้เวลาในการดำเนินการของคิวรีเพิ่มขึ้น ตัวอย่าง:
WITH sales AS ( SELECT product_id, SUM(amount) AS total_sales FROM orders GROUP BY product_id ) SELECT * FROM sales WHERE total_sales > 1000; SELECT COUNT(*) FROM sales;
ในกรณีข้างต้น sales ถูกอ้างอิงสองครั้ง ดังนั้นจึงคำนวณสองครั้ง เพื่อหลีกเลี่ยงนี้ เมื่อคุณต้องการอ้างอิงผลลัพธ์หลายครั้ง การเก็บผลลัพธ์ไว้ในตารางชั่วคราวอาจมีประสิทธิภาพ
วิธีแก้ไข:
CREATE TEMPORARY TABLE temp_sales AS
SELECT product_id, SUM(amount) AS total_sales
FROM orders
GROUP BY product_id;
SELECT * FROM temp_sales WHERE total_sales > 1000;
SELECT COUNT(*) FROM temp_sales;
- การแยก CTE ที่ซับซ้อน หาก WITH clause มีการซ้อนกันลึกเกินไป คำสั่งทั้งหมดอาจซับซ้อนและยากต่อการดีบัก การแยกตรรกะอย่างเหมาะสมเป็นสิ่งสำคัญเพื่อให้การประมวลผลภายใน CTE เดียวไม่ซับซ้อนเกินไป.
การใช้ WITH Clause กับชุดข้อมูลขนาดใหญ่
WITH clause สร้างชุดข้อมูลชั่วคราวระหว่างการดำเนินการ เมื่อจัดการข้อมูลจำนวนมาก สิ่งนี้อาจเป็นภาระต่อหน่วยความจำหรือพื้นที่จัดเก็บ
มาตรการแก้ไข:
- กรองข้อมูลด้วย WHERE Clause ลดการคำนวณโดยกรองข้อมูลที่ไม่จำเป็นภายใน CTE.
WITH filtered_orders AS ( SELECT * FROM orders WHERE order_date > '2023-01-01' ) SELECT customer_id, SUM(amount) FROM filtered_orders GROUP BY customer_id;
- ใช้ LIMIT Clause หากชุดข้อมูลใหญ่ ให้ใช้
LIMITเพื่อดึงข้อมูลที่ต้องการเท่านั้น.
ความเข้ากันได้ของเวอร์ชัน MySQL
WITH clause ใน MySQL รองรับตั้งแต่ MySQL 8.0 ขึ้นไป เนื่องจากเวอร์ชันก่อนหน้านี้ไม่รองรับ WITH clause คุณจึงต้องพิจารณาทางเลือกอื่น
ทางเลือก:
- ใช้ Subqueries ใช้ subqueries โดยตรงแทน WITH clause.
SELECT AVG(total_sales) FROM ( SELECT customer_id, SUM(amount) AS total_sales FROM orders GROUP BY customer_id ) AS sales;
- สร้าง View หากต้องการคิวรีที่ใช้ซ้ำได้ การใช้ view จะมีประสิทธิภาพ.
CREATE VIEW sales_view AS SELECT customer_id, SUM(amount) AS total_sales FROM orders GROUP BY customer_id; SELECT AVG(total_sales) FROM sales_view;
วิธีใช้ WITH Clause อย่างถูกต้อง
- ให้ความสำคัญกับความอ่านง่าย จุดประสงค์ของ WITH clause คือการจัดระเบียบคิวรีและเพิ่มความอ่านง่าย การใช้มากเกินไปอาจทำให้คิวรีซับซ้อนขึ้น ดังนั้นใช้เฉพาะเมื่อจำเป็น.
- ตรวจสอบประสิทธิภาพ ตรวจสอบแผนการดำเนินการ (คำสั่ง
EXPLAIN) และพิจารณาวิธีปรับประสิทธิภาพ.EXPLAIN WITH sales AS ( SELECT product_id, SUM(amount) AS total_sales FROM orders GROUP BY product_id ) SELECT * FROM sales WHERE total_sales > 1000;
6. กรณีการใช้งานจริงในสถานการณ์โลกจริง
การสรุปข้อมูลการขาย
ต่อไปนี้เป็นตัวอย่างการสรุปข้อมูลการขายตามเดือนและใช้ผลลัพธ์นั้นเพื่อคำนวณค่าเฉลี่ยการขายต่อเดือน.
ตัวอย่าง: การสรุปยอดขายต่อเดือนและคำนวณค่าเฉลี่ย
WITH monthly_sales AS (
SELECT
DATE_FORMAT(order_date, '%Y-%m') AS sales_month,
SUM(amount) AS total_sales
FROM orders
GROUP BY sales_month
)
SELECT
sales_month,
total_sales,
AVG(total_sales) OVER () AS average_sales
FROM monthly_sales;
ในคำสั่งค้นหานี้ monthly_sales คำนวณยอดขายต่อเดือน และจากผลลัพธ์นั้น คำนวณยอดขายเฉลี่ยโดยรวม ซึ่งช่วยให้ข้อมูลจัดระเบียบได้ชัดเจนและทำให้การวิเคราะห์ง่ายขึ้น
การกรองข้อมูลตามเงื่อนไขเฉพาะ
โดยการแยกตรรกะการกรองที่ซับซ้อนออกเป็น WITH clause จะช่วยปรับปรุงความอ่านง่าย
ตัวอย่าง: สร้างรายการลูกค้าที่ใช้จ่ายสูง
WITH customer_totals AS (
SELECT
customer_id,
SUM(amount) AS total_spent
FROM orders
GROUP BY customer_id
)
SELECT
customer_id,
total_spent
FROM customer_totals
WHERE total_spent > 100000;
ในคำสั่งค้นหานี้ customer_totals คำนวณยอดรวมการซื้อต่อลูกค้า และดึงลูกค้าที่ตรงตามเงื่อนไขที่กำหนด
การวิเคราะห์ข้อมูลลำดับชั้น
เมื่อวิเคราะห์ข้อมูลลำดับชั้น เช่น โครงสร้างองค์กรหรือหมวดหมู่ WITH clause แบบ recursive มีประโยชน์อย่างยิ่ง
ตัวอย่าง: ดึงรายการผู้ใต้บังคับบัญชาโดยตรงและทางอ้อม
WITH RECURSIVE employee_hierarchy AS (
SELECT
employee_id,
name,
manager_id,
1 AS level
FROM employees
WHERE manager_id IS NULL
UNION ALL
SELECT
e.employee_id,
e.name,
e.manager_id,
eh.level + 1
FROM employees e
INNER JOIN employee_hierarchy eh
ON e.manager_id = eh.employee_id
)
SELECT
employee_id,
name,
manager_id,
level
FROM employee_hierarchy
ORDER BY level, manager_id;
คำสั่งค้นหานี้สร้างข้อมูลลำดับชั้นใน employee_hierarchy และดึงพนักงานที่จัดกลุ่มตามระดับ ซึ่งช่วยให้สร้างข้อมูลแผนผังองค์กรแบบไดนามิก
การวิเคราะห์ขั้นสูงโดยใช้ CTE หลายตัว
โดยการใช้ WITH clause หลายตัว ข้อมูลสามารถประมวลผลทีละขั้นตอน ทำให้การวิเคราะห์ที่ซับซ้อนง่ายขึ้น
ตัวอย่าง: ดึงผลิตภัณฑ์ขายดีที่สุดต่อหมวดหมู่
WITH category_sales AS (
SELECT
category_id,
product_id,
SUM(amount) AS total_sales
FROM orders
GROUP BY category_id, product_id
),
ranked_sales AS (
SELECT
category_id,
product_id,
total_sales,
RANK() OVER (PARTITION BY category_id ORDER BY total_sales DESC) AS rank
FROM category_sales
)
SELECT
category_id,
product_id,
total_sales
FROM ranked_sales
WHERE rank <= 3;
ในคำสั่งค้นหานี้ คำนวณยอดขายต่อหมวดหมู่ และดึงสามผลิตภัณฑ์อันดับต้น ๆ ในแต่ละหมวดหมู่ วิธีนี้มีประสิทธิภาพเมื่อต้องการจำกัดข้อมูลตามเงื่อนไขเฉพาะ
ประเด็นสำคัญสำหรับการใช้งานจริง
- ออกแบบคำสั่งค้นหาในขั้นตอนเชิงตรรกะ ใช้ WITH clause เพื่อแบ่งคำสั่งค้นหาและประมวลผลข้อมูลทีละขั้นตอนโดยรักษาความอ่านง่าย
- ดึงเฉพาะข้อมูลที่จำเป็น ใช้ WHERE และ LIMIT clause เพื่อหลีกเลี่ยงการประมวลผลข้อมูลที่ไม่จำเป็นและออกแบบคำสั่งค้นหาที่มีประสิทธิภาพ
- การใช้งานทางธุรกิจที่ยืดหยุ่น WITH clause สามารถนำไปใช้อย่างยืดหยุ่นกับการวิเคราะห์ยอดขาย การแบ่งกลุ่มลูกค้า การจัดการสินค้าคงคลัง และอื่น ๆ

7. คำถามที่พบบ่อย (FAQ)
Q1: เมื่อใดควรใช้ WITH clause?
A1:
WITH clause มีประสิทธิภาพโดยเฉพาะในสถานการณ์ต่อไปนี้:
- เมื่อต้องการทำให้ subquery ที่ซับซ้อนง่ายขึ้น
- เมื่อต้องการใช้ชุดข้อมูลเดียวกันหลายครั้งในคำสั่งค้นหา
- เมื่อต้องการแบ่งคำสั่งค้นหาเชิงตรรกะเพื่อปรับปรุงความอ่านง่าย
ตัวอย่างเช่น ในคำสั่งค้นหาที่ใช้ผลรวม aggregation เดียวกันหลายครั้ง WITH clause ช่วยให้จัดระเบียบได้มีประสิทธิภาพมากขึ้น
Q2: เมื่อใดที่ recursive CTE มีประโยชน์?
A2:
Recursive CTE มีประโยชน์เมื่อจัดการโครงสร้างลำดับชั้นหรือการคำนวณแบบวนซ้ำ โดยเฉพาะ:
- การประมวลผลข้อมูลลำดับชั้น (เช่น ต้นไม้องค์กร โครงสร้างหมวดหมู่)
- การแสดงลำดับชั้นโฟลเดอร์หรือไฟล์
- การคำนวณลำดับของตัวเลขหรือช่วงเวลา (เช่น การคำนวณลำดับฟีโบนัชชี)
การใช้ recursive CTE ทำให้ขยายและประมวลผลข้อมูลที่อ้างอิงตัวเองได้ง่ายขึ้น
Q3: คำสั่ง WITH มีประสิทธิภาพมากกว่าการใช้ view หรือไม่?
A3:
ขึ้นอยู่กับกรณีการใช้งาน
- WITH clause : สร้างชุดผลลัพธ์ชั่วคราวที่ใช้เฉพาะภายใน query เท่านั้น เหมาะกับข้อมูลที่ไม่ต้องการนำกลับมาใช้บ่อย
- View : ถูกเก็บอย่างถาวรในฐานข้อมูลและสามารถนำกลับมาใช้โดย query อื่นได้ เหมาะกับ query ที่ต้องใช้ซ้ำหลายครั้ง
การเลือกวิธีที่เหมาะสมตามสถานการณ์เป็นสิ่งสำคัญ
Q4: สาเหตุใดที่ทำให้ประสิทธิภาพลดลงเมื่อใช้ WITH clause?
A4:
สาเหตุหลักของการลดประสิทธิภาพเมื่อใช้ WITH clause มีดังนี้
- CTE recalculation : ผลลัพธ์จะถูกคำนวณใหม่ทุกครั้งที่อ้างอิง ทำให้เวลาในการประมวลผลเพิ่มขึ้น
- Handling large datasets : การสร้างชุดข้อมูลขนาดใหญ่ภายใน CTE ทำให้ใช้หน่วยความจำมากขึ้นและประสิทธิภาพลดลง
- Lack of proper indexing : หาก query ภายใน CTE ไม่ใช้ดัชนีที่เหมาะสม ประสิทธิภาพอาจช้าลง
แนวทางแก้ไข
- พิจารณาใช้ temporary tables หรือ view หากต้องการนำข้อมูลกลับมาใช้บ่อย
- ใช้เงื่อนไข WHERE และ LIMIT เพื่อจำกัดข้อมูลให้แคบลงอย่างเหมาะสม
Q5: มีทางเลือกใดสำหรับ MySQL เวอร์ชันที่ไม่รองรับ WITH clause?
A5:
ในเวอร์ชันก่อน MySQL 8.0 ไม่รองรับ WITH clause จึงต้องใช้ทางเลือกต่อไปนี้
- Use Subqueries ใช้ subquery โดยตรงแทนการใช้ WITH clause
SELECT AVG(total_sales) FROM ( SELECT customer_id, SUM(amount) AS total_sales FROM orders GROUP BY customer_id ) AS sales;
- Use Temporary Tables เก็บชุดข้อมูลที่ต้องการนำกลับมาใช้ซ้ำใน temporary table
CREATE TEMPORARY TABLE temp_sales AS SELECT customer_id, SUM(amount) AS total_sales FROM orders GROUP BY customer_id; SELECT AVG(total_sales) FROM temp_sales;
Q6: แนวปฏิบัติที่ดีที่สุดเมื่อใช้ WITH clause คืออะไร?
A6:
ควรคำนึงถึงแนวปฏิบัติที่ดีต่อไปนี้
- Prioritize simplicity : อย่าบังคับให้ตรรกะซับซ้อนอยู่ใน WITH clause เดียว ควรแยกออกเป็นส่วนที่เหมาะสม
- Verify performance : ตรวจสอบแผนการทำงานด้วยคำสั่ง
EXPLAINและปรับแต่งตามความจำเป็น - Consider reusability : หากต้องการนำข้อมูลกลับมาใช้บ่อย ควรใช้ view หรือ temporary tables
8. Conclusion
บทความนี้ครอบคลุม WITH clause (Common Table Expression, CTE) ที่แนะนำใน MySQL 8.0 ตั้งแต่พื้นฐานจนถึงการใช้งานขั้นสูง WITH clause เป็นฟีเจอร์ที่มีประโยชน์อย่างยิ่งสำหรับทำให้ query ที่ซับซ้อนสั้นลงและอ่านง่ายขึ้น ด้านล่างนี้คือประเด็นสำคัญที่ควรจำ
Main Benefits of the WITH Clause
- Improved Query Readability จัดระเบียบ subquery ที่ซับซ้อน ทำให้โค้ด SQL อ่านง่ายและบำรุงรักษาได้ดี
- Query Reusability ประมวลผลข้อมูลอย่างมีประสิทธิภาพเมื่ออ้างอิงชุดข้อมูลเดียวกันหลายครั้ง
- Enables Recursive Data Operations CTE แบบ recursive ช่วยให้การประมวลผลข้อมูลเชิงลำดับชั้นและการคำนวณแบบวนซ้ำทำได้ง่ายขึ้น
Practical Usage Points
- เหมาะสำหรับ การวิเคราะห์ข้อมูลการขายและลูกค้า โดยสามารถทำการสรุปผลเป็นขั้นตอนได้
- CTE แบบ recursive มีประโยชน์สำหรับ การประมวลผลข้อมูลเชิงลำดับชั้น (เช่น โครงสร้างองค์กรหรือหมวดหมู่สินค้า)
- การผสานรวม WITH clause กับ view หรือ temporary tables ทำให้การดำเนินงานฐานข้อมูลมีความยืดหยุ่นและมีประสิทธิภาพ
Important Considerations
- แม้ WITH clause จะมีพลังมาก แต่การใช้ไม่เหมาะสมอาจทำให้ประสิทธิภาพลดลง
- ควรประเมินความต้องการนำข้อมูลกลับมาใช้ซ้ำและประสิทธิภาพในแต่ละกรณี และเลือกใช้ view หรือ temporary tables ตามความเหมาะสม
- ตรวจสอบประสิทธิภาพของ query เสมอด้วยแผนการทำงาน (
EXPLAINcommand)
Next Steps
การใช้ WITH clause จะช่วยให้คุณสร้าง query ที่มีประสิทธิภาพและบำรุงรักษาได้ง่ายขึ้น ลองนำไปใช้ในโครงการจริงตามขั้นตอนต่อไปนี้
- เริ่มจาก query ง่าย ๆ และฝึกจัดโครงสร้างด้วย WITH clause
- ท้าทายตัวเองด้วย CTE แบบ recursive สำหรับข้อมูลเชิงลำดับชั้นและสถานการณ์ที่ซับซ้อน
- มุ่งเน้นการปรับประสิทธิภาพเพื่อพัฒนาทักษะ SQL ของคุณต่อไป
บทความนี้จบแล้ว. ใช้ความรู้ของคุณเกี่ยวกับคำสั่ง WITH ของ MySQL ในการทำงานและการศึกษาในชีวิตประจำวันของคุณ.


