Lightning Fast SQL with Real Time Materialized Views: 12 Things Developers Will Love About Oracle Database 12c Release 2 Part 5

Chris Saxon
Oracle Developers
Published in
3 min readNov 10, 2016

Materialized views (MVs) can give amazing performance boost. Once you create one based on your query, Oracle can get the results direct from the MV instead of executing the statement itself. This can make SQL significantly faster. Especially when the query processes millions of rows but there are only a handful in the output.

There’s just one problem.

The data in the MV has to be fresh. Otherwise Oracle won’t do the rewrite.

You could of course query the MV directly. But the data will still be old.

So you need to keep the materialized view up-to-date. The easiest way is to declare it as “fast refresh on commit”.

But this is easier said than done. Doing this has a couple of issues:

So if you have complex SQL you may not be able to use query rewrite. And even if you can, on high transaction systems the refresh overhead may cripple your system.

So instead of “fast refresh on commit”, you make the MV “fast refresh on demand”. And create a job to update it. Which runs every second!

But no matter how frequently you run the job, there will always be times when the MV is stale. So query performance could switch between lightning fast and dog slow. A guaranteed way to upset your users!

So how do you overcome this?

With real time materialized views!

These give the best of both worlds. You can refresh your MV on demand. But still have it return up-to-date information.

To do this, create the MV with the clause:

on query computation

For example:

create table t (
x not null primary key, y not null
) as
select rownum x, mod(rownum, 10) y
from dual connect by level <= 1000;
create materialized view log on t
with rowid (x, y) including new values;
create materialized view mv
refresh fast on demand
enable on query computation
enable query rewrite as
select y , count(*) c1
from t
group by y;

With this, you can add more data to your table:

insert into t 
select 1000+rownum, 1 from dual
connect by level <= 100;
commit;

And Oracle can still use the MV to rewrite. Even though the MV is stale!

select /*+ rewrite */y , count(*) from t group by y;

It does this by:

  • Querying the stale MV
  • Then applying the inserts, updates and deletes in the MV log to it

This can lead to some scary looking execution plans!

The point to remember is Oracle is reading the materialized view log. Then applying the changes to the MV. So the longer you leave it between refreshes, the more data there will be. You’ll need to test to find the sweet spot to balancing the refresh process and applying MV change logs on query rewrite.

You can even get the up-to-date information when you query the MV directly. To do so, add the fresh_mv hint:

select /*+ fresh_mv */* from mv;

The really cool part?

You can convert your existing MVs to real time with the following command:

alter materialized view mv enable on query computation;

This makes MVs much easier to work with, opening up your querying tuning options!

Full article originally published at blogs.oracle.com on November 10, 2016.

--

--

Chris Saxon
Oracle Developers

I’m Chris Saxon, an Oracle Developer Advocate for SQL. My job is to help you get the best out of the Oracle Database and have fun with SQL!