oracle索引的测试实例代码

 更新时间:2021年01月17日 09:07:02   作者:zhangyingchengqi  
这篇文章主要给大家介绍了关于oracle索引测试的相关资料,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

前言

在测试oracle索引性能时大意了,没有仔细分析数据特点,将情况特此记录下来。 

需求:  对一张100w记录的表的 stuname列进行查询,测试在建立索引与不建立索引的区别. 以下是开始用的创建代码及执行效果. 

1. 随机数据生成代码分析

--为测试索引而准备的随机数据生成代码,先分析一下
select rownum as id,
     'smith'||trunc(dbms_random.value(0, 100)) as stu_name,
     dbms_random.string('x', 20) stu_pwd,
     to_char(add_months(sysdate,-DBMS_RANDOM.VALUE(100,200)) + rownum / 24 / 3600, 'yyyy-mm-dd hh24:mi:ss') as birthday ,
     decode( TRUNC(DBMS_RANDOM.VALUE(1,5)),1,'湖南省',2,'湖北省',3,'江西省','北京市') as address
   from dual
   connect by level <= 100; 

--先分析以下上面的代码

-- 伪列:  rownum

--  dual   : 测试表

--   || 字符串联接

--1. 测试生成100条记录     connect by level<=100 : 

--a、利用Oracle特有的“connect by”树形连接语法生成测试记录,“level <= 100”表示要生成100记录;

--b、利用rownum虚拟列生成递增的整数数据;

--c、利用sysdate函数加一些简单运算来生成日期数据,本例中是每条记录的时间加1秒;

--       add_months(sysdate,-DBMS_RANDOM.VALUE(100,200))   用当前时间 减去   至少100个月,最多200个月,来生成生日

--d、利用dbms_random.value函数生成随机的数值型数据,都是double型,所以都加了 trunc(   )以截断小数位,本例中是生成0到100之间的随机整数;

--e、利用dbms_random.string函数生成随机的字符型数据,本例中是生成长度为20的随机字符串,字符串中可以包括字符或数字。

2. 生成测试表及数据

--2. 正式生成100W
drop table stu_test_100w; --如果原来有,则先删除原来的表
 
--创建表及数据
create table stu_test_100w
as
select rownum as id,
     'smith'||trunc(dbms_random.value(0, 99)) as stu_name,
     dbms_random.string('x', 20) stu_pwd,
     to_char(add_months(sysdate,-DBMS_RANDOM.VALUE(100,200)) + rownum / 24 / 3600, 'yyyy-mm-dd hh24:mi:ss') as birthday ,
     decode( TRUNC(DBMS_RANDOM.VALUE(1,5)),1,'湖南省',2,'湖北省',3,'江西省','北京市') as address
   from dual
   connect by level <= 1000000; -- 生成 100w测试数据
-- 查看当前用户模式下所有的表 
select * from tab where tname='STU_TEST_100W';
--先执行一次查询, 注意查询所用的时间,此时并没有加入索引 
select * from stu_test_100w where stu_name='smith13'; 

执行结果: 

以上是没有用到索引时的执行用时  6.781秒. 

下面创建索引后,再用同一查询来测试. 

--********生成索引后,再执行一次查询
drop index index_student_test;
 
create index index_student_test
on stu_test_100w(stu_name);  --索引是针对某个表的某个列
 
--先执行一次查询, 注意查询的时间,此时加入了索引 
select * from stu_test_100w where stu_name='smith13';  

为什么用了索引后时间查询能还下降了呢????

分析如下: 

  1. 索引生成的字段的值分存得太密集了,查看上面的代码会发现我们stu_name只生成在了  smith0-99之间,即只有100种可能性, 对于100w数据则言,即每个名字都有约1w个. 

  2。 因为数据太密集了,所以以上查询的花的时间主要在1w条数据的显示上, 所以我们可以观察到不管是否用到了索引,都要共到6-7秒来显示结果. 

  3.  那为什么用了索引还慢一些呢?  这就与索引的存储结构有关系了.oracle默认使用的是B树索引, 当使用索引列查询时,查询必须先查看索引,通过索引去定位数据,而咱们的数据分布又比较密集,所以使用索引所导致的时间损耗要大于直接磁盘搜索的时间.  

那么如何解决呢?

随机生成的姓名分布广一些(这与真实的数据也一样).  即将随机生成代码修改为     'smith'||trunc(dbms_random.value(0, 9999999)) as stu_name,

