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 ช่วงและตัวห้อยอยู่นอกช่วงนี้