विभाजन। स्वचालित रूप से अनुभाग जोड़ें

संस्करण 11 जी में, ओरेकल ने कई रोमांचक नई विभाजन योजनाओं को पेश किया - उदाहरण के लिए, अंतराल विभाजन का सुविधाजनक कार्य - विभाजन को सीमा से बाहर जाने के लिए स्वचालित रूप से विभाजन बनाने के लिए।

11g से पहले के संस्करणों में, आपको समय-समय पर मैन्युअल रूप से या तो अनुभागों को अग्रिम में जोड़ना होगा या डिफ़ॉल्ट अनुभाग को विभाजित करना होगा। यही है, ऐसी तालिकाओं की स्थिति की निगरानी करना लगातार आवश्यक है। इस लेख में, मैं इस तरह के विभाजन कार्यों को स्वचालित करने के लिए अपने समाधान साझा करूंगा।

पहले मैं 11 जी के लिए एक उदाहरण देता हूं:

  1. create table res ( res_id number not null , res_date date , hotel_id number(3), guest_id number ) partition by range (res_id) interval (100) store in (users) ( partition p1 values less than (101) );



  2. create table res ( res_id number not null , res_date date , hotel_id number(3), guest_id number ) partition by range (res_id) interval (100) store in (users) ( partition p1 values less than (101) );



  3. create table res ( res_id number not null , res_date date , hotel_id number(3), guest_id number ) partition by range (res_id) interval (100) store in (users) ( partition p1 values less than (101) );



  4. create table res ( res_id number not null , res_date date , hotel_id number(3), guest_id number ) partition by range (res_id) interval (100) store in (users) ( partition p1 values less than (101) );



  5. create table res ( res_id number not null , res_date date , hotel_id number(3), guest_id number ) partition by range (res_id) interval (100) store in (users) ( partition p1 values less than (101) );



  6. create table res ( res_id number not null , res_date date , hotel_id number(3), guest_id number ) partition by range (res_id) interval (100) store in (users) ( partition p1 values less than (101) );



  7. create table res ( res_id number not null , res_date date , hotel_id number(3), guest_id number ) partition by range (res_id) interval (100) store in (users) ( partition p1 values less than (101) );



  8. create table res ( res_id number not null , res_date date , hotel_id number(3), guest_id number ) partition by range (res_id) interval (100) store in (users) ( partition p1 values less than (101) );



  9. create table res ( res_id number not null , res_date date , hotel_id number(3), guest_id number ) partition by range (res_id) interval (100) store in (users) ( partition p1 values less than (101) );



  10. create table res ( res_id number not null , res_date date , hotel_id number(3), guest_id number ) partition by range (res_id) interval (100) store in (users) ( partition p1 values less than (101) );



  11. create table res ( res_id number not null , res_date date , hotel_id number(3), guest_id number ) partition by range (res_id) interval (100) store in (users) ( partition p1 values less than (101) );





यह स्क्रिप्ट रिकॉर्ड के लिए एक p1 सेक्शन बनाता है जिसका res_id कॉलम मान 1-100 की सीमा में है। जब 101 से कम res_id स्तंभ मान वाले रिकॉर्ड्स सम्मिलित किए जाते हैं, तो उन्हें p1 अनुभाग में रखा जाता है, और जब नए रिकॉर्ड में इस स्तंभ का मान 101 से अधिक या बराबर होता है, तो Oracle डेटाबेस 11g एक नया अनुभाग बनाता है, जिसका नाम सिस्टम द्वारा उत्पन्न होता है। आप इस उदाहरण और अन्य नई विभाजन योजनाओं के बारे में अधिक जान सकते हैं जो अरुप नंदा द्वारा ओरेकल पत्रिका के रूसी संस्करण में एक लेख के अनुवाद में हैं

नीचे वर्णित समाधानों को अन्य DBMS पर लागू किया जा सकता है जो विभाजन के स्वचालित जोड़ने का समर्थन नहीं करते हैं।



अंतराल के बिना एक समान रूप से बढ़ती विभाजन कुंजी के लिए समाधान



एक परीक्षण तालिका बनाएँ:

  1. टेबल टेस्ट_पार्ट बनाएं (
  2. आईडी संख्या शून्य नहीं है ,
  3. नाम varchar2 (100) शून्य नहीं ,
  4. मालिक varchar2 (100) अशक्त नहीं ,
  5. प्रकार varchar2 (100) शून्य नहीं ,
  6. बनाई गई तारीख शून्य नहीं है ,
  7. constraint test_part_pk
  8. प्राथमिक कुंजी (आईडी)
  9. )
  10. विभाजन (आईडी) द्वारा विभाजन (विभाजन p1 मान (10000) से कम );


