数据库优化方案.docx

上传人:b****7 文档编号:10280160 上传时间:2023-02-09 格式:DOCX 页数:11 大小:54.64KB
下载 相关 举报
数据库优化方案.docx_第1页
第1页 / 共11页
数据库优化方案.docx_第2页
第2页 / 共11页
数据库优化方案.docx_第3页
第3页 / 共11页
数据库优化方案.docx_第4页
第4页 / 共11页
数据库优化方案.docx_第5页
第5页 / 共11页
点击查看更多>>
下载资源
资源描述

数据库优化方案.docx

《数据库优化方案.docx》由会员分享,可在线阅读,更多相关《数据库优化方案.docx(11页珍藏版)》请在冰豆网上搜索。

数据库优化方案.docx

数据库优化方案

数据库优化方案

数据库优化方案

1.高效地进行SQL语句设计:

通常情况下,可以采用下面的方法优化SQL对数据操作的表现:

(1)减少对数据库的查询次数,即减少对系统资源的请求,使用快照和显形图等分布式数据库对象可以减少对数据库的查询次数。

(2)尽量使用相同的或非常类似的SQL语句进行查询,这样不仅充分利用SQL共享池中的已经分析的语法树,要查询的数据在SGA中命中的可能性也会大大增加。

(3)避免不带任何条件的SQL语句的执行。

没有任何条件的SQL语句在执行时,通常要进行FTS,数据库先定位一个数据块,然后按顺序依次查找其它数据,对于大型表这将是一个漫长的过程。

(4)如果对有些表中的数据有约束,最好在建表的SQL语句用描述完整性来实现,而不是用SQL程序中实现。

一、操作符优化:

1、IN操作符

用IN写出来的SQL的优点是比较容易写及清晰易懂,这比较适合现代软件开发的风格。

但是用IN的SQL性能总是比较低的,从Oracle执行的步骤来分析用IN的SQL与不用IN的SQL有以下区别:

ORACLE试图将其转换成多个表的连接,如果转换不成功则先执行IN里面的子查询,再查询外层的表记录,如果转换成功则直接采用多个表的连接方式查询。

由此可见用IN的SQL至少多了一个转换的过程。

一般的SQL都可以转换成功,但对于含有分组统计等方面的SQL就不能转换了。

在业务密集的SQL当中尽量不采用IN操作符。

优化sql时,经常碰到使用in的语句,一定要用exists把它给换掉,因为Oracle在处理In时是按Or的方式做的,即使使用了索引也会很慢。

2、NOTIN操作符

强列推荐不使用的,因为它不能应用表的索引。

用NOTEXISTS或(外连接+判断为空)方案代替

3、ISNULL或ISNOTNULL操作

判断字段是否为空一般是不会应用索引的,因为B树索引是不索引空值的。

用其它相同功能的操作运算代替,aisnotnull改为a>0或a>’’等。

不允许字段为空,而用一个缺省值代替空值,如业扩申请中状态字段不允许为空,缺省为申请。

避免在索引列上使用ISNULL和ISNOTNULL避免在索引中使用任何可以为空的列,ORACLE将无法使用该索引.对于单列索引,如果列包含空值,索引中将不存在此记录.对于复合索引,如果每个列都为空,索引中同样不存在此记录.如果至少有一个列不为空,则记录存在于索引中.举例:

如果唯一性索引建立在表的A列和B列上,并且表中存在一条记录的A,B值为(123,null),ORACLE将不接受下一条具有相同A,B值(123,null)的记录(插入).然而如果所有的索引列都为空,ORACLE将认为整个键值为

记录索引。

用>=替代>

高效:

SELECT…FROMDEPARTMENTWHEREDEPT_CODE>=0;

低效:

SELECT*FROMEMPWHEREDEPTNO>3

两者的区别在于,前者DBMS将直接跳到第一个DEPT等于4的记录而后者将首先定位到DEPTNO=3的记录并且向前扫描到第一个DEPT大于3的记录.

