oracle索引空字段,一文搞懂Oracle中索引和空值的恩恩怨怨

2024-02-10 06:30

本文主要是介绍oracle索引空字段,一文搞懂Oracle中索引和空值的恩恩怨怨,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

在日常的工作中,空值总是有特殊的身份,它和索引之间也是“恩怨”颇多。

(1)有时候创建索引会因为空值出现一些奇怪的结果。

(2)有时候一个简单的查询因为空值却走不了索引。

我们来简单地模拟一下这些问题。

首先创建一个空表,注意,对于ID列我们是加了NOT NULL约束的。

SQL> create table index_test(id number not null,name varchar2(30) ) ;

Table created.

创建一个唯一性索引,包含了ID和name列。

SQL> create unique index inx_test on index_test(id,name);

Index created.

这个时候我们对表index_test插入数据。因为name列没有非空约束,所以可以为空。注意第一条insert语句,如果插入空串也会作为NULL来处理。

SQL> insert into index_test values(2,'');

1 row created.

SQL> insert into index_test values(1,'a');

1 row created.

SQL> insert into index_test values(3,null);

1 row created.

收集一下统计信息。

exec dbms_stats.gather_table_stats(user,'INDEX_TEST');

查看查询的执行计划,如下。

SQL> select *from index_test where id is not null;

Execution Plan

----------------------------------------------------------

Plan hash value: 4273605835

-----------------------------------------------------------------------------

| Id | Operation | Name | Rows | Bytes | Cost (%CPU)| Time |

-----------------------------------------------------------------------------

| 0 | SELECT STATEMENT | | 3 | 12 | 1 (0)| 00:00:01 |

| 1 | INDEX FULL SCAN | INX_TEST | 3 | 12 | 1 (0)| 00:00:01 |

-----------------------------------------------------------------------------

对于这个查询可能没什么感觉,走了全索引扫描。我们在查询条件中添加了id is not null的条件,其实ID列已经存在非空约束了,所以这个过滤条件可有可无。

我们来看看不加过滤条件的情况,还是走了全索引扫描,如下。

SQL> select *from index_test;

Execution Plan

----------------------------------------------------------

Plan hash value: 4273605835

-----------------------------------------------------------------------------

| Id | Operation | Name | Rows | Bytes | Cost (%CPU)| Time |

-----------------------------------------------------------------------------

| 0 | SELECT STATEMENT | | 3 | 12 | 1 (0)| 00:00:01 |

| 1 | INDEX FULL SCAN | INX_TEST | 3 | 12 | 1 (0)| 00:00:01 |

-----------------------------------------------------------------------------

取消ID列的非空约束。

SQL> alter table index_test modify(id number null);

Table altered.

再次查询执行计划,发现走了全表扫描。

SQL> select *from index_test;

Execution Plan

----------------------------------------------------------

Plan hash value: 356488860

-----------------------------------------------------------------------------| Id | Operation | Name | Rows | Bytes | Cost (%CPU)| Time |

-----------------------------------------------------------------------------| 0 | SELECT STATEMENT | | 3 | 12 | 3 (0)| 00:00:01 |

| 1 | TABLE ACCESS FULL| INDEX_TEST | 3 | 12 | 3 (0)| 00:00:01 |

-----------------------------------------------------------------------------

添加非空的过滤条件,又可以走索引了。

SQL> select *from index_test where id is not null;

Execution Plan

----------------------------------------------------------

Plan hash value: 4273605835

-----------------------------------------------------------------------------

| Id | Operation | Name | Rows | Bytes | Cost (%CPU)| Time |

-----------------------------------------------------------------------------

| 0 | SELECT STATEMENT | | 3 | 12 | 1 (0)| 00:00:01 |

|* 1 | INDEX FULL SCAN | INX_TEST | 3 | 12 | 1 (0)| 00:00:01 |

-----------------------------------------------------------------------------

Predicate Information (identified by operation id):

---------------------------------------------------

1 - filter("ID" IS NOT NULL)

那么问题来了,索引对于空值好像总是有些特殊,下面来看看空值在索引中的一些细节。

