back to the basics: ALTER TABLE MOVE vs SHRINK


01.02.2017
by Kamil Stawiarski

It’s time for the next article with ODBV visualisation 🙂

This time let’s examine the difference between move and shrink – this is very common question on a lot of trainings. You can find a lot of great articles in the Internet regarding this subject but I think that visualisation really helps to understand what is really going on at the low level.

Let’s start with MOVE.
At the beginning I’ll create a new tablespace:

1SQL> !cat odbv_tbs.sql
2create tablespace odbv_tbs
3datafile size 8m
4autoextend on next 8m
5maxsize 1g
6/
7 
8select file_name
9from dba_data_files
10where tablespace_name='ODBV_TBS'
11/
12 
13SQL> @odbv_tbs.sql
14 
15Tablespace created.
16 
17 
18FILE_NAME
19--------------------------------------------------------------------------------
20/u01/app/oracle/oradata/SKIPER/datafile/o1_mf_odbv_tbs_d920r9c5_.dbf

And a new table, based on table EMPLOYEES from schema HR.

1SQL> create table hr.employees_move tablespace odbv_tbs
2  2  as
3  3  select *
4  4  from hr.employees;
5 
6Table created.
7 
8SQL> insert into hr.employees_move select * from hr.employees_move;
9 
10107 rows created.
11 
12SQL> /
13 
14214 rows created.
15 
16SQL> /
17 
18428 rows created.
19 
20SQL> /
21 
22856 rows created.
23 
24SQL> /
25 
261712 rows created.
27 
28SQL> /
29 
303424 rows created.
31 
32SQL> /
33 
346848 rows created.
35 
36SQL> commit;
37 
38Commit complete.
39 
40SQL> alter system checkpoint;
41 
42System altered.
43 
44SQL> select data_object_id
45  2  from dba_objects
46  3  where object_name='EMPLOYEES_MOVE';
47 
48DATA_OBJECT_ID
49--------------
50     94710

Now let’s create a config file for ODBV

1[oracle@rico ~]$ cat odbv_show
294710:X|RED

The graphic representation for this table looks like this:
odbv_move1

Now let’s delete a bunch of rows from the table and check how it looks at a datafile level:

1SQL> delete hr.employees_move
2  2  where department_id in (50,80);
3 
410112 rows deleted.
5 
6SQL> commit;
7 
8Commit complete.
9 
10SQL> alter system checkpoint;
11 
12System altered.

odbv_move2

As you can see most of the table blocks have less then 50% of rows present in the block. You should understand that DELETE does not remove anything from the table block – the rows are only marked as deleted by flag 3C (the regular row is marked by flag 2C).

So this means that more then a half rows were marked as deleted in almost every block in a table.
Now let’s check what will happen after ALTER TALBE MOVE.

1SQL> alter table hr.employees_move move;
2 
3Table altered.
4 
5SQL> select data_object_id
6  2        ,object_id
7  3  from dba_objects
8  4  where object_name='EMPLOYEES_MOVE';
9 
10DATA_OBJECT_ID  OBJECT_ID
11-------------- ----------
12     94711      94710

odbv_move3

So what has happened in here? MOVE command forced building of a brand new segment. You can see, that DATA_OBJECT_ID has changed – that’s why I had to update the config file:

1[oracle@rico ~]$ cat odbv_show
294710:X|RED
394711:X|BLUE

Now the blue "X" represents the new instance of the table (the new segment) and the red "X" represents the old one. As you can see, the blocks were not removed from a tablespace. To be honest – they were just marked as reusable.

So let’s check what will happen when I’ll create a new table in the same tablespace:

1SQL> create table hr.employees_move2 tablespace odbv_tbs
2  2  as
3  3  select *
4  4  from hr.employees_move;
5 
6Table created.
7 
8SQL> insert into hr.EMPLOYEES_MOVE2
9  2  select *
10  3  from hr.EMPLOYEES_MOVE2;
11 
123584 rows created.
13 
14SQL> /
15 
167168 rows created.
17 
18SQL> /
19 
2014336 rows created.
21 
22SQL> commit;
23 
24Commit complete.
25 
26SQL> alter system checkpoint;
27 
28System altered.
29 
30SQL> select object_id, data_object_id
31  2  from dba_objects
32  3  where object_name='EMPLOYEES_MOVE2';
33 
34 OBJECT_ID DATA_OBJECT_ID
35---------- --------------
36     94712      94712