5、LIKE操作符:

LIKE操作符可以应用通配符查询,里面的通配符组合可能达到几乎是任意的查询,但是如果用得不好则会产生性能上的问题,如LIKE‘%5400%’这种查询不会引用索引,而LIKE‘X5400%’则会引用范围索引。

一个实际例子:

用YW_YHJBQK表中营业编号后面的户标识号可来查询营业编号YY_BHLIKE‘%5400%’这个条件会产生全表扫描,如果改成YY_BHLIKE’X5400%’ORYY_BHLIKE’B5400%’

则会利用YY_BH的索引进行两个范围的查询,性能肯定大大提高。

6、用EXISTS替换DISTINCT:

当提交一个包含一对多表信息(比如部门表和雇员表)的查询时,避免在SELECT子句中使用DISTINCT.一般可以考虑用EXIST替换,EXISTS使查询更为迅速,因为RDBMS核心模块将在子查询的条件一旦满足后,立刻返回结果.

例子:

(低效):

SELECTDISTINCTDEPT_NO,DEPT_NAMEFROMDEPTD,EMPEWHERED.DEPT_NO=E.DEPT_NO

(高效):

SELECTDEPT_NO,DEPT_NAMEFROMDEPTDWHEREEXISTS

  (SELECT'X'FROMEMPEWHEREE.DEPT_NO=D.DEPT_NO);

如:

用EXISTS替代IN、用NOTEXISTS替代NOTIN:

在许多基于基础表的查询中,为了满足一个条件,往往需要对另一个表进行联接.在这种情况下,使用EXISTS(或NOTEXISTS)通常将提高查询的效率.在子查询中,NOTIN子句将执行一个内部的排序和合并.无论在哪种情况下,NOTIN都是最低效的(因为它对子查询中的表执行了一个全表遍历).为了避免使用NOTIN,我们可以把它改写成外连接(OuterJoins)或NOTEXISTS.

例子:

(高效):

SELECT*FROMEMP(基础表)WHEREEMPNO>0ANDEXISTS

  (SELECT'X'FROMDEPTWHEREDEPT.DEPTNO=EMP.DEPTNOANDLOC='MELB')

(低效):

SELECT*FROMEMP(基础表)WHEREEMPNO>0ANDDEPTNOIN

  (SELECTDEPTNOFROMDEPTWHERELOC='MELB')

7、用UNION替换OR(适用于索引列)

通常情况下,用UNION替换WHERE子句中的OR将会起到较好的效果.对索引列使用OR将造成全表扫描.注意,以上规则只针对多个索引列有效.如果有column没有被索引,查询效率可能会因为你没有选择OR而降低.在下面的例子中,LOC_ID和REGION上都建有索引.

(高效):

SELECTLOC_ID,LOC_DESC,REGIONFROMLOCATIONWHERELOC_ID=10

  UNIONSELECTLOC_ID,LOC_DESC,REGIONFROMLOCATIONWHEREREGION='MELBOURNE'

(低效):

SELECTLOC_ID,LOC_DESC,REGIONFROMLOCATIONWHERELOC_ID=10ORREGION='MELBOURNE'

 

如果你坚持要用OR,那就需要返回记录最少的索引列写在最前面.

8、用IN来替换OR

这是一条简单易记的规则,但是实际的执行效果还须检验,在ORACLE8i下,两者的执行路径似乎是相同的.

低效:

SELECT….FROMLOCATIONWHERELOC_ID=10ORLOC_ID=20ORLOC_ID=30

高效:

SELECT…FROMLOCATIONWHERELOC_ININ(10,20,30);

二、SQL语句结构优化

1、SELECT子句中避免使用‘*‘:

2、用TRUNCATE替代DELETE:

用TRUNCATE替代DELETE删除全表记录:

(大数据量的表用次方法)