यह तर्कसंगत है कि यदि ऐसी तालिका में कोई अंतराल नहीं है, तो विभाजन खंड के अधिकतम खंड की सीमा से पहले नए खंड बनाने के लिए यह वांछनीय होगा। सीमा पर हमारे पास कितने प्रमुख मूल्य हैं, हम एक सरल सूत्र के अनुसार आसानी से निर्धारित कर सकते हैं: विभाजन_साइज - (की-स्टार्ट_की_इन_पार्टिशन) , जहां कुंजी वर्तमान विभाजन कुंजी है, start_key_in_partition पहली कुंजी है जो इस खंड में आती है, विभाजन_साइज अनुभाग में कुंजियों की संख्या है। और% पूर्णांक विभाजन (div) का संचालन है। आमतौर पर, इस तरह के विभाजन को समान वर्गों में किया जाता है, और इसे ध्यान में रखते हुए, हम इस सूत्र को निम्न में सरल कर सकते हैं: पार्टीशन_साइज़ - कुंजी% पार्टीशन_साइज़।

यह हमें क्या देता है: पल को जानना, हम एक ट्रिगर बना सकते हैं जो इस घटना के होने पर वर्गों को जोड़ देगा।

यह ट्रिगर बनाएं:

  1. tr_test_part ट्रिगर बनाएं या बदलें
  2. test_part पर डालने से पहले
  3. प्रत्येक पंक्ति के लिए
  4. जब (mod ( NEW .id, 10000) = 6000)
  5. घोषित
  6. l_part_name नंबर;
  7. l_maxvalue संख्या;
  8. l_exist संख्या;
  9. l_partition_exists अपवाद ;
  10. PRAGMA EXCEPTION_INIT (l_partition_exists, -14074);
  11. PRAGMA AUTONOMOUS_TRANSACTION;
  12. शुरू करना
  13. l_part_name: = ceil (: NEW .ID / 10000) +1;
  14. शुरू
  15. तत्काल 'अदला-बदली तालिका xtender.test_part add विभाजन p' || l_part_name || 'मान से कम (' || l_maxvalue || ')' ?
  16. अपवाद
  17. जब l_partition_exists तब शून्य ;
  18. अंत;
  19. अंत tr_test_part;




स्वायत्त लेनदेन का उपयोग करते हुए यह ट्रिगर, स्वचालित रूप से 'P' + 10,000 की एक संख्या के नाम के साथ एक नया खंड बनाता है, जब ID हमारी विभाजन कुंजी है, अनुभाग सीमा (10000-4000 = 6000, अर्थात आईडी = 6000, तक 4,000 मान शेष हैं) 16000,26000, आदि), लेकिन पहले यह जाँच की जाती है कि क्या दिया गया खंड पहले से मौजूद है (यह हो सकता है, उदाहरण के लिए, 6000 वां रिकॉर्ड फिर से जोड़कर, या मैन्युअल रूप से अनुभाग जोड़कर)। विभाजन के पैरामीटर - 10000 और 4000, आपको अपनी विशिष्ट स्थिति के आधार पर चुनना चाहिए, लेकिन ध्यान रखें कि सीमा (उदाहरण में 4000) एक बार में जोड़े गए रिकॉर्ड की अधिकतम संख्या से अधिक होनी चाहिए, क्योंकि अन्यथा, डेटा प्रविष्टि लेन-देन के समय, लेनदेन नए अनुभाग के बारे में "पता" नहीं करेगा, क्योंकि लेन-देन की शुरुआत में, यह मौजूद नहीं था, इसलिए इस कुंजी को अनुभाग मैपिंग की अनुपस्थिति के बारे में शिकायत के साथ डेटा नहीं डाला जाएगा। यह परिवर्तन तालिका विभाजन default_partition के उपयोग से बचा जा सकता था, जिसकी मैं बाद में चर्चा करूंगा, लेकिन यह रनटाइम को प्रभावित करेगा।

अनुभाग भरकर हमारे ट्रिगर की जाँच करें:

insert into xtender.test_part

select rownum, o.OBJECT_NAME, o.OWNER, o.OBJECT_TYPE, o.CREATED

