Oracle调整表空间大小——ORA-03297: 文件包含在请求的 RESIZE 值之外使用的数据

Oracle数据文件在有数据的状况下能自动扩展,却不能自动收缩,形成存储空间的浪费。
若是直接修改数据文件的大小,可能会遇到以下错误:ORA-03297: 文件包含在请求的 RESIZE 值之外使用的数据html

转载网址:http://blog.sina.com.cn/s/blog_54eeb5d901000bvg.htmlsql

SQL> ALTER DATABASE DATAFILE 'D:\ORACLE\ORADATA\ICAPP\IC_DATA6.ORA' RESIZE 300m;
ALTER DATABASE DATAFILE 'D:\ORACLE\ORADATA\ICAPP\IC_DATA6.ORA' RESIZE 300m
*
ERROR
位于第 1 行:
ORA-03297: 文件包含在请求的 RESIZE 值之外使用的数据
数据库

可是ui

SQL>select d.filename,d.file_id,d.bytes/1024/1024 as d_byte,sum(f.bytes/1024/1024) as free_bytespa

2   from dba_data_files d,dba_free_space fhtm

3   where d.file_id=f.file_id and d.file_id=18blog

4   group by d.file_name,d.file_id,d.bytes/1024/1024;索引

FILE_NAME                             FILE_ID    D_BYTE    FREE_BYTEget

---------------------------------    ---------- ---------- ----------it

D:\ORACLE\ORADATA\ICAPP\IC_DATA6.ORA     18       1536     1482.0625

能够看到实际上ID=18的文件只使用了大概50M左右,只是数据分布在(按必定的顺序)50M甚至在300M之外的地方,因此这里虽然看到只使用了约50M空间,可是却不能resize datafile.

为此,要改小数据文件,咱们先要对文件上的表和索引移动一下位置,具体作法以下:

 1移动表前先对表空间作整理

SQL>alter tablespace ic_data coalesce;

 2dba_extents找到与ID=18的数据文件相关的表及索引

SQL>select segment_name,partition_name,segment_type

2   from dba_extents

3   where file_id=18;

 3id=18的文件上的表和索引移动位置

SQL> set heading off
SQL> set echo off
SQL> set feedback off
SQL> set termout on
SQL> spool d:\aaa.sql

//移动表

SQL>select DISTINCT 'alter table '|| segment_name || ' move tablespace test_space;' from dba_extents where segment_type='TABLE' and file_id=18;

//移动索引

SQL>select DISTINCT 'alter index '|| segment_name || ' rebuild tablespace test_space;' from dba_extents where segment_type='INDEX' and file_id=18;

//移动分区表

SQL>select DISTINCT 'alter table '|| segment_name || ' move partition '|| partition_name || ' tablespace test_space;' from dba_extents where segment_type='TABLE PARTITION' and file_id=18;

//移动分区索引

SQL>select DISTINCT 'alter index '|| segment_name || ' rebuild partition '|| partition_name || ' tablespace test_space;' from dba_extents where segment_type='INDEX PARTITION' and file_id=18;

SQL>spool off 

而后执行aaa.sql,注意保证test_space有足够的空间容纳这些数据,

其实能够不移动全部的数据,可是总的测验是否是移动了300M之外的数据,因此仍是移动全部数据的方便

 4这样移动了全部的数据之后就能够对datafile resize

SQL> c/9/3
  1* ALTER DATABASE DATAFILE 'D:\ORACLE\ORADATA\ICAPP\IC_DATA6.ORA' RESIZE 300M
SQL> /

数据库已更改。

 5把原来表空间ic_data中的数据再移动回来,修改aaa.sql中的表空间名为ic_data再执行,而后drop tablespace test_space including contents and datafiles

 

 以上操做起源于我删除了database里的一个大表形成不少空间浪费,想回收空间:)

-----------------------分割线-----------------------------分割线-----------------------------分割线----------------------------

通过实践,以上方法能够处理ORA-03297: 文件包含在请求的 RESIZE 值之外使用的数据问题。

可是若是一个表空间被多个Oracle用户使用,在导出sql文件时要指定用户,不然在执行sql文件时会报错

相关文章
相关标签/搜索