drop table stu_test_100w; --如果原来有,则先删除原来的表
--重新生成表及随机数据,注意 stu_name列的取值范围加大
create table stu_test_100w
as
select rownum as id,
     'smith'||trunc(dbms_random.value(0, 9999999)) as stu_name,
     dbms_random.string('x', 20) stu_pwd,
     to_char(add_months(sysdate,-DBMS_RANDOM.VALUE(100,200)) + rownum / 24 / 3600, 'yyyy-mm-dd hh24:mi:ss') as birthday ,
     decode( TRUNC(DBMS_RANDOM.VALUE(1,5)),1,'湖南省',2,'湖北省',3,'江西省','北京市') as address
   from dual
   connect by level <= 1000000; 
 
--先执行一次查询, 注意查询的时间,此时并没有加入索引 
select * from stu_test_100w where stu_name='smith8821228'; 

执行结果如下:

用时 0.312秒. 

接着创建索引后,再测试同一个查询

--********生成索引后,再执行一次查询
drop index index_student_test;
 
create index index_student_test
on stu_test_100w(stu_name);  --索引是针对某个表的某个列
--先执行一次查询, 注意查询的时间,此时加入了索引 
select * from stu_test_100w where stu_name='smith8821228';  

使用索引后,同一个查询只需0.015秒,在原来用时0.312的基础下,下降了n倍.  

总结

到此这篇关于oracle索引测试的文章就介绍到这了,更多相关oracle索引测试内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • Oracle数据库分析函数用法

    Oracle数据库分析函数用法

    大家好,本篇文章主要讲的是Oracle数据库分析函数用法,感兴趣的同学赶快来看一看吧,对你有帮助的话记得收藏一下,方便下次浏览
    2021-12-12
  • Oracle修改默认的时间格式的四种方式

    Oracle修改默认的时间格式的四种方式

    这篇文章主要介绍了Oracle修改默认的时间格式的四种方式,默认的日期和时间格式由参数NLS_DATE_FORMAT控制,如果需要修改默认的时间格式,可以通过修改会话级别或系统级别的参数来实现,需要的朋友可以参考下
    2024-06-06
  • oracle求同比,环比函数(LAG与LEAD)的详解

    oracle求同比,环比函数(LAG与LEAD)的详解

    本篇文章是对oracle求同比,环比函数(LAG与LEAD)进行了详细的分析介绍,需要的朋友参考下
    2013-05-05
  • Oracle删除表前判断表名是否存在若存在则删除

    Oracle删除表前判断表名是否存在若存在则删除

    在Oracle中若删除一个不存在的表则会提示表或视图不存在,这就需要我们再删除表前判断该表是否存在,若存在则删除,感兴趣的朋友可以了解下本文
    2013-11-11
  • Oracle用户密码过期报错的解决办法

    Oracle用户密码过期报错的解决办法

    Oracle数据库 11g默认密码过期时间为180天过期,针对密码过期企业一般是采用修改密码的方式,个人电脑上则可以将密码过期时间修改为永久,本文给大家介绍了Oracle用户密码过期报错的解决办法,需要的朋友可以参考下
    2024-03-03
  • Oracle中查询重复记录的几种方法实现

    Oracle中查询重复记录的几种方法实现

    这篇文章主要介绍了Oracle中查询重复记录的方法实现,包含使用GROUP BY和HAVING语句,使用窗口函数ROW_NUMBER()和使用自连接查询这三种方式,具有一定的参考价值,感兴趣的可以了解一下
    2024-06-06
  • 升级和卸载Oracle数据库软件的命令整理

    升级和卸载Oracle数据库软件的命令整理

    这篇文章主要介绍了升级和卸载Oracle数据库软件的命令整理,包括升级时可能用到的查看版本号和备份操作的命令介绍,需要的朋友可以参考下
    2015-12-12
  • 使用PLSQL查看表属性乱码的解决方法

    使用PLSQL查看表属性乱码的解决方法

    PL/SQL是Oracle数据库的编程语言,用于编写存储过程、触发器、函数等,今天用plsql想查看表的属性,看看各个字段的注释,可是打开一看,居然是乱码的,所以接下来本文给大家介绍了使用PLSQL查看表属性乱码的解决方法,需要的朋友可以参考下
    2024-03-03
  • oracle+mybatis 使用动态Sql当插入字段不确定的情况下实现批量insert

    oracle+mybatis 使用动态Sql当插入字段不确定的情况下实现批量insert

    最近接了一个项目,其中项目需求,有一个非常纠结的问题,由于业务的关系,DB的数据表无法确定,在使用过程中字段可能会增加,这样在insert时给我造成了很大的困扰。接下来,通过本篇文章给大家介绍oracle+mybatis 使用动态Sql当插入字段不确定的情况下实现批量insert
    2015-11-11
  • 在Oracle的函数中,返回表类型的语句

    在Oracle的函数中,返回表类型的语句

    在SQL Server中有表变量,可以在function中方便地返回,习惯SQL Server或者需要把脚本从SQL Server转到Oracle中的朋友可以都会碰到这个问题
    2012-10-10

最新评论