PL/SQL Challenge 每日一题:2014-3-14 11gR2中带RELIES_ON子句的RESULT_CACHE函数

最先答对且答案未经编辑的puber将获得纪念章一枚(答案不可编辑但可发新贴补充或纠正),其他会员如果提供有价值的分析、讨论也可获得纪念章一枚。

以往旧题索引:
http://www.itpub.net/forum.php?mod=forumdisplay&fid=3&filter=typeid&typeid=1808

原始出处:
http://www.plsqlchallenge.com/

作者:        mentzel.iudith

运行环境:SQLPLUS, SERVEROUTPUT已打开
注:本题给出答案时候要求给予简要说明才能得到奖品
执行环境: 11gR2

我创建了如下的表和数据:

CREATE TABLE plch_employees
(
   employee_id  INTEGER
, last_name    VARCHAR2(100)
, salary       NUMBER
)
/

BEGIN
   INSERT INTO plch_employees
      VALUES (100, 'Steven', 10000);

COMMIT;
END;
/

哪些选项实现了一个名为plch_salary的函数,使得下列代码块执行之后:

BEGIN
    DBMS_OUTPUT.put_line( plch_salary(100) );
    DBMS_OUTPUT.put_line( plch_salary(100) );

UPDATE plch_employees
    SET   salary = salary * 2
    WHERE employee_id = 100 ;

COMMIT;

DBMS_OUTPUT.put_line( plch_salary(100) );
END;
/

我们可以看到结果中有三个完全一致的数值?

(A)
CREATE OR REPLACE FUNCTION plch_salary
           (p_employee_id_in IN plch_employees.employee_id%TYPE)
       RETURN plch_employees.salary%TYPE
       RESULT_CACHE
       RELIES_ON (plch_employees)
AS
    l_salary   plch_employees.salary%TYPE;
BEGIN
    SELECT salary
    INTO l_salary
    FROM plch_employees
    WHERE employee_id = p_employee_id_in ;

RETURN l_salary ;
END;
/

(B)
CREATE OR REPLACE FUNCTION plch_salary
           (p_employee_id_in IN plch_employees.employee_id%TYPE)
       RETURN plch_employees.salary%TYPE
       RESULT_CACHE
       RELIES_ON (plch_employees)
AS
    l_salary   plch_employees.salary%TYPE;
BEGIN
    UPDATE plch_employees
    SET   salary = salary * 2
    WHERE employee_id = p_employee_id_in
    RETURNING salary INTO l_salary ;

COMMIT;

RETURN l_salary ;
END;
/

(C)
CREATE OR REPLACE FUNCTION plch_salary
            (p_employee_id_in IN plch_employees.employee_id%TYPE)
       RETURN plch_employees.salary%TYPE
       RESULT_CACHE
       RELIES_ON (plch_employees)
AS
    l_salary   plch_employees.salary%TYPE;
BEGIN
    UPDATE plch_employees
    SET   salary = salary * 2
    WHERE employee_id = p_employee_id_in ;

COMMIT;

SELECT salary
    INTO l_salary
    FROM plch_employees
    WHERE employee_id = p_employee_id_in ;

RETURN l_salary ;
END;
/

(D)
CREATE OR REPLACE FUNCTION plch_salary
            (p_employee_id_in IN plch_employees.employee_id%TYPE)
       RETURN plch_employees.salary%TYPE
       RESULT_CACHE
       RELIES_ON (plch_employees)
AS
    l_salary   plch_employees.salary%TYPE;
BEGIN
  EXECUTE IMMEDIATE
     'SELECT salary FROM plch_employees
        WHERE employee_id = :employee_id'
      INTO l_salary
      USING p_employee_id_in ;

RETURN l_salary ;
END;
/

(E)
CREATE OR REPLACE FUNCTION plch_salary
           (p_employee_id_in IN plch_employees.employee_id%TYPE)
       RETURN plch_employees.salary%TYPE
       RESULT_CACHE
       RELIES_ON (plch_employees)
AS
    l_salary   plch_employees.salary%TYPE;
BEGIN
    DELETE FROM plch_employees
      WHERE employee_id = p_employee_id_in
      RETURNING salary INTO l_salary ;

COMMIT;

RETURN l_salary ;
END;
/

(F)
CREATE OR REPLACE FUNCTION plch_salary
           (p_employee_id_in IN plch_employees.employee_id%TYPE)
       RETURN plch_employees.salary%TYPE
       RESULT_CACHE
       RELIES_ON (plch_employees)
AS
    l_salary   plch_employees.salary%TYPE;
BEGIN
    INSERT INTO plch_employees
       VALUES ( 500 + p_employee_id_in,
                'New Employee',
                (SELECT 2*salary FROM plch_employees
                    WHERE employee_id = p_employee_id_in)
              )
       RETURNING salary INTO l_salary ;
      
    COMMIT;

