Thứ Ba, 8 tháng 11, 2022

TRANSACT-SQL: VÍ DỤ MINH HỌA GROUP BY ... WITH ROLLUP

Để thống kê dữ liệu, các hệ quản trị cơ sở dữ liệu cung cấp các hàm tổng hợp dữ liệu như AVG()SUM()COUNT(),… sử dụng kết hợp với mệnh đề GROUP BY. Bài viết này giới thiệu cách sử dụng mệnh đề GROUP BY và các tùy chọn mở rộng ROLLUPCUBE với các hàm tổng hợp dữ liệu trong hệ quản trị CSDL SQL Server.

Tạo dữ liệu minh họa

Trước hết, ta cần tạo dữ liệu để minh họa cho ví dụ này. Xét một TABLE Student chứa thông tin về điểm các môn học của sinh viên:

Bảng dữ liệu minh họa

Tạo bảng trên bằng lệnh CREATE TABLE sau:

create table Student ( [name] nchar(30), [subject] nchar(10), [mark] float);

Chèn dữ liệu vào trong TABLE Student như sau:

insert into Student values('Hung', 'JAVA', 8);
insert into Student values('Hung', 'C', 9);
insert into Student values('Hung', 'C', 7);
insert into Student values('Hung', 'SQL', 5);
insert into Student values('Tuan', 'JAVA', 4);
insert into Student values('Tuan', 'C', 7);
insert into Student values('Tuan', 'SQL', 10);
insert into Student values('Tuan', 'SQL', 5);

Để kiểm tra dữ liệu đã được thêm vào đủ chưa, ta chạy lệnh sau:

SELECT [name] AS [Student name], [subject] AS [Subject name], [mark] AS [Mark] FROM Student;

Khi đó, ta thu được kết quả giống như hình minh họa ở trên.

Thống kê dữ liệu

Sử dụng mệnh đề GROUP BY cùng với hàm tính giá trị trung bình AVG, ta có thể tính điểm trung bình theo môn của từng sinh viên. Gõ lệnh SELECT (1) như sau:

SELECT [name] as [Student name], [subject] as [Subject name], 
    AVG([mark])  as [Average mark] FROM Student GROUP BY [name], [subject];

Lệnh này sẽ trả lại kết quả như sau:

Tính giá trị trung bình với Group By

Cột bên thứ 3 (không có tên) chứa điểm trung bình từng môn của sinh viên.

Trong việc quản lý sinh viên, người dùng còn có nhu cầu tính điểm trung bình tất cả các môn của mỗi sinh viên và điểm trung bình chung của tất cả sinh viên như sau:

Tính giá trị trung bình với Group By ... WITH ROLLUP

Khi đó, ta sử dụng thêm các tùy chọn ROLLUP và CUBE vào trong mệnh đề GROUP BY. Thay lệnh SELECT (1) ở trên bằng lệnh SELECT (2) có thêm tùy chọn ROLLUP như sau:

SELECT [name] as [Student name], [subject] as [Subject name], AVG([mark]) as [Average mark] 
    FROM Student GROUP BY [name], [subject] WITH ROLLUP;

Ta thu được kết quả:

Tính giá trị trung bình với Group By ... WITH ROLLUP (kết quả thô)

Trong kết quả trên, tùy chọn ROLLUP đã chèn thêm các dòng tính điểm trung bình tất cả môn học của từng sinh viên. Tại ô tên môn học, ROLLUP cho giá trị NULL. Ta có thể sửa các giá trị NULL này thành tên gọi mang ý nghĩa tại các ô đó bằng cách sử dụng Hàm GROUPING() để xác định xem trường (FIELD) [name] hoặc [subject] có phải hiện đang được nhóm hay không với lệnh SELECT (3) như sau:

SELECT 
   CASE when GROUPING([name]) = 1 THEN 'All students' ELSE [name] END 
      AS [Student name],
   CASE when GROUPING([subject]) = 1 THEN 'All subjects' ELSE [subject] END
      AS [Subject name], 
   CAST(AVG([mark]) AS DECIMAL(9, 2)) AS [Average mark] 
   FROM Student GROUP BY [name], [subject] WITH ROLLUP;

Khi đó, ta sẽ nhận được kết quả như mong muốn:

Tính giá trị trung bình với Group By ... WITH ROLLUP (kết quả cuối)