Now we have modify the config file again to display a blocks representation for the new table:

1[oracle@rico ~]$ cat odbv_show
294710:X|RED
394711:X|BLUE
494712:Y|WHITE

odbv_move4

The blocks marked as white Y consumed a lot of space, used previously by original segment (red X).

OK. Now let’s repeat the situation with SHRINK instead of MOVE.
I’ve created a new table in a new tablespace, called EMPLOYEES_SHRINK (I did it in a same way as a previous table)

1SQL> select data_object_id
2  2  from dba_objects
3  3  where object_name='EMPLOYEES_SHRINK';
4 
5DATA_OBJECT_ID
6--------------
7     94714

This is the visualisation of the table blocks, after deleting rows in a same way as previously and executing SHRINK command:

1SQL> delete hr.EMPLOYEES_SHRINK
2  2  where department_id in (50,80);
3 
410112 rows deleted.
5 
6SQL> commit;
7 
8Commit complete.
9 
10SQL> alter system checkpoint;
11 
12System altered.
13 
14SQL> alter table hr.EMPLOYEES_SHRINK enable row movement;
15 
16Table altered.
17 
18SQL> alter table hr.EMPLOYEES_SHRINK shrink space;
19 
20Table altered.
21 
22SQL> alter system checkpoint;
23 
24System altered.

odbv_shrink1

We can see above, that rows were packed into the first blocks of the segment.
Now let’s check how the situation will change after adding another table (like in previous example).

1SQL> ed
2Wrote file afiedt.buf
3 
4  1  create table hr.employees_shrink2 tablespace odbv_tbs
5  2  as
6  3  select *
7  4* from hr.employees_shrink
8SQL>
9SQL> /
10 
11Table created.
12 
13SQL> insert into hr.employees_shrink2 select * from hr.employees_shrink2;
14 
153584 rows created.
16 
17SQL> /
18 
197168 rows created.
20 
21SQL> /
22 
2314336 rows created.
24 
25SQL> commit;
26 
27Commit complete.
28 
29SQL> alter system checkpoint;
30 
31System altered.
32 
33SQL> select data_object_id
34  2  from dba_objects
35  3  where object_name='EMPLOYEES_SHRINK2';
36 
37DATA_OBJECT_ID
38--------------
39     94715

odbv_shrink2

As you can see, most of the blocks were consumed by blue "Y" (representation of table EMPLOYEES_SHRINK2).

Below you can see what will happen after executing ALTER TABLE SHRINK SPACE COMPACT and creating a new table in the same tablespace:

odbv_shrink3

As you can see, that a new table didn’t consume blocks of the old one. Old blocks are being reserved for the first table and will reused by it, when new data arrives.

So to sum things up:

  • ALTER TABLE MOVE creates a new physical segment at the end of datafile and assigns new DATA_OBJECT_ID (this is one of the reasons, you have to rebuild indexes after that)
  • ALTER TABLE SHRINK SPACE doesn’t create a new segment and does not change DATA_OBJECT_ID – instead the rows are being moved inside a blocks to defragment the object. Old extents can be reused by the same object or by other objects in the same tablespace
  • ALTER TABLE SHRINK SPACE COMPACT doesn’t create a new segment and does not change DATA_OBJECT_ID – instead the rows are being moved inside a blocks to defragment the object. Old extents can be reused only by the same object

That’s it. Hope I cleared some things up 🙂


Contact us

Database Whisperers sp. z o. o. sp. k.
al. Jerozolimskie 200, 3rd floor, room 342
02-486 Warszawa
NIP: 5272744987
REGON:362524978
+48 508 943 051
+48 661 966 009
info@ora-600.pl

Newsletter Sign up to be updated