from all_objects o

where rownum<1000;








इसके अलावा, अनुक्रम के उपयोग के मामले में कि "कदम" कैशिंग के कारण, क्रमिक रूप से ट्रिगर को बदलना संभव नहीं है, ताकि यह अनुभाग के अंत से 4000 से 3900 रिकॉर्ड तक मानों के एक सेट के लिए चलता है:

स्थिति को बदलें

  जब (mod (NEW.id, 10000) = 6000) 
पर

  जब (mod (NEW.id, 10000) 6000 और 6100 के बीच) 




अन्य समाधान



उन मामलों में जहां हम डिफ़ॉल्ट अनुभाग को निर्दिष्ट करते हैं, हम इसे अलग कर सकते हैं जब रिकॉर्ड पहले ही मिल चुके हैं, तो सवाल यह है कि यह स्वचालित रूप से कैसे ट्रैक करें



डेटा डिक्शनरी में हम dba_tab_partitions से चयन करके पार्टीशन टेबल के सभी सेक्शन के बारे में जानकारी प्राप्त कर सकते हैं, जिसमें पार्टीशन टेबल में सेक्शन के क्रम को इंगित करता है, और high_value सेक्शन पैरामीटर को इंगित करता है। इसलिए, हम तालिका में अंतिम अनुभाग का नाम प्राप्त कर सकते हैं और उसमें से रिकॉर्ड प्राप्त करने के लिए उसमें से एक चयन कर सकते हैं।



उन तालिका के बारे में जानकारी प्राप्त करने के बाद, जिनमें डिफ़ॉल्ट अनुभागों में रिकॉर्डिंग शुरू हुई, हमें एक सूचना भेजनी होगी। ऐसा करने के लिए, हम विकल्पों का उपयोग कर सकते हैं:





पहला विकल्प dbms_system.ksdwrt प्रक्रिया का उपयोग करके लागू किया गया है, जिसमें दो पैरामीटर हैं:

एक उदाहरण:

 exec dbms_system.ksdwrt (2, 'टेस्ट अलर्ट संदेश');




दूसरा विकल्प utl_mail पैकेज या निचले वाले का उपयोग करना है - utl_smtp या utl_tcp

utl_mail utl_smtp के लिए अधिक सुविधाजनक आवरण है, लेकिन इसका उपयोग करने के लिए, आपको smtp_out_server पैरामीटर सेट करना होगा। आप इसे सत्र के लिए जैसे ही कर सकते हैं - “ALTER SESSION SET smtp_out_server =…” और सिस्टम के लिए “ALTER SYSTEM SET smtp_out_server =…”

अगर आपको यह पैकेज घर पर नहीं मिल रहा है तो आश्चर्यचकित न हों - शुरू में इसमें शामिल नहीं है और इसे बनाने के लिए आपको दो स्क्रिप्ट्स चलाने होंगे:

 sqlplus sys/<pwd> SQL> @$ORACLE_HOME/rdbms/admin/utlmail.sql SQL> @$ORACLE_HOME/rdbms/admin/prvtmail.plb
      
      







यह सब एक साथ pkg_partitions पैकेज में लाना

पैकेज के तरीके:





