使用 MAX() 在提交物化视图上可刷新

发布于 2024-12-28 14:22:03 字数 2102 浏览 0 评论 0原文

我受到实体属性值反模式的沉重打击。几年前的某一天,一个人认为 DDL 并不性感,并希望开发一些“足够灵活”的东西来保存有关人们的信息。他忽略了这样一个事实,即人们过去至少具有一些基本属性,如姓名、出生日期等。不仅如此,他还在该模式之上放置了一堆(充满副作用的)PL/SQL 包。这个东西成功地成为了其他应用程序所依赖的关键子系统。

快进几年,有 2000 万行。那家伙已经不在公司了,我必须处理这件事。我需要实现一些基本搜索,现在需要多个内部联接,并且在某些情况下需要永远进行。重写整个事情是不可能的,所以我想“枢轴”最重要的属性。

我认为物化视图可能是一个可行的选择,但我需要一些指导,因为我从未使用过它们。我想要一个这样的表:

  select
      uid,
       max(case when att = 'NAME' then UPPER(value) end) name,
       max(case when att = 'SURNAME' then UPPER(value) end) surname,
       max(case when att = 'BIRTH' then DATEORNULL(value) end) birth,
     ....,
     count(*) cnt
 from t
 group by uid

据我了解,阅读 Oracle 文档,我应该能够使用 MAX() 创建一个“REFRESHABLE ON COMMIT”物化视图 如果查询没有where子句

但无法让它发挥作用。我已经尝试过:

create materialized view log on t WITH SEQUENCE,ROWID,(value) INCLUDING NEW VALUES;

create materialized view t_view
 refresh fast on commit
 as
  select
      uid,
       max(case when att = 'NAME' then UPPER(value) end) name,
       max(case when att = 'SURNAME' then UPPER(value) end) surname,
       max(case when att = 'BIRTH' then DATEORNULL(value) end) birth,
     count(*) cnt
 from t
 group by uid

它适用于插入,但不适用于更新。我看到它能够完成这些事情:

  REFRESH_COMPLETE                                                             

  REFRESH_FAST                                                                 

  REFRESH_FAST_AFTER_INSERT

但我认为我还应该看到 REFRESH_FAST_AFTER_ONETAB_DML。有什么想法吗?

更新:dbms_mview.explain_mview 的输出

REFRESH_COMPLETE             |Y|
REFRESH_FAST                 |Y|
REFRESH_FAST_AFTER_INSERT    |Y|
REFRESH_FAST_AFTER_ONETAB_DML|N|mv uses the MIN or MAX aggregate functions
REFRESH_FAST_AFTER_ANY_DML   |N|see the reason why REFRESH_FAST_AFTER_ONETAB_DML is disabled
REFRESH_FAST_PCT             |N|PCT is not possible on any of the detail tables in the mater

I'm being hit hard by the entity-attribute-value antipattern. Some day, years ago, a guy decided that DDL wasn't sexy, and wanted to develop something "flexible enough" to keep information about people. He ignored the fact that people uses to have at least some basic attributes, as name, date of birth, and the like. Not only that, he put a bunch of (side-effects-ridden) PL/SQL packages on top of that schema. The thing managed to be a key subsystem in which other applications relied on.

Fast forward some years and 20 million rows. The guy isn't at the company anymore, and I have to deal with this. I need to implement some basic searches that right now would require multiple inner joins and just take forever for some cases. Rewriting the whole thing it's not possible, so I want to "pivot" the most important attributes.

I thought that materialized views could be a viable alternative, but I need some guidance, since I have never used them. I would like to get a table like this:

  select
      uid,
       max(case when att = 'NAME' then UPPER(value) end) name,
       max(case when att = 'SURNAME' then UPPER(value) end) surname,
       max(case when att = 'BIRTH' then DATEORNULL(value) end) birth,
     ....,
     count(*) cnt
 from t
 group by uid

as I understand reading Oracle docs, I should be able to create a "REFRESHABLE ON COMMIT" materialized view with MAX() if the query has no where clause.

But can't get it to work. I've tried:

create materialized view log on t WITH SEQUENCE,ROWID,(value) INCLUDING NEW VALUES;

create materialized view t_view
 refresh fast on commit
 as
  select
      uid,
       max(case when att = 'NAME' then UPPER(value) end) name,
       max(case when att = 'SURNAME' then UPPER(value) end) surname,
       max(case when att = 'BIRTH' then DATEORNULL(value) end) birth,
     count(*) cnt
 from t
 group by uid

It works for inserts, but not for updates. I see that is capable of these things:

  REFRESH_COMPLETE                                                             

  REFRESH_FAST                                                                 

  REFRESH_FAST_AFTER_INSERT

but I think I should see also REFRESH_FAST_AFTER_ONETAB_DML. Any ideas?

Update: Output of dbms_mview.explain_mview

REFRESH_COMPLETE             |Y|
REFRESH_FAST                 |Y|
REFRESH_FAST_AFTER_INSERT    |Y|
REFRESH_FAST_AFTER_ONETAB_DML|N|mv uses the MIN or MAX aggregate functions
REFRESH_FAST_AFTER_ANY_DML   |N|see the reason why REFRESH_FAST_AFTER_ONETAB_DML is disabled
REFRESH_FAST_PCT             |N|PCT is not possible on any of the detail tables in the mater

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。

评论(1

苦笑流年记忆 2025-01-04 14:22:03

MV_CAPABILITIES_TABLE.MSGTXT 是错误的,您真正需要做的是将 case 替换为 decode

当我在 11g 上尝试此操作时,我收到消息物化视图中存在 CASE 表达式。将其更改为使用 decode 修复了 10g 和 11g 上的问题。

The MV_CAPABILITIES_TABLE.MSGTXT is wrong, what you really need to do is replace case with decode.

When I tried this on 11g I got the message CASE expressions present in materialized view. Changing it to use decode fixed it on both 10g and 11g.

~没有更多了~
我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
原文