SQL> set autot off

这个时候表index_test的索引还是唯一性索引,尝试插入一些值来对比一下。

如果插入name列为NULL,可以正常插入。

SQL> insert into index_test values(1,null);

1 row created.

如果插入ID列为NULL,也可以正常插入。因为我们取消了ID列的非空约束。

SQL> insert into index_test values(null,1);

1 row created.

插入ID、name列为NULL,这个时候竟然可以正常插入。

SQL> insert into index_test values(null,null);

1 row created.

再尝试一条,竟然还可以正常插入。

SQL> insert into index_test values(null,null);

1 row created.

我们再来试试空串的情况,发现结果和NULL是一致的,都可以插入。

SQL> insert into index_test values('','');

1 row created.

SQL> insert into index_test values('','');

1 row created.

简单分析一下索引,来看看空值在索引中的存储情况。

SQL> analyze index inx_test validate structure;

Index analyzed.

首先来看index_test中的数据情况,因为有些行存在空值,把rownum打印出来方便查看。

SQL> select rownum,id,name from index_test;

ROWNUM ID NAME

---------- ---------- ------------------------------

1 2

2 1 a

3 3

4 1

5 1

6

7

8

9

9 rows selected.

一共有9行数据,索引只有5行,最后4行都不在索引中。这也就基本能够说明为什么上面的查询条件中id is NOT NULL有时候走索引,有时候又不走索引。

至于为什么可以成功插入ID、name列为空的行,是因为对于Oracle来说,(NULL,NULL)和(NULL,NULL)是不同的,NULL值总是介于一种很模糊的状态。

SQL> select name,lf_rows from index_stats;

NAME LF_ROWS

------------------------------ ----------

INX_TEST 5

对表index_test中的数据进行统计,NULL值的统计结果是包含了NULL和空串。

SQL> select id,name,count(*)from index_test group by id,name ;

ID NAME COUNT(*)

---------- ------------------------------ ----------

1 1

4

1 a 1

3 1

2 1

1 1

6 rows selected.

此外,在平时的工作中,如果需要对某个表创建索引,就需要考虑NULL值的情况,为了使得索引能够正常启用,我们需要索引列中至少有一列存在非空约束。

就如下面的情况,我们已经存在唯一性索引,但是因为b树索引不会存储NULL的条目,所以对表中已有的空值就需要使用全表扫描。

SQL> set autot trace exp

SQL> select id,name from index_test;

Execution Plan

----------------------------------------------------------

Plan hash value: 356488860

-----------------------------------------------------------------------------

| Id | Operation | Name | Rows | Bytes | Cost (%CPU)| Time |

-----------------------------------------------------------------------------

| 0 | SELECT STATEMENT | | 3 | 12 | 3 (0)| 00:00:01 |

| 1 | TABLE ACCESS FULL| INDEX_TEST | 3 | 12 | 3 (0)| 00:00:01 |

-----------------------------------------------------------------------------

如果需要输出非空的数据,加入is NOT NULL的过滤条件,索引就能够正常启用。

select id,name from index_test where id is not null

Execution Plan

----------------------------------------------------------

Plan hash value: 4273605835

-----------------------------------------------------------------------------

| Id | Operation | Name | Rows | Bytes | Cost (%CPU)| Time |

-----------------------------------------------------------------------------

| 0 | SELECT STATEMENT | | 3 | 12 | 1 (0)| 00:00:01 |

|* 1 | INDEX FULL SCAN | INX_TEST | 3 | 12 | 1 (0)| 00:00:01 |

-----------------------------------------------------------------------------

Predicate Information (identified by operation id):

---------------------------------------------------

1 - filter("ID" IS NOT NULL)

193230fa0641c2bf05b6c2f7ce138177.png

小结: Oracle中的空值会影响索引扫描,会出现一些潜在的性能问题,在表设计中,字段如果可以设置为非空,就绝对不要使用null值。

举报/反馈

这篇关于oracle索引空字段,一文搞懂Oracle中索引和空值的恩恩怨怨的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/696367

相关文章

pandas中位数填充空值的实现示例

