oracle查所有表的索引个数

1. 查看当前用户所有表的索引数量

sql 复制代码
SELECT 
    t.table_name,
    COUNT(i.index_name) as index_count,
    LISTAGG(i.index_name, ', ') WITHIN GROUP (ORDER BY i.index_name) as index_names
FROM user_tables t
LEFT JOIN user_indexes i ON t.table_name = i.table_name
GROUP BY t.table_name
ORDER BY COUNT(i.index_name) DESC, t.table_name;

2. 查看所有用户所有表的索引数量(需要DBA权限)

sql 复制代码
SELECT 
    i.table_owner,
    i.table_name,
    COUNT(i.index_name) as index_count,
    LISTAGG(i.index_name, ', ') WITHIN GROUP (ORDER BY i.index_name) as index_names
FROM dba_indexes i
WHERE i.table_owner NOT IN ('SYS', 'SYSTEM', 'XDB', 'CTXSYS', 'MDSYS', 'ORDSYS')  -- 排除系统用户
GROUP BY i.table_owner, i.table_name
ORDER BY i.table_owner, COUNT(i.index_name) DESC, i.table_name;

3.查询表及其索引的详细信息--推荐使用,oracle国产化转换到tidb,最好明确知道所有需要迁移的生产表的条数等令牌

sql 复制代码
SELECT 
    t.owner,
    t.table_name,
    t.num_rows as table_rows,
    COUNT(i.index_name) as total_indexes,
    SUM(CASE WHEN i.uniqueness = 'UNIQUE' THEN 1 ELSE 0 END) as unique_indexes,
    SUM(CASE WHEN i.uniqueness = 'NONUNIQUE' THEN 1 ELSE 0 END) as nonunique_indexes,
    SUM(CASE WHEN i.index_type = 'FUNCTION-BASED NORMAL' THEN 1 ELSE 0 END) as function_based_indexes
FROM dba_tables t
LEFT JOIN dba_indexes i ON t.owner = i.table_owner AND t.table_name = i.table_name
WHERE t.owner = 'YOUR_SCHEMA_NAME'  -- 替换为你的模式名
GROUP BY t.owner, t.table_name, t.num_rows
ORDER BY COUNT(i.index_name) DESC, t.table_name;

4.按索引类型统计

sql 复制代码
SELECT 
    i.table_owner,
    i.table_name,
    i.index_type,
    COUNT(*) as count_per_type,
    LISTAGG(i.index_name, ', ') WITHIN GROUP (ORDER BY i.index_name) as index_list
FROM dba_indexes i
WHERE i.table_owner = 'YOUR_SCHEMA_NAME'  -- 替换为你的模式名
GROUP BY i.table_owner, i.table_name, i.index_type
ORDER BY i.table_name, i.index_type;

5.查询没有索引的表

sql 复制代码
-- 查找当前用户下没有索引的表
SELECT 
    t.table_name,
    t.num_rows,
    t.blocks
FROM user_tables t
WHERE NOT EXISTS (
    SELECT 1 
    FROM user_indexes i 
    WHERE i.table_name = t.table_name
)
AND t.table_name NOT LIKE 'BIN$%'  -- 排除回收站中的表
ORDER BY t.num_rows DESC NULLS LAST;

-- 查找所有用户下没有索引的表(需要DBA权限)
SELECT 
    t.owner,
    t.table_name,
    t.num_rows
FROM dba_tables t
WHERE NOT EXISTS (
    SELECT 1 
    FROM dba_indexes i 
    WHERE i.table_owner = t.owner 
    AND i.table_name = t.table_name
)
AND t.owner NOT IN ('SYS', 'SYSTEM', 'XDB')
AND t.table_name NOT LIKE 'BIN$%'
ORDER BY t.owner, t.num_rows DESC NULLS LAST;

6.索引列数统计

sql 复制代码
-- 统计每个索引的列数
SELECT 
    i.table_name,
    i.index_name,
    i.uniqueness,
    i.status,
    COUNT(ic.column_position) as column_count,
    LISTAGG(ic.column_name, ', ') WITHIN GROUP (ORDER BY ic.column_position) as columns
FROM user_indexes i
JOIN user_ind_columns ic ON i.index_name = ic.index_name
GROUP BY i.table_name, i.index_name, i.uniqueness, i.status
ORDER BY i.table_name, i.index_name;

