Oracle实例之HWM(高水位线)性能优化

最近,bi同事的反馈指出了一张表的数据查询速度非常慢,尽管该表的数据总量不足1万行。我们首先考虑的是高水位线(hwm)带来的性能问题,即高水位线下占用了大量数据块,而这些数据块中大部分是空闲的。

我们知道,在全表扫描时,高水位线下的所有数据块都会被扫描,因此扫描的数据块数量可能远远超过实际存储数据的数据块数量。

一、收集表的统计信息

要获取准确的高水位信息,首先需要收集统计信息,这样得到的信息才会相对准确。

ANALYZE TABLE table_name ESTIMATE STATISTICS;ANALYZE TABLE table_name COMPUTE STATISTICS FOR TABLE FOR ALL INDEXES FOR ALL INDEXED COLUMNS;execute dbms_stats.gather_table_stats(ownname => 'OWNER', tabname => 'TABLE_NAME', estimate_percent => null, method_opt => 'for all indexed columns', cascade => true);

二、查看表信息

查看表的块和行信息:

select t.TABLE_NAME, t.NUM_ROWS, t.BLOCKS, t.empty_blocks, t.LAST_ANALYZED from dba_tables t where table_name in ('TABLE_NAME');SELECT COUNT(DISTINCT DBMS_ROWID.ROWID_BLOCK_NUMBER(ROWID)) USED_BLOCK FROM TABLE_NAME;

Oracle实例之HWM(高水位线)性能优化

上述查询结果显示,当前表的行数为9651行,HWM下的数据块使用了716119个,而未使用的数据块为0个。

实际数据占用的数据块数量为152个。

从中可以看出,高水位线下有716119-152个数据块可以释放,这样每次全表扫描只需扫描152个数据块即可。

通过查看段大小来验证记录数和表大小是否一致,段大小为5.5G,9651行的记录几乎不可能达到这个大小,因此可以断定其中有很多空闲块。

select segment_name, bytes/1024/1024/1024 TSize_GB from dba_segments where segment_name='table_name' ---5876219904

Oracle实例之HWM(高水位线)性能优化

三、问题原因

ViiTor实时翻译 ViiTor实时翻译

AI实时多语言翻译专家!强大的语音识别、AR翻译功能。

ViiTor实时翻译 116 查看详情 ViiTor实时翻译

什么情况下会导致上述问题,即高水位线下存在大量未使用的数据块?通常是大表(插入大量记录后)经过批量删除操作(delete),未释放高水位线所致。

全表扫描需要读取高水位线下的所有数据块,无论是否包含数据。如果在插入数据时使用了append关键字,即使高水位线下有空闲的数据块,也会从高水位线上方的数据块进行分配,导致高水位线上升。

四、降低高水位的方法

alter table table_name move; 此方法可以释放高水位,但需要重建索引。alter table table_name shrink space; 此方法可以释放高水位,但在执行前需要启用行移动:alter table table_name enable row movement;使用导出/导入(exp/imp)方式重建表数据。使用删除/创建(drop/create)方式重建表。使用truncate表。alter table table_name deallocate unused; 此方法用于释放HWM上方的未使用空间,但不会释放HWM下方的自由空间,也不会移动HWM的位置。

五、高水位调整实施

统计信息收集(如上所述)。查看执行计划:

SQL> set autotrace trace;SQL> set timing on;SQL> SELECT count(*) FROM TABLE_NAME;

Oracle实例之HWM(高水位线)性能优化

表移动:

alter table table_name move;

报错:ORA-00054: resource busy and acquire with NOWAIT specified or timeout expired查看被锁对象:

select object_name, machine, s.sid, s.serial# from v$locked_object l, dba_objects o, v$session s where l.object_id=o.object_id and l.session_id=s.sid;

执行后再查看执行计划统计信息:

Oracle实例之HWM(高水位线)性能优化

可以看到,统计信息中访问的数据块已经减少,全表扫描速度也大大提升。

索引重建:

alter index index_name rebuild online;

六、库高水位对象统计

①比较表的行数和表的大小关系。如果行数为0,而表的当前占用大小减去初始化时的大小(INITIAL_EXTENT)后依然很大,说明该表有高水位。

②行数和块数的比率,即查看一个块可以存储多少行数据。如果一个块存储的行数少于5行甚至更少,说明有高水位。注意,这两种方法都不是十分准确,需要对查询结果进行进一步筛选。在查询表的高水位时,首先需要分析表,以获取最准确的统计信息。

