セクショニング。 セクションを自動的に追加する

バージョン11gでは、範囲が範囲外になると自動的にパーティションを作成するためのいくつかのエキサイティングな新しいパーティションスキーム(間隔パーティションの便利な機能など)が導入されました。

11gより前のバージョンでは、定期的に手動でセクションを追加するか、デフォルトのセクションを分割する必要があります。 つまり、このようなテーブルのステータスを常に監視する必要があります。 この記事では、このようなパーティション分割タスクを自動化するためのソリューションを共有します。

まず、11gの例を示します。

  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) );





このスクリプトは、res_id列の値が1〜100の範囲にあるレコードのp1セクションを作成します。 res_id列の値が101未満のレコードが挿入されると、それらはp1セクションに配置され、新しいレコードのこの列の値が101以上の場合、Oracle Database 11gは新しいセクションを作成し、その名前はシステムによって生成されます。 Oracle Magazineのロシア語版のArup Nandaによる記事翻訳で、この例と他の新しいパーティション化スキームの詳細を確認できます

以下で説明するソリューションは、パーティションの自動追加をサポートしない他のDBMSに適用できます。



隙間なく均一に増加するパーティションキーのソリューション



テストテーブルを作成します。

  1. テーブル test_partの作成
  2. ID番号 nullはありません
  3. name varchar2(100) not null
  4. 所有者varchar2(100) null以外
  5. タイプvarchar2(100) not null
  6. 作成 nullはない
  7. 制約 test_part_pk
  8. キー (id)
  9. 範囲(id) によるパーティション(パーティションp1 (10000) 未満 );


そのようなテーブルにギャップがない場合、パーティションキーが最大セクションの境界に近づく前に新しいセクションを作成することが望ましいと考えられます。 境界に残したキー値の数、簡単な式に従って簡単に決定できます: partition_size-(key-start_key_in_partition) 、ここでkeyは現在のパーティションキー、start_key_in_partitionはこのセクションに含まれる最初のキー、partition_sizeはセクション内のキーの数、 and%は整数除算(div)の演算です。 通常、このようなパーティション化は等しいセクションに実行され、これを念頭に置いて、この式を次のように単純化できます:partition_size-key%partition_size。

これにより得られるもの:瞬間を知って、このイベントが発生したときにセクションを追加するトリガーを作成できます。