《pandas中位数填充空值的实现示例》中位数填充是一种简单而有效的方法,用于填充数据集中缺失的值,本文就来介绍一下pandas中位数填充空值的实现,具有一定的参考价值,感兴趣的可以了解一下... 目录什么是中位数填充?为什么选择中位数填充?示例数据结果分析完整代码总结在数据分析和机器学习过程中,处理缺失数

Pandas统计每行数据中的空值的方法示例

《Pandas统计每行数据中的空值的方法示例》处理缺失数据(NaN值)是一个非常常见的问题,本文主要介绍了Pandas统计每行数据中的空值的方法示例,具有一定的参考价值,感兴趣的可以了解一下... 目录什么是空值?为什么要统计空值?准备工作创建示例数据统计每行空值数量进一步分析www.chinasem.cn处

一文详解Java异常处理你都了解哪些知识

《一文详解Java异常处理你都了解哪些知识》:本文主要介绍Java异常处理的相关资料,包括异常的分类、捕获和处理异常的语法、常见的异常类型以及自定义异常的实现,文中通过代码介绍的非常详细,需要的朋... 目录前言一、什么是异常二、异常的分类2.1 受检异常2.2 非受检异常三、异常处理的语法3.1 try-

C# foreach 循环中获取索引的实现方式

《C#foreach循环中获取索引的实现方式》:本文主要介绍C#foreach循环中获取索引的实现方式,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录一、手动维护索引变量二、LINQ Select + 元组解构三、扩展方法封装索引四、使用 for 循环替代

一文带你搞懂Python中__init__.py到底是什么

《一文带你搞懂Python中__init__.py到底是什么》朋友们,今天我们来聊聊Python里一个低调却至关重要的文件——__init__.py,有些人可能听说过它是“包的标志”,也有人觉得它“没... 目录先搞懂 python 模块(module)Python 包(package)是啥?那么 __in

MySQL索引的优化之LIKE模糊查询功能实现

《MySQL索引的优化之LIKE模糊查询功能实现》:本文主要介绍MySQL索引的优化之LIKE模糊查询功能实现,本文通过示例代码给大家介绍的非常详细,感兴趣的朋友一起看看吧... 目录一、前缀匹配优化二、后缀匹配优化三、中间匹配优化四、覆盖索引优化五、减少查询范围六、避免通配符开头七、使用外部搜索引擎八、分

一文详解如何在Python中从字符串中提取部分内容

《一文详解如何在Python中从字符串中提取部分内容》:本文主要介绍如何在Python中从字符串中提取部分内容的相关资料,包括使用正则表达式、Pyparsing库、AST(抽象语法树)、字符串操作... 目录前言解决方案方法一:使用正则表达式方法二:使用 Pyparsing方法三:使用 AST方法四:使用字

电脑死机无反应怎么强制重启? 一文读懂方法及注意事项

《电脑死机无反应怎么强制重启?一文读懂方法及注意事项》在日常使用电脑的过程中,我们难免会遇到电脑无法正常启动的情况,本文将详细介绍几种常见的电脑强制开机方法,并探讨在强制开机后应注意的事项,以及如何... 在日常生活和工作中,我们经常会遇到电脑突然无反应的情况,这时候强制重启就成了解决问题的“救命稻草”。那

一文详解JavaScript中的fetch方法

《一文详解JavaScript中的fetch方法》fetch函数是一个用于在JavaScript中执行HTTP请求的现代API,它提供了一种更简洁、更强大的方式来处理网络请求,:本文主要介绍Jav... 目录前言什么是 fetch 方法基本语法简单的 GET 请求示例代码解释发送 POST 请求示例代码解释

Oracle数据库常见字段类型大全以及超详细解析

《Oracle数据库常见字段类型大全以及超详细解析》在Oracle数据库中查询特定表的字段个数通常需要使用SQL语句来完成,:本文主要介绍Oracle数据库常见字段类型大全以及超详细解析,文中通过... 目录前言一、字符类型(Character)1、CHAR:定长字符数据类型2、VARCHAR2:变长字符数