PL / SQL - คอลเล็กชัน
ในบทนี้เราจะพูดถึง Collections ใน PL / SQL คอลเลกชันคือกลุ่มขององค์ประกอบที่เรียงลำดับซึ่งมีชนิดข้อมูลเดียวกัน แต่ละองค์ประกอบถูกระบุโดยตัวห้อยที่ไม่ซ้ำกันซึ่งแสดงถึงตำแหน่งในคอลเล็กชัน
PL / SQL มีคอลเลกชันสามประเภท -
- ตารางดัชนีหรืออาร์เรย์ Associative
- ตารางที่ซ้อนกัน
- อาร์เรย์ขนาดตัวแปรหรือ Varray
เอกสาร Oracle มีคุณสมบัติดังต่อไปนี้สำหรับคอลเล็กชันแต่ละประเภท -
ประเภทคอลเลกชัน | จำนวนองค์ประกอบ | ประเภทตัวติดตาม | หนาแน่นหรือเบาบาง | สร้างที่ไหน | สามารถเป็นแอตทริบิวต์ประเภทวัตถุ |
---|---|---|---|---|---|
อาร์เรย์เชื่อมโยง (หรือตารางดัชนี) | ไม่มีขอบเขต | สตริงหรือจำนวนเต็ม | ทั้ง | เฉพาะในบล็อก PL / SQL | ไม่ |
ตารางที่ซ้อนกัน | ไม่มีขอบเขต | จำนวนเต็ม | เริ่มหนาแน่นอาจเบาบางลง | ทั้งในบล็อก PL / SQL หรือที่ระดับสคีมา | ใช่ |
ตัวแปรขนาดอาร์เรย์ (Varray) | ผูกมัด | จำนวนเต็ม | หนาแน่นเสมอ | ทั้งในบล็อก PL / SQL หรือที่ระดับสคีมา | ใช่ |
เราได้กล่าวถึง varray ในบทนี้แล้ว 'PL/SQL arrays'. ในบทนี้เราจะพูดถึงตาราง PL / SQL
ตาราง PL / SQL ทั้งสองประเภท ได้แก่ ตารางดัชนีโดยและตารางที่ซ้อนกันมีโครงสร้างเหมือนกันและมีการเข้าถึงแถวโดยใช้สัญกรณ์ตัวห้อย อย่างไรก็ตามตารางทั้งสองประเภทนี้แตกต่างกันในแง่มุมหนึ่ง ตารางที่ซ้อนกันสามารถเก็บไว้ในคอลัมน์ฐานข้อมูลและตารางดัชนีโดยไม่สามารถ
ตารางดัชนีโดย
อัน index-by ตาราง (เรียกอีกอย่างว่า associative array) เป็นชุดของ key-valueคู่ แต่ละคีย์ไม่ซ้ำกันและใช้เพื่อค้นหาค่าที่เกี่ยวข้อง คีย์อาจเป็นจำนวนเต็มหรือสตริงก็ได้
ตารางดัชนีโดยสร้างขึ้นโดยใช้ไวยากรณ์ต่อไปนี้ ที่นี่เรากำลังสร้างไฟล์index-by ตารางชื่อ table_nameซึ่งคีย์จะเป็น subscript_type และค่าที่เกี่ยวข้องจะเป็นของelement_type
TYPE type_name IS TABLE OF element_type [NOT NULL] INDEX BY subscript_type;
table_name type_name;
ตัวอย่าง
ตัวอย่างต่อไปนี้แสดงวิธีสร้างตารางเพื่อจัดเก็บค่าจำนวนเต็มพร้อมกับชื่อและหลังจากนั้นจะพิมพ์รายชื่อเดียวกัน
DECLARE
TYPE salary IS TABLE OF NUMBER INDEX BY VARCHAR2(20);
salary_list salary;
name VARCHAR2(20);
BEGIN
-- adding elements to the table
salary_list('Rajnish') := 62000;
salary_list('Minakshi') := 75000;
salary_list('Martin') := 100000;
salary_list('James') := 78000;
-- printing the table
name := salary_list.FIRST;
WHILE name IS NOT null LOOP
dbms_output.put_line
('Salary of ' || name || ' is ' || TO_CHAR(salary_list(name)));
name := salary_list.NEXT(name);
END LOOP;
END;
/
เมื่อรหัสด้านบนถูกเรียกใช้ที่พรอมต์ SQL จะให้ผลลัพธ์ดังต่อไปนี้ -
Salary of James is 78000
Salary of Martin is 100000
Salary of Minakshi is 75000
Salary of Rajnish is 62000
PL/SQL procedure successfully completed.
ตัวอย่าง
องค์ประกอบของตารางดัชนีโดยอาจเป็นไฟล์ %ROWTYPE ของตารางฐานข้อมูลหรือ %TYPEของฟิลด์ตารางฐานข้อมูลใด ๆ ตัวอย่างต่อไปนี้แสดงให้เห็นถึงแนวคิด เราจะใช้ไฟล์CUSTOMERS ตารางที่จัดเก็บในฐานข้อมูลของเราเป็น -
Select * from customers;
+----+----------+-----+-----------+----------+
| ID | NAME | AGE | ADDRESS | SALARY |
+----+----------+-----+-----------+----------+
| 1 | Ramesh | 32 | Ahmedabad | 2000.00 |
| 2 | Khilan | 25 | Delhi | 1500.00 |
| 3 | kaushik | 23 | Kota | 2000.00 |
| 4 | Chaitali | 25 | Mumbai | 6500.00 |
| 5 | Hardik | 27 | Bhopal | 8500.00 |
| 6 | Komal | 22 | MP | 4500.00 |
+----+----------+-----+-----------+----------+
DECLARE
CURSOR c_customers is
select name from customers;
TYPE c_list IS TABLE of customers.Name%type INDEX BY binary_integer;
name_list c_list;
counter integer :=0;
BEGIN
FOR n IN c_customers LOOP
counter := counter +1;
name_list(counter) := n.name;
dbms_output.put_line('Customer('||counter||'):'||name_lis t(counter));
END LOOP;
END;
/
เมื่อรหัสด้านบนถูกเรียกใช้ที่พรอมต์ SQL จะให้ผลลัพธ์ดังต่อไปนี้ -
Customer(1): Ramesh
Customer(2): Khilan
Customer(3): kaushik
Customer(4): Chaitali
Customer(5): Hardik
Customer(6): Komal
PL/SQL procedure successfully completed
ตารางที่ซ้อนกัน
ก nested tableเปรียบเสมือนอาร์เรย์หนึ่งมิติที่มีองค์ประกอบตามจำนวนที่กำหนด อย่างไรก็ตามตารางที่ซ้อนกันแตกต่างจากอาร์เรย์ในแง่มุมต่อไปนี้ -
อาร์เรย์มีจำนวนองค์ประกอบที่ประกาศไว้ แต่ตารางที่ซ้อนกันไม่มี ขนาดของตารางที่ซ้อนกันสามารถเพิ่มขึ้นแบบไดนามิก
อาร์เรย์มีความหนาแน่นเสมอกล่าวคือมีตัวห้อยที่ต่อเนื่องกันเสมอ อาร์เรย์ที่ซ้อนกันมีความหนาแน่นในตอนแรก แต่อาจกระจัดกระจายเมื่อองค์ประกอบถูกลบออกไป
ตารางที่ซ้อนกันถูกสร้างขึ้นโดยใช้ไวยากรณ์ต่อไปนี้ -
TYPE type_name IS TABLE OF element_type [NOT NULL];
table_name type_name;
คำประกาศนี้คล้ายกับการประกาศ index-by โต๊ะ แต่ไม่มี INDEX BY อนุประโยค
ตารางที่ซ้อนกันสามารถจัดเก็บไว้ในคอลัมน์ฐานข้อมูล นอกจากนี้ยังสามารถใช้เพื่อลดความซับซ้อนของการดำเนินการ SQL โดยที่คุณเข้าร่วมตารางคอลัมน์เดียวกับตารางขนาดใหญ่ ไม่สามารถจัดเก็บอาร์เรย์ที่เชื่อมโยงในฐานข้อมูลได้
ตัวอย่าง
ตัวอย่างต่อไปนี้แสดงให้เห็นถึงการใช้ตารางที่ซ้อนกัน -
DECLARE
TYPE names_table IS TABLE OF VARCHAR2(10);
TYPE grades IS TABLE OF INTEGER;
names names_table;
marks grades;
total integer;
BEGIN
names := names_table('Kavita', 'Pritam', 'Ayan', 'Rishav', 'Aziz');
marks:= grades(98, 97, 78, 87, 92);
total := names.count;
dbms_output.put_line('Total '|| total || ' Students');
FOR i IN 1 .. total LOOP
dbms_output.put_line('Student:'||names(i)||', Marks:' || marks(i));
end loop;
END;
/
เมื่อรหัสด้านบนถูกเรียกใช้ที่พรอมต์ SQL จะให้ผลลัพธ์ดังต่อไปนี้ -
Total 5 Students
Student:Kavita, Marks:98
Student:Pritam, Marks:97
Student:Ayan, Marks:78
Student:Rishav, Marks:87
Student:Aziz, Marks:92
PL/SQL procedure successfully completed.
ตัวอย่าง
องค์ประกอบของไฟล์ nested table ยังสามารถเป็นไฟล์ %ROWTYPEของตารางฐานข้อมูลหรือ% TYPE ของเขตข้อมูลตารางฐานข้อมูลใด ๆ ตัวอย่างต่อไปนี้แสดงให้เห็นถึงแนวคิด เราจะใช้ตารางลูกค้าที่เก็บไว้ในฐานข้อมูลของเราเป็น -
Select * from customers;
+----+----------+-----+-----------+----------+
| ID | NAME | AGE | ADDRESS | SALARY |
+----+----------+-----+-----------+----------+
| 1 | Ramesh | 32 | Ahmedabad | 2000.00 |
| 2 | Khilan | 25 | Delhi | 1500.00 |
| 3 | kaushik | 23 | Kota | 2000.00 |
| 4 | Chaitali | 25 | Mumbai | 6500.00 |
| 5 | Hardik | 27 | Bhopal | 8500.00 |
| 6 | Komal | 22 | MP | 4500.00 |
+----+----------+-----+-----------+----------+
DECLARE
CURSOR c_customers is
SELECT name FROM customers;
TYPE c_list IS TABLE of customerS.No.ame%type;
name_list c_list := c_list();
counter integer :=0;
BEGIN
FOR n IN c_customers LOOP
counter := counter +1;
name_list.extend;
name_list(counter) := n.name;
dbms_output.put_line('Customer('||counter||'):'||name_list(counter));
END LOOP;
END;
/
เมื่อรหัสด้านบนถูกเรียกใช้ที่พรอมต์ SQL จะให้ผลลัพธ์ดังต่อไปนี้ -
Customer(1): Ramesh
Customer(2): Khilan
Customer(3): kaushik
Customer(4): Chaitali
Customer(5): Hardik
Customer(6): Komal
PL/SQL procedure successfully completed.
วิธีการรวบรวม
PL / SQL มีวิธีการรวบรวมในตัวที่ทำให้คอลเลกชันใช้งานง่ายขึ้น ตารางต่อไปนี้แสดงวิธีการและวัตถุประสงค์ -
ส. เลขที่ | ชื่อวิธีการและวัตถุประสงค์ |
---|---|
1 | EXISTS(n) ส่งคืนค่า TRUE หากองค์ประกอบที่ n ในคอลเล็กชันมีอยู่ มิฉะนั้นจะคืนค่า FALSE |
2 | COUNT ส่งคืนจำนวนองค์ประกอบที่คอลเลกชันมีอยู่ในปัจจุบัน |
3 | LIMIT ตรวจสอบขนาดสูงสุดของคอลเลกชัน |
4 | FIRST ส่งคืนตัวเลขดัชนีแรก (เล็กที่สุด) ในคอลเล็กชันที่ใช้ตัวห้อยจำนวนเต็ม |
5 | LAST ส่งคืนหมายเลขดัชนีสุดท้าย (ใหญ่ที่สุด) ในคอลเลกชันที่ใช้ตัวห้อยจำนวนเต็ม |
6 | PRIOR(n) ส่งคืนหมายเลขดัชนีที่นำหน้าดัชนี n ในคอลเล็กชัน |
7 | NEXT(n) ส่งคืนหมายเลขดัชนีที่สำเร็จดัชนี n |
8 | EXTEND ผนวกองค์ประกอบ null หนึ่งเข้ากับคอลเลกชัน |
9 | EXTEND(n) ผนวกองค์ประกอบ n null เข้ากับคอลเลกชัน |
10 | EXTEND(n,i) ต่อท้าย nสำเนาขององค์ประกอบi thไปยังคอลเลกชัน |
11 | TRIM ลบองค์ประกอบหนึ่งออกจากส่วนท้ายของคอลเลกชัน |
12 | TRIM(n) ลบ n องค์ประกอบจากส่วนท้ายของคอลเลกชัน |
13 | DELETE ลบองค์ประกอบทั้งหมดออกจากคอลเลกชันโดยตั้งค่า COUNT เป็น 0 |
14 | DELETE(n) ลบไฟล์ nthองค์ประกอบจากอาร์เรย์ที่เชื่อมโยงกับคีย์ตัวเลขหรือตารางที่ซ้อนกัน หากอาร์เรย์ที่เชื่อมโยงมีคีย์สตริงองค์ประกอบที่ตรงกับค่าคีย์จะถูกลบออก ถ้าn เป็นโมฆะ DELETE(n) ไม่ทำอะไรเลย |
15 | DELETE(m,n) ลบองค์ประกอบทั้งหมดในช่วง m..nจากอาร์เรย์ที่เชื่อมโยงหรือตารางที่ซ้อนกัน ถ้าm มีขนาดใหญ่กว่า n หรือถ้า m หรือ n เป็นโมฆะ DELETE(m,n) ไม่ทำอะไรเลย |
ข้อยกเว้นของคอลเลกชัน
ตารางต่อไปนี้ระบุข้อยกเว้นของคอลเลกชันและเวลาที่ยกขึ้น -
ข้อยกเว้นการรวบรวม | เพิ่มขึ้นในสถานการณ์ |
---|---|
COLLECTION_IS_NULL | คุณพยายามดำเนินการกับคอลเลกชันที่เป็นโมฆะอะตอม |
ไม่พบข้อมูล | ตัวห้อยกำหนดองค์ประกอบที่ถูกลบหรือองค์ประกอบที่ไม่มีอยู่ของอาร์เรย์ที่เชื่อมโยง |
SUBSCRIPT_BEYOND_COUNT | ตัวห้อยเกินจำนวนองค์ประกอบในคอลเล็กชัน |
SUBSCRIPT_OUTSIDE_LIMIT | ตัวห้อยอยู่นอกช่วงที่อนุญาต |
VALUE_ERROR | ตัวห้อยเป็นโมฆะหรือไม่สามารถแปลงเป็นประเภทคีย์ได้ ข้อยกเว้นนี้อาจเกิดขึ้นหากคีย์ถูกกำหนดเป็นไฟล์PLS_INTEGER ช่วงและตัวห้อยอยู่นอกช่วงนี้ |