このトリガーを作成します。

  1. トリガー tr_test_partを作成 または置換します
  2. test_part 挿入 する
  3. ごとに
  4. when (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. すぐに 「alter table xtender.test_part add partition p」を 実行し ます || l_part_name || '(より小さい値(' || l_maxvalue || ')' ;
  16. 例外
  17. l_partition_existsがnullの 場合
  18. 終了
  19. end tr_test_part;




このトリガーは、自律型トランザクションを使用して、「P」という名前とセクション番号10,000の新しいセクションを自動的に作成します。IDがセクションキーである場合、セクションの境界まで4,000個の値が残ります(10000-4000 = 6000、つまりID = 6000、 16000、26000など)、ただし、最初に特定のセクションが既に存在するかどうかが確認されます(これは、たとえば、6000番目のレコードを再追加するか、セクションを手動で追加することによって発生します)。 パーティショニングパラメータ-10000および4000。特定の状況に基づいて選択する必要がありますが、境界線(この例では4000)は一度に追加されるレコードの最大数よりも大きくする必要があることに注意してください。 そうでない場合、データ挿入トランザクションの時点で、トランザクションは新しいセクションを「認識」しません。 トランザクションの開始時には存在していなかったため、このキーにマッピングされたセクションが存在しないという苦情とともにデータは挿入されません。 これは、後で説明するalter table split 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レコードの値のセットに対して実行されるようにトリガーを順番に変更することはできません。

状態を置き換える

  when(mod(NEW.id、10000)= 6000) 


  when(mod(NEW.id、10000)between 6000 and 6100) 




その他の解決策



デフォルトセクションを指定する場合、レコードがすでにそこにあるときにそれを分離できます。質問は、これを自動的に追跡する方法です。



データディクショナリでは、 dba_tab_partitionsから選択を行うことにより、パーティションテーブルのすべてのセクションに関する情報を取得できます。partition_positionはテーブル内のセクションの順序を示しhigh_valueはセクションパラメーターを示します。 したがって、テーブルの最後のセクションの名前を取得し、そこから選択して、その中のレコード数を取得できます。



デフォルトセクションで記録が開始されたテーブルに関する情報を受信した後、通知を送信する必要があります。 これを行うには、次のオプションを使用できます。





最初のオプションは、2つのパラメーターを持つdbms_system.ksdwrtプロシージャを使用して実装されます。

例:

 exec dbms_system.ksdwrt(2、 'テストアラートメッセージ');




2番目のオプションは、 utl_mailパッケージまたはそれ以下のパッケージ-utl_smtpまたはutl_tcpを使用することです

utl_mailutl_smtpのより便利なラッパーですが、使用するにはsmtp_out_serverパラメーターを設定する必要があります。 これは、セッション「ALTER SESSION SET smtp_out_server = ...」およびシステム「ALTER SYSTEM SET smtp_out_server = ...」に対してすぐに実行できます。

自宅でこのパッケージを見つけられなくても驚かないでください-最初は含まれていません。作成するには、2つのスクリプトを実行する必要があります。

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







すべてをpkg_partitionsパッケージにまとめます

パッケージ方法:





パッケージコード:





  1. パッケージ本体の作成 または置換pkg_partitions
  2. / **最後から2番目のセクションのパラメーターを返す関数
  3. * @param i_table_nameテーブル名
  4. * @ return varchar2
  5. * /
  6. 関数 get_penultimate_maxvalue(p_table_owner varchar2、p_table_name varchar2) return varchar2
  7. l_cursor integer default dbms_sql.open_cursor;
  8. l_ignore number;
  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 like:xおよびp.partition_position =(select max(p1.partition_position)-1 from all_tab_partitions p1 where p.table_owner like:o and p1。 table_name like: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. if (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. return l_long_val;
  28. 終わり ;
  29. / **テーブルの名前とその最後のセクションが埋め始めた所有者を返す関数
  30. * @ リターン
  31. * table_name varchar2(4000)、
  32. * table_owner varchar2(4000)、
  33. * partitions_count番号、
  34. * partition_name varchar2(4000));
  35. * /
  36. 関数 get_maxvalued_pa​​rtitions は、 tables_props_arrayパイプライン化を返します
  37. l_cursor integer default dbms_sql.open_cursor;
  38. l_count番号。
  39. l_ignore integer ;
  40. l_data table_props;
  41. カーソル l_partitions
  42. 選択する
  43. pl.table_owner、
  44. pl.table_name、
  45. count (1)cnt、
  46. max (pl.partition_name)keep(dense_rank last order by (pl.partition_position))partition_name
  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. 「select count(1) from」|| part.table_owner || 「。」 || part.table_name
  56. || 'パーティション(' || part.partition_name || ')'
  57. || 'where rownum <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. if (l_count> 0) then
  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. ループの終了。
  70. 終了
  71. / **テーブルの名前とその所有者をhtmlの形式で返す関数。最後のセクションが埋め始めた
  72. * @ リターン
  73. * table_name varchar2(4000)、
  74. * table_owner varchar2(4000)、
  75. * partitions_count番号、
  76. * partition_name varchar2(4000));
  77. * /
  78. function get_maxvalued_pa​​rtitions_html return varchar2
  79. l_cursor integer default dbms_sql.open_cursor;
  80. l_count番号。
  81. l_ignore integer ;
  82. l_data varchar2(4000);
  83. カーソル l_partitions
  84. 選択する
  85. pl.table_owner、
  86. pl.table_name、
  87. count (1)cnt、
  88. max (pl.partition_name)keep(dense_rank last order by (pl.partition_position))partition_name
  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 border = 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. 「select count(1) from」|| part.table_owner || 「。」 || part.table_name
  104. || 'パーティション(' || part.partition_name || ')'
  105. || 'where rownum <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. if (l_count> 0) then
  111. l_data:= l_data || '<tr> <td>'
  112. || part.table_name
  113. || '</ td> <td>'
  114. || part.table_owner
  115. || '</ td> <td>'
  116. || part.cnt
  117. || '</ td> <td>'
  118. || part.partition_name
  119. || '</ td> </ tr>' ;
  120. 終了する 場合 ;
  121. ループ終了;
  122. l_data:= l_data || '</ table> </ body> </ html>' ;
  123. return l_data;
  124. 終了
  125. / **
  126. *テーブルの最後のセクションが埋め始めたレポートを送信する手順
  127. * /
  128. プロシージャ send_partitions_report(メールvarchar2)
  129. msg_body varchar2(4000);
  130. 開始
  131. pkg_partitions.get_maxvalued_pa​​rtitions_html デュアルから msg_bodyに選択します
  132. -EXECUTE IMMEDIATE 'ALTER SESSION SET smtp_out_server =' 'our_mailserver' '' ;
  133. utl_mail.send(
  134. 送信者=> 'oracleDBA@dbdomain.com'
  135. 受信者=>メール、
  136. 件名=> '最大値パーティションレポート'
  137. メッセージ=> msg_body、
  138. mime_type => 'text / html' );
  139. 終了
  140. pkg_partitionsを終了します。
*このソースコードは、 ソースコードハイライターで強調表示されました。


UPD

zhekappp正しくプロンプトを表示したため、自動統計収集を有効にするときにnum_rowsを使用できます。 統計収集を有効にするには、dbms_jobsとdbms_stats.gather_table_statsを使用してジョブを追加します。

次に、セクション内のレコード数のリクエストを削除し、リクエストを次のように変更する必要があります。

  1. 選択する
  2. pl.table_owner、
  3. pl.table_name、
  4. count (1)cnt、
  5. max (pl.num_rows)keep(dense_rank last order by (pl.partition_position))partition_rows、
  6. max (pl.partition_name)keep(dense_rank last order by (pl.partition_position))partition_name
  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. job binary_integer;
  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(ジョブ);
  10. 終わり ;



All Articles