ORACLE TEXT DATASTORE PREFERENCE(七)
這篇文章討論全文索引DATASTORE的最后一種屬性,介紹USER_DATASTORE。
Oracle功能的強大體現(xiàn)在很多方面,除了很多Oracle預(yù)定義好的接口以外,Oracle還提供了強大的自定義功能。USER_DATASTORE就是一個例子。如果上面幾篇文章介紹的DATASTORE屬性無法滿足用戶的要求,Oracle允許用戶自定義一個過程,根據(jù)用戶的需求來確定被索引的內(nèi)容。
USER_DATASTORE具有很高的靈活性,這里只介紹一個簡單的例子:
SQL> CREATE TABLE T (ID NUMBER PRIMARY KEY, AUTHER VARCHAR2(30), TITLE VARCHAR2(30));
表已創(chuàng)建。
SQL> CREATE TABLE T1 (ID NUMBER PRIMARY KEY, FID NUMBER, SEQ NUMBER, DOCS VARCHAR2(1000),?
2 CONSTRAINT FK_T1_FID FOREIGN KEY (FID) REFERENCES T(ID));
表已創(chuàng)建。
SQL> INSERT INTO T VALUES (1, 'YTK', 'DETAIL DATASTORE');
已創(chuàng)建?1 行。
SQL> INSERT INTO T1 VALUES (1, 1, 1, 'THIS IS A DETAIL DATASTORE EXAMPLE');
已創(chuàng)建?1 行。
SQL> INSERT INTO T1 VALUES (2, 1, 2, 'THE CONTEXT IS STORE IN DETAIL TABLES');
已創(chuàng)建?1 行。
SQL> INSERT INTO T VALUES (2, 'YTK', 'DETAIL DATASTORE DOC');
已創(chuàng)建?1 行。
SQL> INSERT INTO T1 VALUES (3, 2, 1, 'USE THE DETAIL_DATASTORE TYPE FOR TEXT');
已創(chuàng)建?1 行。
SQL> INSERT INTO T1 VALUES (4, 2, 2, 'STORED DEIRECTLY IN THE DATABASE IN DETAIL TABLES');
已創(chuàng)建?1 行。
SQL> INSERT INTO T1 VALUES (5, 2, 3, 'WITH INDEXED TEXT COLUMN');
已創(chuàng)建?1 行。
SQL> INSERT INTO T1 VALUES (6, 2, 4, 'LOCATED IN THE MASTER TABLE');
已創(chuàng)建?1 行。
SQL> COMMIT;
提交完成。
SQL> CONN CTXSYS/CTXSYS@YANGTK
已連接。
SQL> BEGIN
2 CTX_DDL.CREATE_PREFERENCE('TEST_DETAIL', 'DETAIL_DATASTORE');
3 CTX_DDL.SET_ATTRIBUTE('TEST_DETAIL', 'BINARY', 'TRUE');
4 CTX_DDL.SET_ATTRIBUTE('TEST_DETAIL', 'DETAIL_TABLE', 'T1');
5 CTX_DDL.SET_ATTRIBUTE('TEST_DETAIL', 'DETAIL_KEY', 'FID');
6 CTX_DDL.SET_ATTRIBUTE('TEST_DETAIL', 'DETAIL_LINENO', 'SEQ');
7 CTX_DDL.SET_ATTRIBUTE('TEST_DETAIL', 'DETAIL_TEXT', 'DOCS');
8 END;
9 /
PL/SQL 過程已成功完成。
SQL> CONN YANGTK/YANGTK@YANGTK
已連接。
SQL> CREATE INDEX IND_T_DOCS ON T (TITLE) INDEXTYPE IS CTXSYS.CONTEXT
2 PARAMETERS ('DATASTORE CTXSYS.TEST_DETAIL');
索引已創(chuàng)建。
SQL> SELECT * FROM T WHERE CONTAINS(TITLE, 'MASTER') > 0;
ID AUTHER TITLE
---------- ------------------------------ ------------------------------
2 YTK DETAIL DATASTORE DOC
SQL> SELECT * FROM T WHERE CONTAINS(TITLE, 'DOC') > 0;
未選定行
上面這個例子是前面DETAIL_DATASTORE的例子,這個例子中雖然子表中的DOCS列的內(nèi)容被索引了,但是主表中建立索引的TITLE字段并沒有包含在索引中。
如果想要把TITLE內(nèi)容也包含到索引中,一般的方法是建立一個TITLE列的索引,但是這種方法有幾個問題,首先如果在TITLE上建立普通索引,那么就無法對TITLE列進行一些只有CONTEXT索引才支持的高級搜索,且普通索引和全文索引的配合也不是很好。如果在TITLE列上再建立一個全文索引,那么兩個全文索引的配合肯定不如一個全文索引效率高,而且,由于同一個字段上不能建立兩個全文索引,那么主子表的索引字段還要選擇其他的列。顯然這種方法問題比較多。
如果要建立一個CTXCAT索引似乎可以很好的解決這個問題,只要將TITLE列的普通索引添加到全文索引的索引組中即可,可惜的是,CTXCAT索引只支持普通的DIRECT_DATASTORE類似的索引。也就是說,CTXCAT索引的內(nèi)容必須存儲在數(shù)據(jù)庫的一張表的一個列里面。
其實想要解決上面提出的那個問題并不算復(fù)雜,只需要使用USER_DATASTORE建立一個自定義的存儲過程:
SQL> DROP INDEX IND_T_DOCS;
索引已丟棄。
SQL> CREATE OR REPLACE PROCEDURE P_MYSTORE (P_ROWID IN ROWID, P_CLOB IN OUT NOCOPY CLOB) AS
2 BEGIN
3 FOR C_T IN (SELECT ID, AUTHER, TITLE FROM T WHERE ROWID = P_ROWID) LOOP
4 DBMS_LOB.WRITEAPPEND(P_CLOB, LENGTH(C_T.AUTHER) + 1, C_T.AUTHER || ' ');
5 DBMS_LOB.WRITEAPPEND(P_CLOB, LENGTH(C_T.TITLE) + 1, C_T.TITLE || ' ');
6 FOR C_T1 IN (SELECT DOCS FROM T1 WHERE FID = C_T.ID ORDER BY SEQ) LOOP
7 DBMS_LOB.WRITEAPPEND(P_CLOB, LENGTH(C_T1.DOCS) + 1, C_T1.DOCS || ' ');
8 END LOOP;
9 END LOOP;
10 END;
11 /
過程已創(chuàng)建。
SQL> CONN CTXSYS/CTXSYS@YANGTK
已連接。
SQL> CREATE OR REPLACE PROCEDURE P_USERSTORE (P_ROWID IN ROWID, P_CLOB IN OUT NOCOPY CLOB) AS
2 BEGIN
3 YANGTK.P_MYSTORE(P_ROWID, P_CLOB);
4 END;
5 /
過程已創(chuàng)建。
SQL> GRANT EXECUTE ON P_USERSTORE TO YANGTK;
授權(quán)成功。
SQL> BEGIN
2 CTX_DDL.CREATE_PREFERENCE('TEST_USER', 'USER_DATASTORE');
3 CTX_DDL.SET_ATTRIBUTE('TEST_USER', 'PROCEDURE', 'P_USERSTORE');
4 CTX_DDL.SET_ATTRIBUTE('TEST_USER', 'OUTPUT_TYPE', 'CLOB');
5 END;
6 /
PL/SQL 過程已成功完成。
SQL> CONN YANGTK/YANGTK@YANGTK
已連接。
SQL> CREATE INDEX IND_T_DOCS ON T(TITLE) INDEXTYPE IS CTXSYS.CONTEXT
2 PARAMETERS ('DATASTORE CTXSYS.TEST_USER');
索引已創(chuàng)建。
SQL> SELECT * FROM T WHERE CONTAINS(TITLE, 'MASTER') > 0;
ID AUTHER TITLE
---------- ------------------------------ ------------------------------
2 YTK DETAIL DATASTORE DOC
SQL> SELECT * FROM T WHERE CONTAINS(TITLE, 'DOC') > 0;
ID AUTHER TITLE
---------- ------------------------------ ------------------------------
2 YTK DETAIL DATASTORE DOC
SQL> SELECT * FROM T WHERE CONTAINS(TITLE, 'YTK') > 0;
ID AUTHER TITLE
---------- ------------------------------ ------------------------------
2 YTK DETAIL DATASTORE DOC
1 YTK DETAIL DATASTORE
通過上面的例子可以看到,使用Oracle提供的接口,即使存儲的列的邏輯在復(fù)雜,也可以很方便的根據(jù)自己的需求來設(shè)計索引的存儲屬性。
總結(jié)
以上是生活随笔為你收集整理的ORACLE TEXT DATASTORE PREFERENCE(七)的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: ORACLE TEXT DATASTOR
- 下一篇: ORACLE TEXT FILTER P