7.实用的汇总查询

sql 复制代码
-- 索引统计汇总
WITH index_stats AS (
    SELECT 
        owner,
        table_name,
        COUNT(*) as total_indexes,
        ROUND(AVG(blevel), 2) as avg_blevel,
        ROUND(AVG(leaf_blocks), 2) as avg_leaf_blocks,
        SUM(CASE WHEN status != 'VALID' THEN 1 ELSE 0 END) as invalid_indexes
    FROM dba_indexes
    WHERE owner = 'YOUR_SCHEMA_NAME'
    GROUP BY owner, table_name
)
SELECT 
    owner,
    COUNT(DISTINCT table_name) as tables_with_indexes,
    SUM(total_indexes) as total_index_count,
    ROUND(AVG(total_indexes), 2) as avg_indexes_per_table,
    ROUND(MEDIAN(total_indexes), 2) as median_indexes_per_table,
    MAX(total_indexes) as max_indexes_in_table,
    SUM(invalid_indexes) as total_invalid_indexes
FROM index_stats
GROUP BY owner;

8.生产监控大表无索引情况

sql 复制代码
-- 查找行数超过10000但索引数少于2个的表
SELECT 
    t.owner,
    t.table_name,
    t.num_rows,
    COUNT(i.index_name) as index_count
FROM dba_tables t
LEFT JOIN dba_indexes i ON t.owner = i.table_owner AND t.table_name = i.table_name
WHERE t.num_rows > 10000
AND t.owner = 'YOUR_SCHEMA_NAME'
GROUP BY t.owner, t.table_name, t.num_rows
HAVING COUNT(i.index_name) < 2
ORDER BY t.num_rows DESC;

9.查看索引使用情况(需要Oracle 11g及以上)

sql 复制代码
SELECT 
    table_name,
    index_name,
    used
FROM v$object_usage
WHERE used = 'NO'  -- 查看未使用的索引
ORDER BY table_name;

10.生成创建索引的脚本

sql 复制代码
SELECT 
    'CREATE INDEX idx_' || table_name || '_' || column_name || 
    ' ON ' || table_name || '(' || column_name || ');' as create_index_sql
FROM (
    SELECT DISTINCT
        t.table_name,
        tc.column_name
    FROM user_tables t
    JOIN user_tab_columns tc ON t.table_name = tc.table_name
    WHERE NOT EXISTS (
        SELECT 1 
        FROM user_ind_columns ic 
        WHERE ic.table_name = t.table_name 
        AND ic.column_name = tc.column_name
    )
    AND t.table_name NOT LIKE 'BIN$%'
    AND tc.column_name NOT LIKE '%ID'  -- 排除ID列
    AND tc.data_type IN ('VARCHAR2', 'CHAR', 'NUMBER', 'DATE')  -- 只对某些数据类型创建索引
)
WHERE ROWNUM <= 10;  -- 限制生成的数量
相关推荐
happyboy198621116 小时前
2026大专财富管理可以转数据分析吗?
数据库·数据挖掘·数据分析
杰克尼16 小时前
苍穹外卖--day11
java·数据库·spring boot·mybatis·notepad++
Y0011123616 小时前
Day2-MySQL-SQL-1
sql·mysql·oracle
LaughingZhu16 小时前
Product Hunt 每日热榜 | 2026-03-12
大数据·数据库·人工智能·经验分享·搜索引擎
jnrjian16 小时前
ORA-06512 CTXSYS drop user cascade
oracle
白云如幻16 小时前
【JDBC】面向对象的思路编写JDBC程序
java·数据库
艾莉丝努力练剑17 小时前
【Linux进程间通信:共享内存】为什么共享内存的 key 值由用户设置
java·linux·运维·服务器·开发语言·数据库·mysql
悲伤小伞18 小时前
Git_原理及使用_撤销修改_删除文件
数据库·git
fengxin_rou18 小时前
一文读懂 Redis 集群:从哈希槽到透明访问
java·数据库·redis·算法·spring·缓存
m0_6356474818 小时前
Qt开发与MySQL数据库教程(二)——MySQL常用命令以及示例
java·开发语言·数据库·mysql