`
zhuozuoying
  • 浏览: 135639 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

Oracle建立全文索引详解

阅读更多
最近总结了一下,相关例子和部分内容也是搜索于网络,具体分下面几部分:

1.全文检索和普通检索的区别

不使用Oracle text功能,当然也有很多方法可以在Oracle数据库中搜索文本,比如INSTR函数和LIKE操作:

 1 2
 SELECT *FROM mytext WHERE INSTR (thetext, 'Oracle') > 0; SELECT * FROM mytext WHERE thetext LIKE '%Oracle%';


有很多时候,使用instr和like是很理想的, 特别是搜索仅跨越很小的表的时候。然而通过这些文本定位的方法将导致全表扫描,对资源来说消耗比较昂贵,而且实现的搜索功能也非常有限,因此对海量的文本数据进行搜索时,建议使用oralce提供的全文检索功能。

附:这里顺带记录一下INSTR和LIKE:

Oracle中,可以使用 Instr 函数对某个字符串进行判断,判断其是否含有指定的字符。其语法为:Instr(string, substring, position, occurrence)。

string:代表源字符串(写入字段则表示此字段的内容)。

substring:代表想从源字符串中查找的子串。

position:代表查找的开始位置,该参数可选的,默认为1。

occurrence:代表想从源字符中查找出第几次出现的substring,该参数也是可选的,默认为1。

position 的值为负数,那么代表从右往左进行查找。

instr和like的性能比较

其实从效率角度来看,谁能用到索引,谁的查询速度就会快。

like有时可以用到索引,例如:name like ‘李%’,而当下面的情况时索引会失效:name like ‘%李’。所以一般我们查找中文类似于‘%字符%’时,索引都会失效。与其他数据库不同的是,oracle支持函数索引。例如在name字段上建个instr索引,查询速度就比较快了,这也是为什么instr会比like效率高的原因。

注:instr(title,’手册’)>0 相当于like‘%手册%’

instr(title,’手册’)=0 相当于not like‘%手册%’

2.设置全文检索

步骤步骤一:检查和设置数据库角色

首先检查数据库中是否有CTXSYS用户和CTXAPP脚色。如果没有这个用户和角色,意味着你的数据库创建时未安装intermedia功能(10G默认安装都有此用户和角色)。你必须修改数据库以安装这项功能。默认安装情况下,ctxsys用户是被锁定的,因此要先启用ctxsys的用户。

步骤二:赋权

在ctxsys用户下,授予测试用户oratext以下权限:

 1 2 3 4 5 6 7 8 9
 GRANT resource, connect, ctxapp TO oratext; GRANT execute ON ctxsys.ctx_cls TO oratext; GRANT execute ON ctxsys.ctx_ddl TO oratext; GRANT execute ON ctxsys.ctx_doc TO oratext; GRANT execute ON ctxsys.ctx_output TO oratext; GRANT execute ON ctxsys.ctx_query TO oratext; GRANT execute ON ctxsys.ctx_report TO oratext; GRANT execute ON ctxsys.ctx_thes TO oratext; GRANT execute ON ctxsys.ctx_ulexer TO oratext;


步骤三:设置词法分析器(lexer)

Oracle实现全文检索,其机制其实很简单。即通过Oracle专利的词法分析器(lexer),将文章中所有的表意单元(Oracle 称为 term)找出来,记录在一组以dr$开头的表中,同时记下该term出现的位置、次数、hash值等信息。检索时,Oracle从这组表中查找相应的term,并计算其出现频率,根据某个算法来计算每个文档的得分(score),即所谓的‘匹配率’。而lexer则是该机制的核心,它决定了全文检索的效率。Oracle针对不同的语言提供了不同的lexer,而我们通常能用到其中的三个:

basic_lexer:针对英语。它能根据空格和标点来将英语单词从句子中分离,还能自动将一些出现频率过高已经失去检索意义的单词作为‘垃圾’处理,如if,is等,具有较高的处理效率。但该lexer应用于汉语则有很多问题,由于它只认空格和标点,而汉语的一句话中通常不会有空格,因此,它会把整句话作为一个term,事实上失去检索能力。以‘中国人民站起来了’这句话为例,basic_lexer分析的结果只有一个term,就是‘中国人民站起来了’。此时若检索‘中国’,将检索不到内容。

chinese_vgram_lexer:专门的汉语分析器,支持所有汉字字符集(ZHS16CGB231280 ZHS16GBK ZHT32EUC ZHT16BIG5 ZHT32TRIS ZHT16MSWIN950 ZHT16HKSCS UTF8 )。该分析器按字为单元来分析汉语句子。‘中国人民站起来了’这句话,会被它分析成如下几个term:‘中’,‘中国’,‘国人’,‘人民’,‘民站’,‘站起’,起来’,‘来了’,‘了’。可以看出,这种分析方法,实现算法很简单,并且能实现‘一网打尽’,但效率则是差强人意。

chinese_lexer:这是一个新的汉语分析器,只支持utf8字符集。上面已经看到,chinese vgram lexer这个分析器由于不认识常用的汉语词汇,因此分析的单元非常机械,像上面的‘民站’,‘站起’在汉语中根本不会单独出现,因此这种term是没有意义的,反而影响效率。chinese_lexer的最大改进就是该分析器能认识大部分常用汉语词汇,因此能更有效率地分析句子,像以上两个愚蠢的单元将不会再出现,极大提高了效率。但是它只支持utf8,如果你的数据库是zhs16gbk字符集,则只能使用笨笨的那个Chinese vgram lexer。如果不做任何设置,Oracle缺省使用basic_lexer这个分析器。

要指定使用哪一个lexer,可以这样操作:

第一.建立一个preference:

1
exec ctx_ddl.create_preference ('my_lexer', 'chinese_vgram_lexer');
第二.在建立全文索引索引时,指明所用的lexer:

1
                                                                                          
这样建立的全文检索索引,就会使用chinese_vgram_lexer作为分析器。

3.测试全文检索

测试用户为oratext,建立此用户和对应表空间的内容就不写了:

步骤一:授权,ctxsys登陆并对oratext用户授权:

1 2 3 4 5 6 7 8 9
 GRANT resource, connect, ctxapp TO oratext; GRANT execute ON ctxsys.ctx_cls TO oratext; GRANT execute ON ctxsys.ctx_ddl TO oratext; GRANT execute ON ctxsys.ctx_doc TO oratext; GRANT execute ON ctxsys.ctx_output TO oratext; GRANT execute ON ctxsys.ctx_query TO oratext; GRANT execute ON ctxsys.ctx_report TO oratext; GRANT execute ON ctxsys.ctx_thes TO oratext; GRANT execute ON ctxsys.ctx_ulexer TO oratext;


步骤二:设置词法分析器,使用chinese_vgram_lexer作为分析器:

1 2 3 4
BEGIN --设置词法分析器 ctx_ddl.create_preference ('oratext_lexer', 'chinese_vgram_lexer'); END;
可以通过下面的语句查看系统默认及设置的oracle text参数:

1
SELECT pre_name, pre_object FROM ctx_preferences
可以看到我刚刚设置的语法分析器参数oratext_lexer,(默认的有一个MY_LEXER的语法分析器参数)。

步骤三:建立测试表,插入测试数据:

 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18
 CREATE TABLE textdemo( id number NOT NULL PRIMARY KEY, book_author varchar2(20),--作者 publish_time date,--发布日期 title varchar2(400),--标题 book_abstract varchar2(2000),--摘要 path varchar2(200)--路径 ); commit; INSERT INTO textdemo VALUES(1,'宫琦峻',to_date('2008-10-07','yyyy-mm-dd'),' 移动城堡','故事发生在19世纪末的欧洲,善良可爱的苏菲被恶毒的女巫施下魔咒,从18岁的女孩变成90岁的婆婆,孤单无助的她无意中走入镇外的移动城堡,据说它的主人哈尔以吸取女孩的灵魂为乐,但是事情并没有人们传说的那么可怕,性情古怪的哈尔居然收留了苏菲,两个人在四脚的移动城堡中开始了奇妙的共同生活,一段交织了爱与痛、乐与悲的爱情故事在战火中悄悄展开','E:\textsearch\moveingcastle.doc'); INSERT INTO textdemo VALUES(2,'莫贝克曼贝托夫',to_date('2008-10-07','yyyy-mm-dd'),' 子弹转弯','这部由俄罗斯导演提莫贝克曼贝托夫执导的影片自6 月末在北美上映以来,已经在全球取得了超过3亿美元的票房收入。在亚洲上映后也先后拿下日本、韩国等地的票房冠军宝座。虽然不少网友在此之前也相继通过各种渠道接触到本片,但相信影片凭着在大银幕上呈现出的超酷的视听效果,依然能够吸引大量影迷前往影院捧场。','E:\textsearch\catch.pdf'); INSERT INTO textdemo VALUES(3,'袁泉',to_date('2008-10-07','yyyy-mm-dd'),'主演吴彦祖和袁泉现身','电影《如梦》在上海同乐坊拍摄,主演吴彦祖和袁泉现身。由于是深夜拍摄,所以周围并没有过多的fans注意到,给了剧组一个很清净的拍摄环境,站在街头的袁泉低着头,在寒冷的夜里看上去还真有些像女鬼,令人毛骨悚然。','E:\textsearch\dream.txt'); commit;


步骤四:在book_abstract字段建立索引使用刚刚设置的ORATEXT_LEXER :chinese_vgram_lexer作为分析器。

 1 2
 CREATE INDEX demo_abstract ON textdemo(book_abstract) indextype IS ctxsys.context parameters('lexer ORATEXT_LEXER'); commit;


之后如上所述多出很多dr$开头的表和索引,系统会创建四个相关的表:

DR$DEMO_ABSTRACT$I(分词后的TOKEN表)\DR$DEMO_ABSTRACT$K\DR$DEMO_ABSTRACT$N \DR$DEMO_ABSTRACT$R

下面的语句可以查看索引创建过程中是否发生了错误:

 1
 SELECT * FROM ctx_USER_index_errors


附:对于建立索引的类型(例如ctxsys.context),包括四种:context,ctxcat,ctxrule,ctxxpath。

CONTEXT用于对含有大量连续文本数据进行检索。支持word、html、xml、text等很多数据格式。支持范围(range)分区,支持并行创建索引(Parallel indexing)的索引类型。

支持类型:VARCHAR2, CLOB, BLOB, CHAR, BFILE, XMLType, and URIType.DML。操作后,需要CTX_DDL.SYNC_INDEX手工同步索引如果有查询包含多个词语,直接用空格隔开(如 oracle itpub)。

查询标识符CONTAINS

CTXCAT适用于混合查询语句(如查询条件包括产品id,价格,描述等)。适合于查询较小的具有一定结构的文本段。具有事务性。DML 操作后,索引会自动进行同步。

操作符:and,or,&gt,;<, =,between,in

查询标识符CATSEARCH

CTXRULE查询标识符MATCHES。

CTXXPATH(这两个索引没有去更多搜索相关内容)

一般来说我们建立CONTEXT类型的索引(CONTAINS来查询)。

步骤五:查询测试

 1 2 3 4 5 6 7
 --查询或 SELECT score(20),t.* FROM textdemo t WHERE contains(book_abstract,'移动城堡 or 俄罗斯',20)>0; SELECT score(20),t.* FROM textdemo t WHERE contains(book_abstract,'移动城堡 or 欧洲',20)>0; --基本查询 SELECT score(20),t.* FROM textdemo t WHERE contains(book_abstract,'移动城堡',20)>0; --查询包含多个词语and测试通过 SELECT score(20),t.* FROM textdemo t WHERE contains(book_abstract,'移动城堡 and 欧洲',20)>0;


测试通过。

4.对多字段建立全文索引(还在揣摩中)

很多时候需要从多个文本字段中查询满足条件的记录,这时就需要建立针对多个字段的全文索引,例如需要从pmhsubjects(专题表)的 subjectname(专题名称)和briefintro(简介)上进行全文检索,则需要按以下步骤进行操作:

建立多字段索引的preference,以ctxsys登录,并执行:

 1 2 3
 BEGIN ctx_ddl.create_preference('ctx_demo_abstract_title','MULTI_COLUMN_DATASTORE'); END;


建立preference对应的字段值(以ctxsys登录) 对应title path book_abstract三个字段建立索引:

 1 2 3
 BEGIN ctx_ddl.set_attribute('ctx_demo_abstract_title ','columns','title,path'); END;


建立全文索引:

 1 2
 CREATE INDEX demo_abstract_title ON textdemo(book_abstract) indextype IS ctxsys.context parameters(' DATASTORE ctxsys. ctx_demo_ abstract_title lexer ORATEXT_LEXER'); commit;


测试

1
SELECT score(20),t.* FROM textdemo t WHERE contains(book_abstract,'移动城堡 or 俄罗斯',20)>0;
5.对大字段进行检索测试

 1 2 3 4 5 6 7 8 9 10
 CREATE TABLE mytable(id number PRIMARY KEY, docs clob); INSERT INTO mytable VALUES(111555,'this text will be indexed'); INSERT INTO mytable VALUES(111556,'this is a direct_datastore example'); Commit; CREATE INDEX myindex ON mytable(docs) indextype IS ctxsys.context parameters ('datastore ctxsys.default_datastore'); SELECT * FROM mytable WHERE contains(docs, 'text') > 0;
分享到:
评论

相关推荐

    Oracle全文检索技术详解

    1. **建立表空间**:为全文索引创建专门的表空间,以便存储索引数据。 2. **定义索引**:使用`CREATE INDEX`语句定义全文索引,指定索引的字段和语言。 3. **建立索引**:调用`DBMS_TEXT_INDEX`包中的过程来创建索引...

    详解Oracle数据库中文全文索引

    ### 详解Oracle数据库中文全文索引 Oracle数据库的全文检索技术已经非常成熟和完善,其中Oracle Text使得Oracle 9i具备了强大的文本检索能力和智能化的文本管理能力。在不同的Oracle版本中,这一技术有着不同的名称...

    oracle索引类型及扫描方式大整理new

    InterMedia全文索引是Oracle提供的用于全文检索的索引类型,它能够支持复杂的文本查询,如关键词搜索、短语搜索等。这种索引通过对文档进行分词,并建立词与文档之间的关联,使得在海量文本数据中进行高效检索成为...

    oracle全文检索的实现

    3. **建立索引**:使用Oracle Text提供的工具或API,建立全文索引,这是全文检索的基础。 4. **发出查询**:通过SQL语句执行全文检索,利用Oracle Text提供的搜索语法,如CONTAINS函数。 5. **索引维护**:定期...

    oracle全文检索.pdf

    为了实现高效的全文检索,Oracle Text 需要为待检索的数据项建立索引。索引构建过程通常包括以下几个阶段: 1. **数据检索 (Datastore)**:从数据源(如 Web 页面、数据库 LOB 对象或本地文件系统)获取数据,并将...

    mysql数据库以及索引详解.pptx

    ### MySQL数据库及索引详解 #### 一、MySQL简介与数据库发展 MySQL是一款非常流行的开源关系型数据库管理系统,由瑞典MySQL AB公司开发。它以其高性能、稳定性和易用性著称,广泛应用于Web应用程序和企业级系统中...

    SQL详解_oracle

    【SQL详解_oracle】 SQL(Structured Query Language,结构化查询语言)是用于管理和处理关系数据库的标准语言,Oracle数据库系统就是支持SQL的一种强大的关系型数据库管理系统。Oracle SQL在SQL的基础上进行了扩展...

    ORACLE学习手册和案例

    - **用于文本搜索**:如果数据列主要用于全文搜索,则更适合使用全文索引。 **1.2.1.6 限制索引** 某些查询条件下,即使存在索引也无法有效利用。 **1.2.1.6.1 使用不等于操作符(、!=)** 当查询条件包含不等于...

    oracle 全文检索

    ### Oracle全文检索技术详解 #### 一、Oracle Text 概述 Oracle Text 是 Oracle 数据库提供的一个强大且灵活的全文检索工具,它允许用户在大量文本数据中进行高效的搜索。Oracle Text 在不同的版本中有着不同的名称...

    JAVA日记

    #### 全文索引实现步骤详解 1. **确认ORACLE Intermedia安装**:确保该模块已随ORACLE数据库一同安装。 2. **配置词法解析器**:ORACLE提供了针对不同语言的解析器,如英语环境下的`basic_lexer`,以及针对中文的`...

    省市县行政区域数据字典(mysql+oracle)

    此外,为了方便查询和使用,我们还应考虑建立合适的索引,比如针对常用查询字段(如省份或城市名称)创建全文索引或普通索引。同时,定期更新数据,以适应行政区划的变化。 总的来说,“省市县行政区域数据字典”是...

    oracle like 的优化

    ### Oracle LIKE 语句优化详解 #### 一、引言 在Oracle数据库中,LIKE操作符是用于模糊查询的关键字之一。它允许用户通过模式匹配的方式检索数据,从而增加了查询的灵活性。然而,这种灵活性也带来了性能上的挑战...

    Oracle SQL优化

    如果必须使用,则考虑创建全文索引或者优化查询逻辑。 ##### 2.4 ORDER BY 语句 - **问题**: ORDER BY 子句可能导致额外的数据排序操作,这会增加CPU负载。 - **原因**: 排序操作可能涉及大量的内存和磁盘I/O。 - ...

    mysql问题,常见的mysql面试问题

    - 支持全文索引,适合大数据量的只读应用。 - **InnoDB**: - 支持事务处理、行级锁定以及外键约束。 - 提供了更高级别的并发控制,更适合高并发的在线事务处理(OLTP)应用。 - 自动恢复机制,能够保证数据一致性...

    基于lucene搜索引擎的java源码

    1. **建立索引(Indexing)**:在源码中,你会看到`IndexWriter`类的使用,它是创建和更新索引的主要接口。首先,需要配置Analyzer,然后调用`addDocument(Document doc)`方法为每个待索引的文档创建一个新的条目。 ...

    资料管理系统源码.zip

    通常使用关键词搜索,通过在数据库中进行全文索引,实现快速定位。此外,还可支持模糊匹配、高级搜索等功能,提升用户体验。 六、界面设计 良好的用户界面是提升系统易用性的关键。界面应简洁明了,操作直观,符合...

    MYSQL开发学习笔记

    - 索引类型: 主键索引、唯一索引、全文索引等。 - 索引原理: 通过构建特殊的数据结构,如B树,加速数据检索。 - **分表**: 减少单个表的数据量(sharding)。 - 分表策略: 按时间、按ID范围等方式分散数据到多个表...

    基于JSP的家用电器销售网站源码.zip

    搜索功能可能涉及到SQL的LIKE操作或者全文索引技术。 8. **购物车与订单系统** 购物车功能需要维护用户的临时选购物品,订单系统则处理用户的购买请求,包括订单创建、支付接口集成(如支付宝、微信支付)、订单...

Global site tag (gtag.js) - Google Analytics