Oracle調整表空間大小resize
阿新 • • 發佈:2019-01-05
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 值以外使用的資料 但是 SQL>select d.filename,d.file_id,d.bytes/1024/1024 as d_byte,sum(f.bytes/1024/1024) as free_byte 2 from dba_data_files d,dba_free_space f 3 where d.file_id=f.file_id and d.file_id=18 4 group by d.file_name,d.file_id,d.bytes/1024/1024; FILE_NAME FILE_ID D_BYTE FREE_BYTE
--------------------------------- ---------- ---------- ----------
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;
2、在dba_extents找到與ID=18的資料檔案相關的表及索引
SQL>select segment_name,partition_name,segment_type
2 from dba_extents
3 where file_id=18;
3、對id=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了
ALTER DATABASE DATAFILE 'D:\ORACLE\ORADATA\ICAPP\IC_DATA6.ORA' RESIZE 300m
*
ERROR 位於第 1 行:
ORA-03297: 檔案包含在請求的 RESIZE 值以外使用的資料 但是 SQL>select d.filename,d.file_id,d.bytes/1024/1024 as d_byte,sum(f.bytes/1024/1024) as free_byte 2 from dba_data_files d,dba_free_space f 3 where d.file_id=f.file_id and d.file_id=18 4 group by d.file_name,d.file_id,d.bytes/1024/1024; FILE_NAME FILE_ID D_BYTE FREE_BYTE
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、
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檔案時會報錯