पैकेज कोड:





  1. पैकेज बॉडी pkg_partitions बनाएं या बदलें
  2. / ** समारोह अनुभाग के लिए एक पैरामीटर लौटाता है
  3. * @ अपरम i_table_name तालिका नाम
  4. * @ वापसी varchar2
  5. * /
  6. फ़ंक्शन get_penultimate_maxvalue (p_table_owner varchar2, p_table_name varchar2) रिटर्न varchar2 है
  7. l_cursor पूर्णांक डिफ़ॉल्ट dbms_sql.open_cursor;
  8. l_ignore नंबर;
  9. l_long_val varchar2 (4000);
  10. l_long_len संख्या;
  11. l_buflen संख्या: = 4000;
  12. l_curpos संख्या: = 0;
  13. शुरू करना
  14. dbms_sql.parse (l_cursor,
  15. ' all_tab_partitions p से p.high_value का चयन करें जहां p.table_owner जैसे: o और p.table_name जैसे: x और p.partition_position = (select max (p1.partition_position) -1 all_tab_partitions p1 से जहाँ p.table_owner पसंद है: o और p1। table_name जैसे: x) '
  16. dbms_sql.native);
  17. dbms_sql.bind_variable (l_cursor, ': x' , p_table_name);
  18. dbms_sql.bind_variable (l_cursor, ': o' , p_table_owner);
  19. dbms_sql.define_column_long (l_cursor, 1);
  20. l_ignore: = dbms_sql निष्पादित (l_cursor);
  21. अगर (dbms_sql.fetch_rows (l_cursor)> 0)
  22. तो
  23. dbms_sql.column_value_long (l_cursor, 1, l_buflen, l_curpos,
  24. l_long_val, l_long_len);
  25. अंत यदि ;
  26. dbms_sql.close_cursor (l_cursor);
  27. वापसी l_long_val;
  28. अंत ;
  29. / ** फ़ंक्शन तालिकाओं और उनके मालिकों के नाम लौटाते हैं, जिनके अंतिम अनुभाग को भरना शुरू हुआ
  30. * @ वापसी
  31. * table_name varchar2 (4000),
  32. * table_owner varchar2 (4000),
  33. * विभाजन_ संख्या संख्या,
  34. * विभाजन_नाम varchar2 (4000));
  35. * /
  36. फ़ंक्शन get_maxvalued_partitions वापसी टेबल_प्रॉप्स_अरे पाइपलाइनबद्ध है
  37. l_cursor पूर्णांक डिफ़ॉल्ट dbms_sql.open_cursor;
  38. l_count संख्या;
  39. l_ignore पूर्णांक ;
  40. l_data table_props;
  41. कर्सर l_partitions है
  42. चुनना
  43. pl.table_owner,
  44. pl.table_name,
  45. गिनती (1) cnt,
  46. अधिकतम (pl.partition_name) रखना (d.partition_position द्वारा अंतिम क्रम में रखना) (विभाजन)
  47. dba_tab_partitions pl से
  48. जहां pl.table_name 'BIN $%' को पसंद नहीं करता
  49. pl.table_owner, pl.table_name द्वारा समूह
  50. गिनती (1)> 1;
  51. शुरू
  52. L_partitions में भाग के लिए
  53. लूप
  54. dbms_sql.parse (l_cursor,
  55. 'सेलेक्ट काउंट (1) ' || part.table_owner || '।' part.table_name
  56. || 'विभाजन (' || part.partition_name || ')'
  57. || 'जहां पंक्तिम <2
  58. dbms_sql.native);
  59. dbms_sql.define_column (l_cursor, 1, l_count);
  60. l_ignore: = dbms_sql.execute_and_fetch (l_cursor);
  61. dbms_sql.column_value (l_cursor, 1, l_count);
  62. अगर (l_count> 0) तो
  63. l_data.table_name: = part.table_name;
  64. l_data.table_owner: = part.table_owner;
  65. l_data.partitions_count: = part.cnt;
  66. l_data.partition_name: = part.partition_name;
  67. पाइप पंक्ति (l_data);
  68. अंत यदि ;
  69. END LOOP;
  70. अंत;
  71. / ** फ़ंक्शन जो HTML के रूप में तालिकाओं और उनके मालिकों के नाम लौटाता है, जिसमें अंतिम अनुभाग भरना शुरू हुआ था
  72. * @ वापसी
  73. * table_name varchar2 (4000),
  74. * table_owner varchar2 (4000),
  75. * विभाजन_ संख्या संख्या,
  76. * विभाजन_नाम varchar2 (4000));
  77. * /
  78. फ़ंक्शन get_maxvalued_partitions_html वापसी varchar2 है
  79. l_cursor पूर्णांक डिफ़ॉल्ट dbms_sql.open_cursor;
  80. l_count संख्या;
  81. l_ignore पूर्णांक ;
  82. l_data varchar2 (4000);
  83. कर्सर l_partitions है
  84. चुनना
  85. pl.table_owner,
  86. pl.table_name,
  87. गिनती (1) cnt,
  88. अधिकतम (pl.partition_name) रखना (d.partition_position द्वारा अंतिम क्रम में रखना) (विभाजन)
  89. dba_tab_partitions pl से
  90. जहां pl.table_name 'BIN $%' को पसंद नहीं करता
  91. pl.table_owner, pl.table_name द्वारा समूह
  92. गिनती (1)> 1;
  93. शुरू
  94. l_data: = '<html> <body> <table बॉर्डर = 1>'
  95. || '<tr> <th> तालिका का नाम </ th>'
  96. || '<th> टेबल के मालिक </ th>'
  97. || '<th> विभाजन की गिनती </ th>'
  98. || '<th> विभाजन नाम </ th>'
  99. || '<th> प्री मैक्सवेलु </ th>' ;
  100. L_partitions में भाग के लिए
  101. लूप
  102. dbms_sql.parse (l_cursor,
  103. 'सेलेक्ट काउंट (1) ' || part.table_owner || '।' part.table_name
  104. || 'विभाजन (' || part.partition_name || ')'
  105. || 'जहां पंक्तिम <2
  106. dbms_sql.native);
  107. dbms_sql.define_column (l_cursor, 1, l_count);
  108. l_ignore: = dbms_sql.execute_and_fetch (l_cursor);
  109. dbms_sql.column_value (l_cursor, 1, l_count);
  110. अगर (l_count> 0) तो
  111. l_data: = l_data || '<tr> <td>'
  112. part.table_name
  113. || '</ td> <td>'
  114. || part.table_owner ||
  115. || '</ td> <td>'
  116. || भाग
  117. || '</ td> <td>'
  118. || part.partition_name
  119. || '</ td> </ tr>' ;
  120. अंत यदि ;
  121. END LOOP;
  122. l_data: = l_data || '</ table> </ body> </ html>' ;
  123. वापसी l_data;
  124. अंत;
  125. / **
  126. * तालिकाओं के साथ एक रिपोर्ट भेजने की प्रक्रिया जिसका अंतिम भाग भरने लगा
  127. * /
  128. प्रक्रिया send_partitions_report (मेल varchar2)
  129. है
  130. msg_body varchar2 (4000);
  131. शुरू
  132. pkg_partitions.get_maxvalued_partitions_html को ड्यूस से msg_body में चुनें ;
  133. - EXECUTE IMMEDIATE 'पहले सेशन सेट smtp_out_server =' 'our_mailserver' '' ;
  134. utl_mail.send (
  135. प्रेषक => 'oracleDBA@dbdomain.com' ,
  136. प्राप्तकर्ता => मेल,
  137. विषय => 'मैक्सवेल्ड विभाजन रिपोर्ट' ,
  138. संदेश => msg_body,
  139. mime_type => 'text / html' );
  140. अंत;
  141. अंत pkg_partitions;