Ta có nhận xét là, trong kết quả trên, ta chỉ tính được điểm trung bình từng môn của từng sinh viên cũng như điểm trung bình tất cả các môn của từng sinh viên, (tức là tính trung bình theo sinh viên), nhưng không có điểm trung bình của tất cả sinh viên cho từng môn.(chẳng hạn điểm trung bình của tất cả sinh viên trong môn lập trình C). Để tính loại điểm trung bình thứ 3, ta có thể đảo ngược vị trí của [name] và [subject] trong mệnh đề GROUP BY như sau:

SELECT [subject] as [Subject name], [name] as [Student name], AVG([mark]) 
     FROM Student GROUP BY [subject], [name] WITH ROLLUP;

Tuy nhiên, để kết quả có cả 3 loại điểm trung bình trên, tùy chọn ROLLUP không đáp ứng được, mà ta cần sử dụng tùy chọn CUBE. Thay thế tùy chọn CUBE cho ROLLUP trong lệnh SELECT (3), ta có lệnh SELECT (4):

SELECT 
   CASE when GROUPING([name]) = 1 THEN 'All students' ELSE [name] END 
      AS [Student name],
   CASE when GROUPING([subject]) = 1 THEN 'All subjects' ELSE [subject] END
      AS [Subject name], 
   CAST(AVG([mark]) AS DECIMAL(9, 2)) AS [Average mark] 
   FROM Student GROUP BY [name], [subject] WITH CUBE;

Khi chạy lệnh này, ta thu được kết quả:

Tính giá trị trung bình với Group By ... WITH CUBE

=============================
* KHOÁ HỌC ORACLE DATABASE A-Z ENTERPRISE trực tiếp từ tôi giúp bạn bước đầu trở thành những chuyên gia DBA, đủ kinh nghiệm đi thi chứng chỉ OA/OCP, đặc biệt là rất nhiều kinh nghiệm, bí kíp thực chiến trên các hệ thống Core tại VN chỉ sau 1 khoá học.
* CÁCH ĐĂNG KÝ: Gõ (.) hoặc để lại số điện thoại hoặc inbox https://m.me/tranvanbinh.vn hoặc Hotline/Zalo 090.29.12.888
* Chi tiết tham khảo:
https://bit.ly/oaz_w
=============================
KẾT NỐI VỚI CHUYÊN GIA TRẦN VĂN BÌNH:
📧 Mail: binhoracle@gmail.com
☎️ Mobile: 0902912888
⚡️ Skype: tranbinh48ca
👨 Facebook: https://www.facebook.com/BinhOracleMaster
👨 Inbox Messenger: https://m.me/101036604657441 (profile)
👨 Fanpage: https://www.facebook.com/tranvanbinh.vn
👨 Inbox Fanpage: https://m.me/tranvanbinh.vn
👨👩 Group FB: https://www.facebook.com/groups/DBAVietNam
👨 Website: https://www.tranvanbinh.vn
👨 Blogger: https://tranvanbinhmaster.blogspot.com
🎬 Youtube: http://bit.ly/ytb_binhoraclemaster
👨 Tiktok: https://www.tiktok.com/@binhoraclemaster?lang=vi
👨 Linkin: https://www.linkedin.com/in/binhoracle
👨 Twitter: https://twitter.com/binhoracle
👨 Địa chỉ: Tòa nhà Sun Square - 21 Lê Đức Thọ - Phường Mỹ Đình 1 - Quận Nam Từ Liêm - TP.Hà Nội

=============================
học oracle database, Tự học Oracle, Tài liệu Oracle 12c tiếng Việt, Hướng dẫn sử dụng Oracle Database, Oracle SQL cơ bản, Oracle SQL là gì, Khóa học Oracle Hà Nội, Học chứng chỉ Oracle ở đầu, Khóa học Oracle online,khóa học pl/sql, học dba, học dba ở việt nam, khóa học dba, khóa học dba sql, tài liệu học dba oracle, Khóa học Oracle online, học oracle sql, học oracle ở đâu tphcm, học oracle bắt đầu từ đâu, học oracle ở hà nội, oracle database tutorial, oracle database 12c, oracle database là gì, oracle database 11g, oracle download, oracle database 19c, oracle dba tutorial, oracle tunning, sql tunning , oracle 12c, oracle multitenant, Container Databases (CDB), Pluggable Databases (PDB), oracle cloud, oracle security, oracle fga, audit_trail, oracle dataguard, oracle goldengate, mview, oracle exadata, oracle oca, oracle ocp, oracle ocm , oracle weblogic, middleware, hoc solaris, hoc linux, hoc aix, unix, securecrt, xshell, mobaxterm, putty

ĐỌC NHIỀU

Trần Văn Bình - Oracle Database Master