当删除表中的记录时,在通常情况下,回滚段(rollbacksegments)用来存放可以被恢复的信息.如果你没有COMMIT事务,ORACLE会将数据恢复到删除之前的状态(准确地说是恢复到执行删除命令之前的状况)而当运用TRUNCATE时,回滚段不再存放任何可被恢复的信息.

3、用Where子句替换HAVING子句:

避免使用HAVING子句,HAVING只会在检索出所有记录之后才对结果集进行过滤.这个处理需要排序,总计等操作.如果能通过WHERE子句限制记录的数目,那就能减少这方面的开销.(非oracle中)on、where、having这三个都可以加条件的子句中,on是最先执行,where次之,having最后,因为on是先把不符合条件的记录过滤后才进行统计,它就可以减少中间运算要处理的数据,按理说应该速度是最快的,where也应该比having快点的

4、sql语句用大写

因为oracle总是先解析sql语句,把小写的字母转换成大写的再执行。

5、在Java代码中尽量少用连接符“+”连接字符串!

6、避免改变索引列的类型.:

当比较不同数据类型的数据时,ORACLE自动对列进行简单的类型转换.假设EMPNO是一个数值类型的索引列.

SELECT…FROMEMPWHEREEMPNO=‘123'实际上,经过ORACLE类型转换,语句转化为:

SELECT…FROMEMPWHEREEMPNO=TO_NUMBER(‘123')

幸运的是,类型转换没有发生在索引列上,索引的用途没有被改变.现在,假设EMP_TYPE是一个字符类型的索引列.

SELECT…FROMEMPWHEREEMP_TYPE=123

这个语句被ORACLE转换为:

SELECT…FROMEMPWHERETO_NUMBER(EMP_TYPE)=123

因为内部发生的类型转换,这个索引将不会被用到!

为了避免ORACLE对你的SQL进行隐式的类型转换,最好把类型转换用显式表现出来.注意当字符和数值比较时,ORACLE会优先转换数值类型到字符类型

7、优化GROUPBY:

提高GROUPBY语句的效率,可以通过将不需要的记录在GROUPBY之前过滤掉.下面两个

查询返回相同结果但第二个明显就快了许多.

低效:

1SELECTJOB,AVG(SAL)FROMEMPGROUPbyJOBHAVINGJOB='PRESIDENT'ORJOB='MANAGER'

高效:

1SELECTJOB,AVG(SAL)FROMEMPWHEREJOB='PRESIDENT'ORJOB='MANAGER'GROUPbyJOB

数据库优化方案

1.利用表分区

分区将数据在物理上分隔开,不同分区的数据可以制定保存在处于不同磁盘上的数据文件里。

这样,当对这个表进行查询时,只需要在表分区中进行扫描,而不必进行全表扫描,明显缩短了查询时间,另外处于不同磁盘的分区也将对这个表的数据传输分散在不同的磁盘I/O,一个精心设置的分区可以将数据传输对磁盘I/O竞争均匀地分散开。

对数据量大的时时表可采取此方法。

可按月自动建表分区。

2.别名的使用

别名是大型数据库的应用技巧,就是表名、列名在查询中以一个字母为别名,查询速度要比建连接表快1.5倍。

3.索引Index的优化设计

索引可以大大加快数据库的查询速度,索引把表中的逻辑值映射到安全的RowID,因此索引能进行快速定位数据的物理地址。

对一个建有索引的大型表的查询时,索引数据可能会用完所有的数据块缓存空间,ORACLE不得不频繁地进行磁盘读写来获取数据,因此在对一个大型表进行分区之后,可以根据相应的分区建立分区索引。

但是个人觉得不是所有的表都需要建立索引,只针对大数据量的表建立索引。

缺点:

第一,创建索引和维护索引要耗费时间,这种时间随着数据量的增加而增加。

第二,索引需要占物理空间,除了数据表占数据空间之外,每一个索引还要占一定的物理空间,如果要建立聚簇索引,那么需要的空间就会更大。

第三,当对表中的数据进行增加、删除和修改的时候,索引也要动态的维护,这样就降低了数据的维护速度。

索引需要维护:

为了维护系统性能,索引在创建之后,由于频繁地对数据进行增加、删除、修改等操作使得索引页发生碎块,因此,必须对索引进行维护。

4.调整硬盘I/O

这一步是在信息系统开发之前完成的。

数据库管理员可以将组成同一个表空间的数据文件放在不同的硬盘上,做到硬盘之间I/O负载均衡。

在磁盘比较富裕的情况下还应该遵循以下原则:

●将表和索引分开;

●创造用户表空间,与系统表空间(system)分开磁盘;

●创建表和索引时指定不同的表空间;

●创建回滚段专用的表空间,防止空间竞争影响事务的完成;

●创建临时表空间用于排序操作,尽可能的防止数据库碎片存在于多个表空间中。

我们在使用物化视图的过程中基本可以“把它当作一个实际的数据表来看待”,不用再担心视图本身的基础表的效率、优化等

物化视图

1.对于复杂而高消耗的查询,如果使用频繁,应建成物化视图

2.物化视图是一种典型的以空间换时间的性能优化方式

3.对于更新频繁的表慎用物化视图

4.选择合适的刷新方式

一般的视图是虚拟的,而物化视图是实实在在的数据区域,是要占据存储空间的。

当然,物化视图在创建和管理上和一般的视图有不同的地方。

相比来讲,物化视图占用了一定的存储空间,另外系统刷新物化视图也需要耗费一定的资源,但是它却换来了效率和灵活性。

减少IO与网络传输次数

1.尽量用较少的数据库请求,获取到需要的数据,能一次性取出的不分多次取出

2.对于频繁操作数据库的批量操作,应采用存储过程,减少不必要的网络传输

死锁与阻塞

1.对于需要频繁更新的数据,尽量避免放在长事务中,以免导致连锁反应

2.不是迫不得已,最好不要在ORACLE锁机制外再加自己设计的锁

3.减少事务大小,及时提交事务

4.尽量避免跨数据库的分布式事务,因为环境的复杂性,很容易导致阻塞

5.慎用位图索引,更新时容易导致死锁

自动增加表分区:

该程序可以做为一个Oracle的JOB执行在每月的28日前执行(考虑2月28天的原因),自动为该用户下的分区表增加分区.

createorreplaceprocedureguan_add_partition

/*

/*为一个用户下所有分区表自动增加分区.分区的列为date类型,分区名类似:

p200706.

/*createbyDavid

*/

as

v_table_namevarchar2(50);

v_partition_namevarchar2(50);

v_monthchar(6);

v_add_month_1char(6);

v_sql_stringvarchar2(2000);

v_add_monthvarchar2(20);

cursorcur_partisselectdistinctu.table_name,max(p.partition_name)max_part_namefromuser_tablesu,user_tab_partitionsp

whereu.table_name=p.table_nameandu.partitioned='YES'

groupbyu.table_name;

Begin

selectto_char(sysdate,'yyyymm')intov_monthfromdual;

selectto_char(add_months(sysdate,1),'yyyymm')intov_add_month_1fromdual;

selectto_char(add_months(trunc(sysdate,'mm'),2),'yyyy-mm-dd')intov_add_monthfromdual;

opencur_part;

loop

fetchcur_partintov_table_name,v_partition_name;

exitwhencur_part%notfound;

ifto_number(substr(v_partition_name,2))<=to_number(substr(v_month,1))then

v_sql_string:

='altertable'||v_table_name||'addpartitionp'||v_add_month_1||

'VALUESLESSTHAN(to_date('''||v_add_month||''',''yyyy-mm-dd''))tablespaceusers';

executeimmediatev_sql_string;

else

null;

endif;

endloop;

closecur_part;

end;

展开阅读全文
相关资源
猜你喜欢
相关搜索

当前位置:首页 > PPT模板 > 商务科技

copyright@ 2008-2022 冰豆网网站版权所有

经营许可证编号:鄂ICP备2022015515号-1