* इस सोर्स कोड को सोर्स कोड हाइलाइटर के साथ हाइलाइट किया गया था।


युपीडी

जैसा कि zhekappp ने सही तरीके से संकेत दिया है, आप स्वचालित सांख्यिकी संग्रह को सक्षम करते समय num_rows का उपयोग कर सकते हैं। Dbms_stats.gather_table_stats के साथ dbms_job का उपयोग करके नौकरी जोड़कर सांख्यिकी संग्रह को सक्षम किया जा सकता है।

तब आपको अनुभाग में रिकॉर्ड की संख्या के लिए अनुरोध को हटाने और अनुरोध को बदलने की आवश्यकता होगी:

  1. चुनना
  2. pl.table_owner,
  3. pl.table_name,
  4. गिनती (1) cnt,
  5. अधिकतम (pl.num_rows) रखें (dense_rank अंतिम क्रम द्वारा (pl.partition_position)) विभाजन_,
  6. अधिकतम (pl.partition_name) रखना (d.partition_position द्वारा अंतिम क्रम में रखना) (विभाजन)
  7. dba_tab_partitions pl से
  8. जहां pl.table_name 'BIN $%' को पसंद नहीं करता
  9. pl.table_owner, pl.table_name द्वारा समूह


इस मामले के लिए पूर्ण पैकेज कोड यहां पाया जा सकता है: http://www.xt-r.com/2010/10/pkgpartitions.html



स्वचालित निष्पादन



यह केवल स्वचालित निष्पादन को कॉन्फ़िगर करने के लिए बनी हुई है। इसे dbms_job के साथ करते हैं।

उदाहरण के लिए, डेटा अधिग्रहण स्क्रिप्ट का दैनिक स्वचालित निष्पादन:

  1. घोषित
  2. नौकरी बाइनरी_इन्टेगर;
  3. शुरू करना
  4. dbms_job.submit (
  5. नौकरी,
  6. 'pkg_partitions.send_partitions_report (' 'dba@domain.ru' ');' '
  7. sysdate,
  8. 'trunc (sysdate) +1' );
  9. dbms_output.put_line (job);
  10. अंत ;



All Articles