SELECT D.OWNER, ROUND(D.NUM_ROWS / D.BLOCKS, 2), D.NUM_ROWS, D.BLOCKS, D.TABLE_NAME, ROUND((d.BLOCKS*8-D.INITIAL_EXTENT/1024)/1024) t_size FROM DBA_TABLES D WHERE D.BLOCKS > 10 AND ROUND(D.NUM_ROWS / D.BLOCKS, 2)  50 AND OWNER NOT LIKE '%SYS%' AND BLOCKS > 100 ORDER BY WASTE_PER DESC;

以上就是Oracle实例之HWM(高水位线)性能优化的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/457434.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年11月8日 02:03:51
下一篇 2025年11月8日 02:04:28

相关推荐

  • C语言如何检查某常量是否存在

    本文将深入探讨c语言如何检查某常量是否存在,相信这对许多程序员来说非常实用,因此分享给大家,希望大家能从中受益。 在C语言中检查常量是否存在 检查预处理常量是否存在 检查预处理常量是否存在的简便方法是使用#ifdef和#ifndef预处理指令。 ifdef用于检查常量是否已定义。如果常量已定义,则在…

    2025年12月18日
    000
  • C语言如何关闭由 zip_open() 函数打开的 zip 档案文件

    本文将详细介绍如何在C语言中关闭由zip_open()函数打开的ZIP文件,希望通过这篇文章,大家能掌握这一实用的编程技巧。 如何关闭ZIP文件: 要关闭由zip_open()函数打开的ZIP文件,可以使用zip_close()函数。该函数接受ZIP文件结构指针作为参数,并执行以下操作: 关闭ZIP…

    2025年12月18日
    000
  • C语言如何返回 zip 档案项目的压缩文件尺寸

    本文将为您详细介绍如何使用c语言来获取zip档案中项目的压缩文件尺寸。这是一项非常实用的技能,希望您在阅读本文后能有所收获。 在C语言中获取ZIP档案项目压缩文件大小 利用C语言中的zip.h库函数,可以轻松获取ZIP档案中项目的压缩文件大小。以下是具体操作步骤: 引入必要的库头文件 #includ…

    2025年12月18日
    000
  • 如何在 eclipse 中配置 c++ 开发

    在ec++lipse中配置c++开发环境需要以下步骤:1. 安装eclipse cdt插件,2. 配置c++编译器,3. 创建并运行c++项目,4. 使用调试工具,5. 优化代码性能。通过这些步骤,你可以在eclipse中高效地进行c++开发。 引言 在当今多语言编程的世界中,C++依然是性能要求高…

    2025年12月18日
    000
  • xcode 怎么创建 c++ 项目

    在 xc++ode 中创建 c++ 项目可以通过以下步骤实现:1. 打开 xcode,点击 “create a new xcode project”。2. 选择 “macos” 平台和 “command line tool” 模…

    2025年12月18日
    000
  • c++ 引用和指针的区别是什么

    引用和指针的主要区别在于:引用是变量的别名,必须初始化且不可更改;指针存储内存地址,可重新赋值。引用在函数参数和返回值中常用,语法简洁且安全;指针用于动态内存分配和复杂数据结构,灵活但易出错。 引言 在 C++ 编程中,引用和指针是两个经常被混淆的概念。今天我们就来深入探讨一下它们之间的区别。通过这…

    2025年12月18日
    000
  • c++ 递归函数怎么实现

    c++++ 中递归函数通过函数调用自身来解决问题。1) 定义递归函数需要基本情况和递归情况。2) 递归函数的工作原理是将问题分解成子问题,直到达到基本情况。3) 使用示例包括计算 fibonacci 数列,优化方法有记忆化递归。4) 常见错误包括栈溢出和无限递归,调试时使用调试器跟踪调用堆栈。5) …

    2025年12月18日
    000
  • dev c++ 如何设置编译选项

    在 dev-c++++ 中设置编译选项的方法是:1. 点击“工具”菜单,选择“编译器选项”。2. 在“编译器”选项卡中的“附加的编译器选项”字段输入选项。编译选项通过 gcc 实现,可以优化代码、生成调试信息等,需权衡选项的优缺点。 引言 在编程世界中,编译选项就像是调味料,能让你的代码变得更加美味…

    2025年12月18日
    000
  • dev c++ 怎么添加外部库

    在 dev-c++++ 中添加外部库的步骤如下:1. 下载库文件:从官方网站下载适合系统的库文件,如 libcurl。2. 添加头文件:在代码中包含头文件并将头文件目录添加到编译器搜索路径中。3. 添加库文件:在代码中链接库文件并将库文件目录添加到链接器搜索路径中。4. 编写和编译代码:使用库编写代…

    2025年12月18日
    000
  • c++ 作用域解析运算符怎么用

    在 c++++ 中,作用域解析运算符 (::) 用于明确指定标识符的作用域。1) 解决命名冲突,如访问全局变量 (::count)。2) 访问命名空间成员 (math::calculatearea)。3) 访问类的静态成员 (myclass::staticvar)。合理使用该运算符可以提高代码的清晰…

    2025年12月18日
    000
  • clion 怎么创建 c++ 项目

    在 c++lion 中创建 c++ 项目可以通过以下步骤实现:1. 启动 clion 并点击 “create new project”。2. 选择 “c++ executable” 作为项目类型。3. 设置项目位置,建议使用专门文件夹。4. 选择 c+…

    2025年12月18日
    000
  • c++ 结构体怎么定义和使用

    c++++ 结构体通过 struct 关键字定义,用于组合不同类型的数据。1) 定义结构体,如 struct person { std::string name; int age; double height;}; 2) 创建实例并访问成员,如 person.name = “alice&…

    2025年12月18日
    000
  • 如何在 sublime text 中运行 c++ 代码

    在 #%#$#%@%@%$#%$#%#%#$%@_348c++880664f2e1458b899ced2a3518e6 text 中运行 c++ 代码需要配置构建系统。1. 安装 c++ 编译器(如 mingw、xcode 或 gcc)。2. 创建并保存 c++.sublime-build 文件,定…

    2025年12月18日
    000
  • c++ 联合体怎么使用

    联合体在 c++++ 中允许在同一内存位置存储不同类型的数据。其优点包括节省内存,缺点是可能导致不可预测的值。使用时应注意初始化和类型安全。联合体允许在同一内存位置存储不同类型的数据,如整数、浮点数或字符数组。其主要优点是节省内存,因为所有成员共享同一块内存。缺点是当给一个成员赋值时,其他成员的值会…

    2025年12月18日
    000
  • c++ 浮点数精度问题怎么解决

    解决c++++中浮点数精度问题的方法包括:1.使用std::setprecision控制输出精度;2.使用std::fixed固定小数点位数;3.使用long double提高计算精度;4.使用整数运算避免浮点数问题;5.使用任意精度库如boost::multiprecision或gmp获得高精度。…

    2025年12月18日
    000
  • C编程中的字符输入问题

    让我们分析这段C代码中字符输入的问题,以及如何解决。 这段代码展示了一个常见的C语言输入陷阱:scanf(“%c”, &ch); 在读取整数后,无法正确读取下一个字符。 这是因为 scanf(“%d”, &num); 读取整数 13 后,输入缓冲区中仍然残留了一个换行符 n (用户按…

    2025年12月18日
    000
  • Gulc:从头开始建造的C库

    本文开启了一个系列,介绍我正在开发的C99库:Gulc(Generic Utility Library的缩写)。该库主要用于学习和娱乐目的,旨在提供C标准库中安全内存管理和实用功能(如向量和无序映射),以简化C语言编程。未来将持续添加更多功能。 目前,该库包含一个简单的验证系统(类似于断言,但在发行…

    2025年12月18日
    000
  • 指针,一个怪异的野兽,适合初学者及以后

    对于C/C++编程初学者来说,指针是最难理解的概念之一。许多学生为此苦恼,许多开发者都尽量避免使用指针。然而,理解指针至关重要,因此,让我们从基础开始。 什么是指针? 指针是一种数据类型,类似于int、float等。声明指针的语法是在类型名后添加*。例如,指向整数的指针声明为int *,指向自定义结…

    2025年12月18日
    000
  • C语言API与高尔夫服务器交谈

    本文演示如何使用C API与Golf Application Server进行交互。由于大多数编程语言都支持C链接,因此可以轻松地从各种编程环境访问Golf服务器。客户端API简洁易用,仅包含少量函数和一种数据类型,并支持多线程环境。本例中,Golf服务器使用树形对象存储键值对,服务器运行期间可进行…

    2025年12月18日
    000
  • Dev-C++ 版的问题

    dev-c++++ 4.9.9.2 编译错误及解决方案 在 Windows 11 系统使用 Dev-C++ 4.9.9.2 编译程序时,编译器记录窗格可能会显示以下错误信息: gcc.exe: internal error: aborted (program collect2)please subm…

    2025年12月18日
    000

发表回复

登录后才能评论
关注微信