﻿<?xml version="1.0" encoding="utf-8" standalone="yes"?><rss version="2.0" xmlns:dc="http://purl.org/dc/elements/1.1/" xmlns:trackback="http://madskills.com/public/xml/rss/module/trackback/" xmlns:wfw="http://wellformedweb.org/CommentAPI/" xmlns:slash="http://purl.org/rss/1.0/modules/slash/"><channel><title>BlogJava-harrygoo learning-文章分类-oracle</title><link>http://www.blogjava.net/harrygoo/category/49691.html</link><description /><language>zh-cn</language><lastBuildDate>Wed, 28 Sep 2011 01:08:42 GMT</lastBuildDate><pubDate>Wed, 28 Sep 2011 01:08:42 GMT</pubDate><ttl>60</ttl><item><title>SQL优化34条  </title><link>http://www.blogjava.net/harrygoo/articles/359396.html</link><dc:creator>harrygoo</dc:creator><author>harrygoo</author><pubDate>Sat, 24 Sep 2011 00:41:00 GMT</pubDate><guid>http://www.blogjava.net/harrygoo/articles/359396.html</guid><description><![CDATA[<div><div fc05="" fc11="" nbw-blog="" ztag="" js-fs2=""><span style="font-family: simsun; line-height: 23px;">我们要做到不但会写SQL,还要做到写出性能优良的SQL,以下为笔者学习、摘录、并汇总部分资料与大家分享！<br />（1） 选择最有效率的表名顺序(只在基于规则的优化器中有效)：<br />ORACLE  的解析器按照从右到左的顺序处理FROM子句中的表名，FROM子句中写在最后的表(基础表 driving  table)将被最先处理，在FROM子句中包含多个表的情况下,你必须选择记录条数最少的表作为基础表。如果有3个以上的表连接查询,  那就需要选择交叉表(intersection table)作为基础表, 交叉表是指那个被其他表所引用的表.<br />（2） WHERE子句中的连接顺序．：<br />ORACLE采用自下而上的顺序解析WHERE子句,根据这个原理,表之间的连接必须写在其他WHERE条件之前, 那些可以过滤掉最大数量记录的条件必须写在WHERE子句的末尾.<br />（3） SELECT子句中避免使用 &#8216; * &#8216;：<br />ORACLE在解析的过程中, 会将'*' 依次转换成所有的列名, 这个工作是通过查询数据字典完成的, 这意味着将耗费更多的时间<br />（4） 减少访问数据库的次数：<br />ORACLE在内部执行了许多工作: 解析SQL语句, 估算索引的利用率, 绑定变量 , 读数据块等；<br />（5） 在SQL*Plus , SQL*Forms和Pro*C中重新设置ARRAYSIZE参数, 可以增加每次数据库访问的检索数据量 ,建议值为200<br />（6） 使用DECODE函数来减少处理时间：<br />使用DECODE函数可以避免重复扫描相同记录或重复连接相同的表.<br />（7） 整合简单,无关联的数据库访问：<br />如果你有几个简单的数据库查询语句,你可以把它们整合到一个查询中(即使它们之间没有关系)<br />（8） 删除重复记录：<br />最高效的删除重复记录方法 ( 因为使用了ROWID)例子：<br />DELETE FROM EMP E WHERE E.ROWID &gt; (SELECT MIN(X.ROWID)&nbsp;&nbsp;<br />FROM EMP X WHERE X.EMP_NO = E.EMP_NO);<br />（9） 用TRUNCATE替代DELETE：<br />当 删除表中的记录时,在通常情况下, 回滚段(rollback segments ) 用来存放可以被恢复的信息.  如果你没有COMMIT事务,ORACLE会将数据恢复到删除之前的状态(准确地说是恢复到执行删除命令之前的状况) 而当运用TRUNCATE时,  回滚段不再存放任何可被恢复的信息.当命令运行后,数据不能被恢复.因此很少的资源被调用,执行时间也会很短. (译者按:  TRUNCATE只在删除全表适用,TRUNCATE是DDL不是DML)<br />（10） 尽量多使用COMMIT：<br />只要有可能,在程序中尽量多使用COMMIT, 这样程序的性能得到提高,需求也会因为COMMIT所释放的资源而减少:&nbsp;&nbsp;<br />COMMIT所释放的资源:&nbsp;&nbsp;<br />a. 回滚段上用于恢复数据的信息.&nbsp;&nbsp;<br />b. 被程序语句获得的锁&nbsp;&nbsp;<br />c. redo log buffer 中的空间&nbsp;&nbsp;<br />d. ORACLE为管理上述3种资源中的内部花费<br />（11） 用Where子句替换HAVING子句：<br />避 免使用HAVING子句, HAVING 只会在检索出所有记录之后才对结果集进行过滤. 这个处理需要排序,总计等操作.  如果能通过WHERE子句限制记录的数目,那就能减少这方面的开销.  (非oracle中)on、where、having这三个都可以加条件的子句中，on是最先执行，where次之，having最后，因为on是先把不 符合条件的记录过滤后才进行统计，它就可以减少中间运算要处理的数据，按理说应该速度是最快的，where也应该比having快点的，因为它过滤数据后 才进行sum，在两个表联接时才用on的，所以在一个表的时候，就剩下where跟having比较了。在这单表查询统计的情况下，如果要过滤的条件没有 涉及到要计算字段，那它们的结果是一样的，只是where可以使用rushmore技术，而having就不能，在速度上后者要慢如果要涉及到计算的字 段，就表示在没计算之前，这个字段的值是不确定的，根据上篇写的工作流程，where的作用时间是在计算之前就完成的，而having就是在计算后才起作 用的，所以在这种情况下，两者的结果会不同。在多表联接查询时，on比where更早起作用。系统首先根据各个表之间的联接条件，把多个表合成一个临时表 后，再由where进行过滤，然后再计算，计算完后再由having进行过滤。由此可见，要想过滤条件起到正确的作用，首先要明白这个条件应该在什么时候 起作用，然后再决定放在那里<br />（12） 减少对表的查询：<br />在含有子查询的SQL语句中,要特别注意减少对表的查询.例子：<br />&nbsp; SELECT TAB_NAME FROM TABLES WHERE (TAB_NAME,DB_VER) = ( SELECT<br />TAB_NAME,DB_VER FROM TAB_COLUMNS WHERE VERSION = 604)<br />（13） 通过内部函数提高SQL效率.：<br />复杂的SQL往往牺牲了执行效率. 能够掌握上面的运用函数解决问题的方法在实际工作中是非常有意义的<br />（14） 使用表的别名(Alias)：<br />当在SQL语句中连接多个表时, 请使用表的别名并把别名前缀于每个Column上.这样一来,就可以减少解析的时间并减少那些由Column歧义引起的语法错误.<br />（15） 用EXISTS替代IN、用NOT EXISTS替代NOT IN：<br />在 许多基于基础表的查询中,为了满足一个条件,往往需要对另一个表进行联接.在这种情况下, 使用EXISTS(或NOT  EXISTS)通常将提高查询的效率. 在子查询中,NOT IN子句将执行一个内部的排序和合并. 无论在哪种情况下,NOT IN都是最低效的  (因为它对子查询中的表执行了一个全表遍历). 为了避免使用NOT IN ,我们可以把它改写成外连接(Outer Joins)或NOT  EXISTS.<br />例子：<br />（高效）SELECT * FROM EMP (基础表) WHERE EMPNO &gt; 0 AND EXISTS (SELECT &#8216;X' FROM DEPT WHERE DEPT.DEPTNO = EMP.DEPTNO AND LOC = &#8216;MELB')<br />(低效)SELECT * FROM EMP (基础表) WHERE EMPNO &gt; 0 AND DEPTNO IN(SELECT DEPTNO FROM DEPT WHERE LOC = &#8216;MELB')<br />（16） 识别'低效执行'的SQL语句：<br />虽然目前各种关于SQL优化的图形化工具层出不穷,但是写出自己的SQL工具来解决问题始终是一个最好的方法：<br />SELECT EXECUTIONS , DISK_READS, BUFFER_GETS,&nbsp;&nbsp;<br />ROUND((BUFFER_GETS-DISK_READS)/BUFFER_GETS,2) Hit_radio,&nbsp;&nbsp;<br />ROUND(DISK_READS/EXECUTIONS,2) Reads_per_run,&nbsp;&nbsp;<br />SQL_TEXT&nbsp;&nbsp;<br />FROM V$SQLAREA&nbsp;&nbsp;<br />WHERE EXECUTIONS&gt;0&nbsp;&nbsp;<br />AND BUFFER_GETS &gt; 0&nbsp;&nbsp;<br />AND (BUFFER_GETS-DISK_READS)/BUFFER_GETS &lt; 0.8&nbsp;&nbsp;<br />ORDER BY 4 DESC;</span><wbr><div><span style="font-family: simsun; line-height: 23px;"><br /></span></div><div><span style="font-family: simsun; line-height: 23px;">（17） 用索引提高效率：<br />索 引是表的一个概念部分,用来提高检索数据的效率，ORACLE使用了一个复杂的自平衡B-tree结构. 通常,通过索引查询数据比全表扫描要快.  当ORACLE找出执行查询和Update语句的最佳路径时, ORACLE优化器将使用索引. 同样在联结多个表时使用索引也可以提高效率.  另一个使用索引的好处是,它提供了主键(primary key)的唯一性验证.。那些LONG或LONG RAW数据类型, 你可以索引几乎所有的列.  通常, 在大型表中使用索引特别有效. 当然,你也会发现, 在扫描小表时,使用索引同样能提高效率.  虽然使用索引能得到查询效率的提高,但是我们也必须注意到它的代价. 索引需要空间来存储,也需要定期维护, 每当有记录在表中增减或索引列被修改时,  索引本身也会被修改. 这意味着每条记录的INSERT , DELETE , UPDATE将为此多付出4 , 5 次的磁盘I/O .  因为索引需要额外的存储空间和处理,那些不必要的索引反而会使查询反应时间变慢.。定期的重构索引是有必要的.：<br />ALTER INDEX &lt;INDEXNAME&gt; REBUILD &lt;TABLESPACENAME&gt;<br />18） 用EXISTS替换DISTINCT：<br />当提交一个包含一对多表信息(比如部门表和雇员表)的查询时,避免在SELECT子句中使用DISTINCT. 一般可以考虑用EXIST替换, EXISTS 使查询更为迅速,因为RDBMS核心模块将在子查询的条件一旦满足后,立刻返回结果. 例子：<br />&nbsp; (低效):&nbsp;&nbsp;<br />SELECT DISTINCT DEPT_NO,DEPT_NAME FROM DEPT D , EMP E&nbsp;&nbsp;<br />WHERE D.DEPT_NO = E.DEPT_NO&nbsp;&nbsp;<br />(高效):&nbsp;&nbsp;<br />SELECT DEPT_NO,DEPT_NAME FROM DEPT D WHERE EXISTS ( SELECT &#8216;X'&nbsp;&nbsp;<br />FROM EMP E WHERE E.DEPT_NO = D.DEPT_NO);<br />（19） sql语句用大写的；因为oracle总是先解析sql语句，把小写的字母转换成大写的再执行<br />（20） 在java代码中尽量少用连接符&#8220;＋&#8221;连接字符串！<br />（21） 避免在索引列上使用NOT 通常，　<br />我们要避免在索引列上使用NOT, NOT会产生在和在索引列上使用函数相同的影响. 当ORACLE&#8221;遇到&#8221;NOT,他就会停止使用索引转而执行全表扫描.<br />（22） 避免在索引列上使用计算．<br />WHERE子句中，如果索引列是函数的一部分．优化器将不使用索引而使用全表扫描．&nbsp;&nbsp;<br />举例:&nbsp;&nbsp;<br />低效：&nbsp;&nbsp;<br />SELECT &#8230; FROM DEPT WHERE SAL * 12 &gt; 25000;&nbsp;&nbsp;<br />高效:&nbsp;&nbsp;<br />SELECT &#8230; FROM DEPT WHERE SAL &gt; 25000/12;<br />（23） 用&gt;=替代&gt;<br />高效:&nbsp;&nbsp;<br />SELECT * FROM EMP WHERE DEPTNO &gt;=4&nbsp;&nbsp;<br />低效:&nbsp;&nbsp;<br />SELECT * FROM EMP WHERE DEPTNO &gt;3&nbsp;&nbsp;<br />两者的区别在于, 前者DBMS将直接跳到第一个DEPT等于4的记录而后者将首先定位到DEPTNO=3的记录并且向前扫描到第一个DEPT大于3的记录.<br />（24） 用UNION替换OR (适用于索引列)<br />通 常情况下, 用UNION替换WHERE子句中的OR将会起到较好的效果. 对索引列使用OR将造成全表扫描. 注意, 以上规则只针对多个索引列有效.  如果有column没有被索引, 查询效率可能会因为你没有选择OR而降低. 在下面的例子中, LOC_ID 和REGION上都建有索引.&nbsp;&nbsp;<br />高效:&nbsp;&nbsp;<br />SELECT LOC_ID , LOC_DESC , REGION&nbsp;&nbsp;<br />FROM LOCATION&nbsp;&nbsp;<br />WHERE LOC_ID = 10&nbsp;&nbsp;<br />UNION&nbsp;&nbsp;<br />SELECT LOC_ID , LOC_DESC , REGION&nbsp;&nbsp;<br />FROM LOCATION&nbsp;&nbsp;<br />WHERE REGION = &#8220;MELBOURNE&#8221;&nbsp;&nbsp;<br />低效:&nbsp;&nbsp;<br />SELECT LOC_ID , LOC_DESC , REGION&nbsp;&nbsp;<br />FROM LOCATION&nbsp;&nbsp;<br />WHERE LOC_ID = 10 OR REGION = &#8220;MELBOURNE&#8221;&nbsp;&nbsp;<br />如果你坚持要用OR, 那就需要返回记录最少的索引列写在最前面.<br />（25） 用IN来替换OR &nbsp;&nbsp;<br />这是一条简单易记的规则，但是实际的执行效果还须检验，在ORACLE8i下，两者的执行路径似乎是相同的．　<br />低效:&nbsp;&nbsp;<br />SELECT&#8230;. FROM LOCATION WHERE LOC_ID = 10 OR LOC_ID = 20 OR LOC_ID = 30&nbsp;&nbsp;<br />高效&nbsp;&nbsp;<br />SELECT&#8230; FROM LOCATION WHERE LOC_IN IN (10,20,30);<br />（26） 避免在索引列上使用IS NULL和IS NOT NULL<br />避 免在索引中使用任何可以为空的列，ORACLE将无法使用该索引．对于单列索引，如果列包含空值，索引中将不存在此记录.  对于复合索引，如果每个列都为空，索引中同样不存在此记录.　如果至少有一个列不为空，则记录存在于索引中．举例:  如果唯一性索引建立在表的A列和B列上, 并且表中存在一条记录的A,B值为(123,null) ,  ORACLE将不接受下一条具有相同A,B值（123,null）的记录(插入).  然而如果所有的索引列都为空，ORACLE将认为整个键值为空而空不等于空. 因此你可以插入1000 条具有相同键值的记录,当然它们都是空!  因为空值不存在于索引列中,所以WHERE子句中对索引列进行空值比较将使ORACLE停用该索引.<br />低效: (索引失效)&nbsp;&nbsp;<br />SELECT &#8230; FROM DEPARTMENT WHERE DEPT_CODE IS NOT NULL;&nbsp;&nbsp;<br />高效: (索引有效)&nbsp;&nbsp;<br />SELECT &#8230; FROM DEPARTMENT WHERE DEPT_CODE &gt;=0;<br />（27） 总是使用索引的第一个列：<br />如果索引是建立在多个列上, 只有在它的第一个列(leading column)被where子句引用时,优化器才会选择使用该索引. 这也是一条简单而重要的规则，当仅引用索引的第二个列时,优化器使用了全表扫描而忽略了索引<br />28） 用UNION-ALL 替换UNION ( 如果有可能的话)：<br />当SQL  语句需要UNION两个查询结果集合时,这两个结果集合会以UNION-ALL的方式被合并, 然后在输出最终结果前进行排序. 如果用UNION  ALL替代UNION, 这样排序就不是必要了. 效率就会因此得到提高. 需要注意的是，UNION ALL 将重复输出两个结果集合中相同记录.  因此各位还是要从业务需求分析使用UNION ALL的可行性. UNION  将对结果集合排序,这个操作会使用到SORT_AREA_SIZE这块内存. 对于这块内存的优化也是相当重要的.  下面的SQL可以用来查询排序的消耗量<br />低效：&nbsp;&nbsp;<br />SELECT ACCT_NUM, BALANCE_AMT&nbsp;&nbsp;<br />FROM DEBIT_TRANSACTIONS&nbsp;&nbsp;<br />WHERE TRAN_DATE = '31-DEC-95'&nbsp;&nbsp;<br />UNION&nbsp;&nbsp;<br />SELECT ACCT_NUM, BALANCE_AMT&nbsp;&nbsp;<br />FROM DEBIT_TRANSACTIONS&nbsp;&nbsp;<br />WHERE TRAN_DATE = '31-DEC-95'&nbsp;&nbsp;<br />高效:&nbsp;&nbsp;<br />SELECT ACCT_NUM, BALANCE_AMT&nbsp;&nbsp;<br />FROM DEBIT_TRANSACTIONS&nbsp;&nbsp;<br />WHERE TRAN_DATE = '31-DEC-95'&nbsp;&nbsp;<br />UNION ALL&nbsp;&nbsp;<br />SELECT ACCT_NUM, BALANCE_AMT&nbsp;&nbsp;<br />FROM DEBIT_TRANSACTIONS&nbsp;&nbsp;<br />WHERE TRAN_DATE = '31-DEC-95'<br />（29） 用WHERE替代ORDER BY：<br />ORDER BY 子句只在两种严格的条件下使用索引.&nbsp;&nbsp;<br />ORDER BY中所有的列必须包含在相同的索引中并保持在索引中的排列顺序.&nbsp;&nbsp;<br />ORDER BY中所有的列必须定义为非空.&nbsp;&nbsp;<br />WHERE子句使用的索引和ORDER BY子句中所使用的索引不能并列.<br />例如:&nbsp;&nbsp;<br />表DEPT包含以下列:&nbsp;&nbsp;<br />DEPT_CODE PK NOT NULL&nbsp;&nbsp;<br />DEPT_DESC NOT NULL&nbsp;&nbsp;<br />DEPT_TYPE NULL<br />低效: (索引不被使用)&nbsp;&nbsp;<br />SELECT DEPT_CODE FROM DEPT ORDER BY DEPT_TYPE&nbsp;&nbsp;<br />高效: (使用索引)&nbsp;&nbsp;<br />SELECT DEPT_CODE FROM DEPT WHERE DEPT_TYPE &gt; 0<br />（30） 避免改变索引列的类型.:<br />当比较不同数据类型的数据时, ORACLE自动对列进行简单的类型转换.&nbsp;&nbsp;<br />假设 EMPNO是一个数值类型的索引列.&nbsp;&nbsp;<br />SELECT &#8230; FROM EMP WHERE EMPNO = &#8216;123'&nbsp;&nbsp;<br />实际上,经过ORACLE类型转换, 语句转化为:&nbsp;&nbsp;<br />SELECT &#8230; FROM EMP WHERE EMPNO = TO_NUMBER(&#8216;123')&nbsp;&nbsp;<br />幸运的是,类型转换没有发生在索引列上,索引的用途没有被改变.&nbsp;&nbsp;<br />现在,假设EMP_TYPE是一个字符类型的索引列.&nbsp;&nbsp;<br />SELECT &#8230; FROM EMP WHERE EMP_TYPE = 123&nbsp;&nbsp;<br />这个语句被ORACLE转换为:&nbsp;&nbsp;<br />SELECT &#8230; FROM EMP WHERETO_NUMBER(EMP_TYPE)=123&nbsp;&nbsp;<br />因为内部发生的类型转换, 这个索引将不会被用到! 为了避免ORACLE对你的SQL进行隐式的类型转换, 最好把类型转换用显式表现出来. 注意当字符和数值比较时, ORACLE会优先转换数值类型到字符类型<br />（31） 需要当心的WHERE子句:<br />某些SELECT 语句中的WHERE子句不使用索引. 这里有一些例子.&nbsp;&nbsp;<br />在 下面的例子里, (1)&#8216;!=' 将不使用索引. 记住, 索引只能告诉你什么存在于表中, 而不能告诉你什么不存在于表中. (2)  &#8216;||'是字符连接函数. 就象其他函数那样, 停用了索引. (3) &#8216;+'是数学函数. 就象其他数学函数那样, 停用了索引.  (4)相同的索引列不能互相比较,这将会启用全表扫描.<br />（32） a. 如果检索数据量超过30%的表中记录数.使用索引将没有显著的效率提高.&nbsp;&nbsp;<br />b. 在特定情况下, 使用索引也许会比全表扫描慢, 但这是同一个数量级上的区别. 而通常情况下,使用索引比全表扫描要块几倍乃至几千倍!<br />（33） 避免使用耗费资源的操作:<br />带有DISTINCT,UNION,MINUS,INTERSECT,ORDER BY的SQL语句会启动SQL引擎&nbsp;&nbsp;<br />执 行耗费资源的排序(SORT)功能. DISTINCT需要一次排序操作, 而其他的至少需要执行两次排序. 通常, 带有UNION, MINUS ,  INTERSECT的SQL语句都可以用其他方式重写. 如果你的数据库的SORT_AREA_SIZE调配得好, 使用UNION , MINUS,  INTERSECT也是可以考虑的, 毕竟它们的可读性很强<br />（34） 优化GROUP BY:<br />提高GROUP BY 语句的效率, 可以通过将不需要的记录在GROUP BY 之前过滤掉.下面两个查询返回相同结果但第二个明显就快了许多.<br />低效:&nbsp;&nbsp;<br />SELECT JOB , AVG(SAL)&nbsp;&nbsp;<br />FROM EMP&nbsp;&nbsp;<br />GROUP by JOB&nbsp;&nbsp;<br />HAVING JOB = &#8216;PRESIDENT'&nbsp;&nbsp;<br />OR JOB = &#8216;MANAGER'&nbsp;&nbsp;<br />高效:&nbsp;&nbsp;<br />SELECT JOB , AVG(SAL)&nbsp;&nbsp;<br />FROM EMP&nbsp;&nbsp;<br />WHERE JOB = &#8216;PRESIDENT'&nbsp;&nbsp;<br />OR JOB = &#8216;MANAGER'&nbsp;&nbsp;<br />GROUP by JOB</span></div></div></div><img src ="http://www.blogjava.net/harrygoo/aggbug/359396.html" width = "1" height = "1" /><br><br><div align=right><a style="text-decoration:none;" href="http://www.blogjava.net/harrygoo/" target="_blank">harrygoo</a> 2011-09-24 08:41 <a href="http://www.blogjava.net/harrygoo/articles/359396.html#Feedback" target="_blank" style="text-decoration:none;">发表评论</a></div>]]></description></item><item><title>oracle中的exists 和not exists 用法详解  </title><link>http://www.blogjava.net/harrygoo/articles/359395.html</link><dc:creator>harrygoo</dc:creator><author>harrygoo</author><pubDate>Sat, 24 Sep 2011 00:16:00 GMT</pubDate><guid>http://www.blogjava.net/harrygoo/articles/359395.html</guid><description><![CDATA[<div><p style="text-indent: 2em">有两个简单例子，以说明 &#8220;exists&#8221;和&#8220;in&#8221;的效率问题</p> <p style="text-indent: 2em">1) select * from T1 where exists(select 1 from T2 where T1.a=T2.a) ;</p> <p style="text-indent: 2em">&nbsp;<wbr>&nbsp;<wbr>&nbsp;<wbr> T1数据量小而T2数据量非常大时，T1&lt;&lt;T2 时，1) 的查询效率高。</p> <p style="text-indent: 2em">2) select * from T1 where T1.a in (select T2.a from T2) ;</p> <p style="text-indent: 2em">&nbsp;<wbr>&nbsp;<wbr>&nbsp;<wbr>&nbsp;<wbr> T1数据量非常大而T2数据量小时，T1&gt;&gt;T2 时，2) 的查询效率高。</p> <p style="text-indent: 2em">exists 用法：</p> <p style="text-indent: 2em">请注意 1）句中的有颜色字体的部分 ，理解其含义；</p> <p style="text-indent: 2em">其中 &#8220;select 1 from T2 where T1.a=T2.a&#8221; 相当于一个关联表查询，相当于</p> <p style="text-indent: 2em">&#8220;select 1 from T1,T2&nbsp;<wbr>&nbsp;<wbr>&nbsp;<wbr>&nbsp;<wbr> where T1.a=T2.a&#8221;</p> <p style="text-indent: 2em">但是，如果你当当执行 1） 句括号里的语句，是会报语法错误的，这也是使用exists需要注意的地方。</p> <p style="text-indent: 2em">&#8220;exists（xxx）&#8221;就表示括号里的语句能不能查出记录，它要查的记录是否存在。</p> <p style="text-indent: 2em">因此&#8220;select 1&#8221;这里的 &#8220;1&#8221;其实是无关紧要的，换成&#8220;*&#8221;也没问题，它只在乎括号里的数据能不能查找出来，是否存在这样的记录，如果存在，这 1） 句的where 条件成立。</p> <p style="text-indent: 2em">&nbsp;<wbr></p> <p style="text-indent: 2em">in 的用法：</p> <p style="text-indent: 2em">继续引用上面的例子</p> <p style="text-indent: 2em">&#8220;2) select * from T1 where T1.a in (select T2.a from T2) &#8221;</p> <p style="text-indent: 2em">这里的&#8220;in&#8221;后面括号里的语句搜索出来的字段的内容一定要相对应，一般来说，T1和T2这两个表的a字段表达的意义应该是一样的，否则这样查没什么意义。</p> <p style="text-indent: 2em">打个比方：T1，T2表都有一个字段，表示工单号，但是T1表示工单号的字段名叫&#8220;ticketid&#8221;，T2则为&#8220;id&#8221;，但是其表达的意义是一样的，而且数据格式也是一样的。这时，用 2）的写法就可以这样：</p> <p style="text-indent: 2em">&#8220;select * from T1 where T1.ticketid in (select T2.id from T2) &#8221;</p> <p style="text-indent: 2em">Select name from employee where name not in (select name from student);</p> <p style="text-indent: 2em">Select name from employee where not exists (select name from student);</p> <p style="text-indent: 2em">第一句SQL语句的执行效率不如第二句。</p> <p style="text-indent: 2em;">通过使用EXISTS，Oracle会首先检查主查询，然后运行子查询直到它找到第一个匹配 项，这就节省了时间。Oracle在执行IN子查询时，首先执行子查询，并将获得的结果列表存放在一个加了索引的临时表中。在执行子查询之前，系统先将主 查询挂起，待子查询执行完毕，存放在临时表中以后再执行主查询。这也就是使用EXISTS比使用IN通常查询速度快的原因。</p><p style="text-indent: 2em;">&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;</p><p style="text-indent: 2em;"><br /></p><p style="text-indent: 2em;"></p><div><p style="text-indent: 2em;">在Oracle SQL中取数据时有时要用到in 和 exists 那么他们有什么区别呢？</p>  <p style="text-indent: 2em;">1 性能上的比较</p>  <p style="text-indent: 2em;">比如Select * from T1 where x in ( select y from T2 )</p>  <p style="text-indent: 2em;">执行的过程相当于:</p>  <p style="text-indent: 2em;">select * </p>  <p style="text-indent: 2em;">&nbsp; from t1, ( select distinct y from t2 ) t2</p>  <p style="text-indent: 2em;">&nbsp;where t1.x = t2.y;</p>  <p style="text-indent: 2em;">相对的</p>  <p style="text-indent: 2em;">select * from t1 where exists ( select null from t2 where y = x )</p>  <p style="text-indent: 2em;">执行的过程相当于:</p><div style="background-color: #eeeeee; font-size: 13px; border: 1px solid #cccccc; padding: 4px 5px 4px 4px; width: 98%;"><!--<br /><br />Code highlighting produced by Actipro CodeHighlighter (freeware)<br />http://www.CodeHighlighter.com/<br /><br />--><span style="color: #0000FF; ">for</span><span style="color: #000000; ">&nbsp;x&nbsp;</span><span style="color: #808080; ">in</span><span style="color: #000000; ">&nbsp;(&nbsp;</span><span style="color: #0000FF; ">select</span><span style="color: #000000; ">&nbsp;</span><span style="color: #808080; ">*</span><span style="color: #000000; ">&nbsp;</span><span style="color: #0000FF; ">from</span><span style="color: #000000; ">&nbsp;t1&nbsp;)<br /><br />&nbsp;&nbsp;&nbsp;loop<br /><br />&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</span><span style="color: #0000FF; ">if</span><span style="color: #000000; ">&nbsp;(&nbsp;</span><span style="color: #808080; ">exists</span><span style="color: #000000; ">&nbsp;(&nbsp;</span><span style="color: #0000FF; ">select</span><span style="color: #000000; ">&nbsp;</span><span style="color: #0000FF; ">null</span><span style="color: #000000; ">&nbsp;</span><span style="color: #0000FF; ">from</span><span style="color: #000000; ">&nbsp;t2&nbsp;</span><span style="color: #0000FF; ">where</span><span style="color: #000000; ">&nbsp;y&nbsp;</span><span style="color: #808080; ">=</span><span style="color: #000000; ">&nbsp;x.x&nbsp;)<br /><br />&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</span><span style="color: #0000FF; ">then</span><span style="color: #000000; "><br /><br />&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;OUTPUT&nbsp;THE&nbsp;RECORD<br /><br />&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</span><span style="color: #0000FF; ">end</span><span style="color: #000000; ">&nbsp;</span><span style="color: #0000FF; ">if</span><span style="color: #000000; "><br /><br /></span><span style="color: #0000FF; ">end</span><span style="color: #000000;">&nbsp;loop</span></div><p style="text-indent: 2em;">表&nbsp;T1 不可避免的要被完全扫描一遍</p>  <p style="text-indent: 2em;">分别适用在什么情况?</p>  <p style="text-indent: 2em;">以子查询&nbsp;( select y from T2 )为考虑方向</p>  <p style="text-indent: 2em;">如果子查询的结果集很大需要消耗很多时间，但是T1比较小执行( select null from t2 where y = x.x )非常快，那么exists就比较适合用在这里</p>  <p style="text-indent: 2em;">相对应得子查询的结果集比较小的时候就应该使用in.</p></div><br /><p>&nbsp;</p></div><img src ="http://www.blogjava.net/harrygoo/aggbug/359395.html" width = "1" height = "1" /><br><br><div align=right><a style="text-decoration:none;" href="http://www.blogjava.net/harrygoo/" target="_blank">harrygoo</a> 2011-09-24 08:16 <a href="http://www.blogjava.net/harrygoo/articles/359395.html#Feedback" target="_blank" style="text-decoration:none;">发表评论</a></div>]]></description></item><item><title>索引与约束  </title><link>http://www.blogjava.net/harrygoo/articles/359394.html</link><dc:creator>harrygoo</dc:creator><author>harrygoo</author><pubDate>Fri, 23 Sep 2011 23:47:00 GMT</pubDate><guid>http://www.blogjava.net/harrygoo/articles/359394.html</guid><description><![CDATA[&nbsp;&nbsp;&nbsp;&nbsp; 摘要: 1.数据库的索引  &nbsp; &nbsp;&nbsp;&nbsp;&nbsp; 可以将索引概念应用到数据库表上。当一个表含有大量的记录时，Oracle查找该表中的特写记录要花很长的时间&#8212;&#8212;就像花很长时间翻看全书来查找某个主题一样。Oracle有一个易于使用的功能，即可以建立一个次隐藏表，该表包含主表中的一个或多个重要的列，以及在主表中的指相应行的指针。 在这里，与书的页...&nbsp;&nbsp;<a href='http://www.blogjava.net/harrygoo/articles/359394.html'>阅读全文</a><img src ="http://www.blogjava.net/harrygoo/aggbug/359394.html" width = "1" height = "1" /><br><br><div align=right><a style="text-decoration:none;" href="http://www.blogjava.net/harrygoo/" target="_blank">harrygoo</a> 2011-09-24 07:47 <a href="http://www.blogjava.net/harrygoo/articles/359394.html#Feedback" target="_blank" style="text-decoration:none;">发表评论</a></div>]]></description></item></channel></rss>