RETURN l_salary ;
END;
/

liwei9902
发表于 2014-3-18 10:13:42

a 猜的

renfengjun
发表于 2014-3-18 10:23:42

正确答案:B、E、F
A 错误,update之后,result cache改变了。
B 结果正确,但是有点无法理解。
C 错误,第二次执行plch_salary(100) 的时候,其实使用的是result cache,没有真正执行update,在匿名块的update以后,result cache就改变了。所以其实一共执行了三次update。
D 错误,前两次的结果肯定是一样的,执行update之后,result cache改变了。
E 正确,结果正确,和B一样,无法理解。
F 正确,结果正确,和BE一样,无法理解。或者说只要加returning 子句的dml语句,都可以认定为不会影响result cache?并且也不受其他语句的影响?

liwei9902
发表于 2014-3-18 14:03:26

看了解答,发现result cache不是很实用,至少在我需要的功能中,不很实用

newkid
发表于 2014-3-19 03:23:49

答案BEF, 3楼得奖。

A: 函数结果被缓存,所以前两个调用都返回10000。但是,在UPDATE被执行后,函数再次被执行,这次返回20000。
B: 在Oracle 11gR2, RELIES_ON子句被忽略,数据库自动跟踪缓存结果所依赖的对象。令人惊奇的是,如果函数只在一张表上执行了一个UPDATE,并利用RETURNING子句返回一个列值,这张表并不会被注册为缓存结果所依赖的对象。

这个直接后果导致所有的UPDATE语句(函数内有一个,匿名块中有一个)都不会影响缓存结果,所以三次输出都是同样的值20000。
必须注意这个现象与11gR1不同,在该版本中RELIES_ON决定了缓存结果的依赖关系。

C:  一旦函数中包含了一个SELECT语句,缓存结果就会标记为依赖于PLCH_EMPLOYEES,所以前两个函数调用都返回20000, 但是在匿名块中的UPDATE语句执行之后,函数再次被执行,这次它返回80000。

在Oracle 11gR1中,这个选项的结果会有所不同,对于本题目同样是不正确的:

20000
40000
160000

D:
结果缓存依赖关系是由Oracle在执行的时候建立的,不是在编译的时候建立的。因此,虽然这个函数包含了一个动态的SELECT语句,结果缓存仍然会依
赖于PLCH_EMPLOYEES表,结果和静态的SELECT一样。前两个函数调用返回10000,
但是在UPDATE执行之后,第三个结果是20000。

E: 如果函数只在一张表上执行了一个DELETE,并利用RETURNING子句返回一个列值,这张表并不会被注册为缓存结果所依赖的对象。结果,不论是DELETE语句或者是匿名块中的UPDATE语句都不会使得缓存结果失效。
结果,我们会看到10000被显示三次。

必须注意这个现象与11gR1不同,在该版本中RELIES_ON决定了缓存结果的依赖关系。在该版本中,我们会看到10000仅仅被显示一次,随后是两个空值,因为在记录删除之后,缓存结果失效,函数再次执行并且返回了空值。

F: 此处函数确实执行了plch_employees表上的SELECT语句,但是仅仅用于INSERT语句中的VALUES子句的标量子查询。
在这种情况下,这张表并不会被注册为缓存结果所依赖的对象,因此,不论是INSERT语句或者是匿名块中的UPDATE语句都不会使得缓存结果失效。
结果,函数只会被执行一次,缓存结果20000会被显示三次。
必须注意这个现象与11gR1不同,在该版本中RELIES_ON决定了缓存结果的依赖关系。在该版本中,每次函数执行都会使得缓存结果失效,原因是
INSERT语句,所以,前两次执行会返回同样的值20000,而第三次执行发生于UPDATE语句之后,会返回40000。

dcswinner
发表于 2014-3-20 23:08:31

newkid 老师真牛逼,学习了。

newkid
发表于 2014-3-21 02:29:35

dcswinner 发表于 2014-3-20 23:08 static/image/common/back.gif
newkid 老师真牛逼,学习了。

NB个P, 这道题考的是11gR2的一个疑似bug的怪异表现,所以我做错了。

Bleach_ud
发表于 2014-3-22 10:27:52

newkid 发表于 2014-3-19 03:23 static/image/common/back.gif
答案BEF, 3楼得奖。

A: 函数结果被缓存,所以前两个调用都返回10000。但是,在UPDATE被执行后,函数再次 ...

漲姿勢了~~~

页:
[1]

