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文件時會報錯

相關文章
相關標籤/搜索