PL/SQL Challenge 每日一题:2014-3-14 11gR2中带RELIES_ON子句的RESULT_CACHE函数的更多相关文章

  1. 【python】Leetcode每日一题-寻找旋转排序数组中的最小元素

    [python]Leetcode每日一题-寻找旋转排序数组中的最小元素 [题目描述] 已知一个长度为 n 的数组,预先按照升序排列,经由 1 到 n 次 旋转 后,得到输入数组.例如,原数组nums ...

  2. [LeetCode每日一题]80. 删除有序数组中的重复项 II

    [LeetCode每日一题]80. 删除有序数组中的重复项 II 问题 给你一个有序数组 nums ,请你 原地 删除重复出现的元素,使每个元素 最多出现两次 ,返回删除后数组的新长度. 不要使用额外 ...

  3. 【python】Leetcode每日一题-寻找旋转排序数组中的最小元素2

    [python]Leetcode每日一题-寻找旋转排序数组中的最小元素2 [题目描述] 已知一个长度为 n 的数组,预先按照升序排列,经由 1 到 n 次 旋转 后,得到输入数组.例如,原数组nums ...

  4. [每日一题2020.06.14]leetcode #70 爬楼梯 斐波那契数列 记忆化搜索 递推通项公式

    题目链接 题意 : 求斐波那契数列第n项 很简单一道题, 写它是因为想水一篇博客 勾起了我的回忆 首先, 求斐波那契数列, 一定 不 要 用 递归 ! 依稀记得当年校赛, 我在第一题交了20发超时, ...

  5. [ X.XX]CF每日一题系列线下更新中

    现在做的都好水啊,就不写博客了,线下有记录滴

  6. Oracle数据库—— PL/SQL基础编程

    一.涉及内容 1. 掌握PL/SQL程序块的结构,理解并熟悉各种变量的应用. 二.具体操作 (一)使用system用户登录SQL*PLUS,使用SQL语句创建用户:u_你的姓名首字母(例如:u_zs) ...

  7. PL/SQL学习笔记(三)

    -----创建一个序列,再创建一个表(主键是数字),通过序列生成该表的主键值. create table mytb1( m_id number primary key, m_name ) not nu ...

  8. PL/SQL精明的调用栈分析

    PL/SQL精明的调用栈分析 原文:http://www.oracle.com/technetwork/issue-archive/2014/14-jan/o14plsql-2045346.html ...

  9. Oracle PL/SQL随堂笔记总结

    1.pl/sql编程 2.存储过程 3.函数 4.触发器 5.包 6.pl/sql基础 -定义并使用变量 7.pl/sql的进阶 8.oracle的视图 1.pl/sql编程 1.理解oracle的p ...

随机推荐

  1. bzoj4591 [Shoi2015]超能粒子炮·改——组合数学(+求阶乘逆元新姿势)

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=4591 这题不是很裸啊(所以我就不会了) 得稍微推导一下,看这个博客好了:https://bl ...

  2. input的type等于mage标签引发的页面提交!

    主要原因是使用了<input type="image",而且放在了<form中. <input type="image"标签类似<inp ...

  3. NOI2015 软件包管理器(树链剖分+线段树)

    P2146 软件包管理器 题目描述 Linux用户和OSX用户一定对软件包管理器不会陌生.通过软件包管理器,你可以通过一行命令安装某一个软件包,然后软件包管理器会帮助你从软件源下载软件包,同时自动解决 ...

  4. 手机不ROOT使用xposed框架-微信群同步-微咖讲师-安装教程

    其实已经很多人知道Xposed框架这个便捷的工具,但是呢由于框架必须root,导致很多用户望而却步,今天就来教下大家,如何不root玩转xposed框架 1.下载 SandVxposed免root 虚 ...

  5. [Codeforces]Codeforces Round #490 (Div. 3)

    Mishka and Contest #pragma comment(linker, "/STACK:102400000,102400000") #ifndef ONLINE_JU ...

  6. [转]五个Linux下用户空间的调试工具

    有几个Linux下的用户空间调试工具和技术,它们用来分析用户空间的问题相当有用.它们是: 'print' 语句 查询 (/proc, /sys 等) 跟踪 (strace/ltrace) Valgri ...

  7. Windows2008 Server 常规设置及基本安全策略

    一.系统及程序 1.屏幕保护与电源 桌面右键--〉个性化--〉屏幕保护程序屏幕保护程序 选择无更改电源设置 选择高性能选择关闭显示器的时间 关闭显示器 选 从不 保存修改 2.安装IIS 管理工具-- ...

  8. mysql有关时间是问题

     mysql中有关时间的类型 date/datetime/time/timestamp/year date:表示日期的类型,格式为:“yyyy-MM-dd” dateTime:表示日期时间的类型,格式 ...

  9. Node.js概述

    Node.js最重要的特性:通过单线程实现异步处理环境 Node.js解决的问题: Node.js修改了客户端连接服务器端的连接方法,不需要为每个客户端连接创建一个新的线程,而是为每个客户端连接触发一 ...

  10. FTP工作原理

    FTP工作原理 FTP两种传输方式:1.ASCII传输2.二进制传输 FTP主被动原理: 主动方式:1.用户与服务器建立控制通道2.客户端发出PORT指令,主动告诉服务器端口号3.服务器主动通过20端 ...