首页
论坛
门户
空间
手机版
IXPUB
插件
收藏
设置
注册
登录
商店
搜索
培训
Wiki
Blog
归档
丛书
退出
ITPUB论坛
»
Oracle Application Server套件
» Oracle8i和9i中PLSQL程序的不同运行结果
‹‹ 上一主题
|
下一主题 ››
投票
交易
悬赏
活动
评价
|
打印
|
推荐
|
订阅
|
收藏
标题:
[精华]
Oracle8i和9i中PLSQL程序的不同运行结果
depp
隐忍猪卷风
精华贴数 3
个人空间
0
技术积分 1545 (1087)
社区积分 58 (4514)
注册日期 2001-11-5
论坛徽章:5
#1
使用道具
发表于 2004-8-24 16:26
Oracle8i和9i中PLSQL程序的不同运行结果
在把ORACLE的数据库从8i升级到9i及以上的时候,一般认为原有的PLSQL程序应该完全兼容,即运行过程和运行结果完全一致。遗憾的是,事实并非如此,由于ORACLE PLSQL引擎的升级,它对某些代码解释做了更改,导致某些代码会有不同的运行结果。各位在升级数据库时必须重视,否则将导致无法估量的损失和难以恢复的灾难。
1, PLSQL表作为参数传递
先看以下代码,在ORACLE 8i和9i中的运行结果。
declare
type test_rec is record
(
col_1 varchar2(100)
);
type test_tbl is table of test_rec index by binary_integer;
l_tbl test_tbl;
procedure change_value
is
begin
l_tbl(1).col_1 := 'I am changed!';
end;
procedure sub_test(pi_str in varchar2)
is
begin
dbms_output.put_line('before: '||pi_str);
change_value;
dbms_output.put_line('after : '||pi_str);
end;
begin
l_tbl(1).col_1 := 'I am ok!';
sub_test(l_tbl(1).col_1);
exception
when others then
dbms_output.put_line(sqlerrm);
end;
示例代码非常简单,即将PLSQL表的某个成员变量当作参数给另一过程,此过程改变了原PLSQL表的值,但未改变传入参数的值(当然不能改,表示为IN的参数的值是不能改的),观看改变前后,传入参数的值在Oracle 8i和9i的变化:
运行结果:
Oracle 8i Oracle 9i
before: I am ok! before: I am ok!
after : I am ok! after : I am changed!
显然运行结果不一样!在Oracle 8i中,传入参数的值在原PLSQL表的值改变前后未变化,而在Oracle 9i中,传入参数的值被改动了。
Oracle给出的解释是:在Oracle 8i中,所有表示为IN的参数传递都是传值的,包括PLSQL表类型的参数。而到了Oracle 9i,他们觉得PLSQL表类型的参数传递应该传引用,在PLSQL引擎上做了这样的修改,而导致这个问题。
我们来回忆以下,传值意味2个变量传递的是真实的数值,各自有不同的内存空间,相当于变量被拷贝了一份,各为其主,互不相干。传引用意味2个变量传递的是内存空间的地址,指向同一块内存空间,如果此内存空间里放的数值被改变了,那么2个变量的值都会被改变。
了解问题产生的原因,回头再读前面的示例代码,就比较容易理解了。同样的程序,在数据库升级后产生了不同的运行结果,这个问题的危险程度相信大家一定能明白,必须重视。
2, PLSQL表类型返回值NO_DATA_FOUND意外
看以下代码:
DECLARE
l_test VARCHAR2(10);
type test_rec is record (col_a varchar2(100));
TYPE test_tab IS TABLE OF test_rec INDEX BY BINARY_INTEGER;
l_test_tab test_tab;
FUNCTION return_tbl ( pi_dummy IN VARCHAR2 )
RETURN test_tab
IS
l_tbl test_tab;
BEGIN
l_tbl.DELETE;
l_tbl(1).col_a := 'I am ok!';
RETURN l_tbl;
EXCEPTION
WHEN OTHERS THEN
l_tbl.DELETE;
RETURN l_tbl;
END;
BEGIN
l_test_tab := return_tbl('');
l_test := l_test_tab(1).col_a;
DBMS_OUTPUT.PUT_LINE ( 'before: ' || l_pol_num );
l_test := return_tbl('')(1).col_a;
DBMS_OUTPUT.PUT_LINE ( 'after : ' || l_test );
EXCEPTION
WHEN NO_DATA_FOUND THEN
DBMS_OUTPUT.PUT_LINE ( 'NO_DATA_FOUND exception!' );
END;
这段代码意味某函数返回一个PLSQL表类型的值,然后不同的引用方式,在屏幕上显示。
运行结果:
Oracle 8i Oracle 9i
before: I am ok! before: I am ok!
after : I am ok! NO_DATA_FOUND exception!
在Oracle 8i中,直接用函数名和下标访问PLSQL表的成员变量是合法的,但到了Oracle 9i,这种方式会导致一个运行期NO_DATA_FOUND意外,而产生不同的运行结果。
这个问题Oracle没有给出严格解释,只是指出这样的方式不再合法而已。大家同样需要重视这个问题,以免掉入这个陷阱。
数据库的升级导致PLSQL程序有不同的运行结果,这样的问题让人担忧,ORACLE的行事方式让人头疼。若有很多的FORM、REPORT和PLSQL存储过程,这个问题导致的代码检查修改和产生的工作量是非常巨大的,而且后期测试也需要消耗大量的资源。希望ORACLE以后不要再发生这样的事情,那真会伤了一直很信任你的粉丝们的心。
__________________
天生蠢材也有用,美眉跑掉还会来混迹于
应用服务器版
,如有兴趣可致邮
depp@itpub.net
欢迎光临我的博客站
MSN:joizhang@21cn.com
只看该作者
gengmao
谦卑之心
精华贴数 1
个人空间
0
技术积分 1011 (1808)
社区积分 27 (6526)
注册日期 2001-10-9
论坛徽章:6
#2
使用道具
发表于 2004-8-24 22:46
Sigh...
pl/sql从来没有严格的语言规范。即使有,oracle也是想改就改。。。
只看该作者
孤岛
资深会员
精华贴数 0
个人空间
0
技术积分 633 (2997)
社区积分 16579 (91)
注册日期 2003-4-12
论坛徽章:62
#3
使用道具
发表于 2008-1-8 16:51
恩,这个帖子真的是非常震撼阿,搞了这么多年这些东西,都没有发现有这样的问题,不知道楼主是这么发现的,佩服一下这种钻研精神啊。
对于第一个,我个人还是比较赞同oracle公司的解释:这个表变量,顾名思义,我们就把他当成一个table来理解:
在同一个session,改变了表的值,最后取得值应该是这个表的最新值。
这样子去想,个人感觉,传引用比传值更合理阿
__________________
空白
只看该作者
flowerbird
老会员
精华贴数 0
个人空间
0
技术积分 750 (2527)
社区积分 1787 (650)
注册日期 2004-12-25
论坛徽章:16
#4
使用道具
发表于 2008-1-14 13:34
......
__________________
以儒修身,以道端行,以佛悯苍生
http://www.frappr.com/?a=myfrappr&id=536261
只看该作者
jetbo
思维创造动力
精华贴数 0
个人空间
0
技术积分 566 (3406)
社区积分 53 (4725)
注册日期 2003-10-10
论坛徽章:2
#5
使用道具
发表于 2008-3-17 13:56
类似问题遇到的太多了,8i,9i,10g,经常会遇到...
__________________
提供Oracle EBS银行、航空、钢铁、贸易、制造行业财务、分销、制造业务及二次开发解决方案
UnderWorld:OracleERP有一感觉叫"口渴",有一种动作叫"喝水"...
只看该作者
ora2008
精华贴数 0
个人空间
40
技术积分 2252 (702)
社区积分 134 (2962)
注册日期 2008-5-6
论坛徽章:5
#6
使用道具
发表于 2008-5-6 21:50
8i可惜沒用過
__________________
享受ORACLE......,我的观点是快乐ORACLE.---米卢
只看该作者
投票
交易
悬赏
活动
相关内容
ITPUB论坛
≡ 数据库技术 ≡
> Oracle数据库管理
> Oracle开发
> Oracle Developer Suite
> Oracle入门与认证
> Oracle专题深入讨论
> Oracle新技术/11g
> Oracle电子文档
> Oracle Application Server套件
> IBM数据库产品
> MS SQL Server
> Sybase管理与开发
> MySQL及其它开源数据库
> 内存数据库
> 数据仓库与数据挖掘
> 移动及嵌入式数据库
≡ 企业信息化 ≡
> ERP产品与实践
> CRM产品与实践
> HR产品与实践
> 物流
> 供应链
> 供应链建模与仿真
> 物流设备与系统工程
> 企业管理咨询
> 管理协同与办公自动化
> IT服务管理
> 数据中心建设
> ERP二次开发
> Oracle ERP
> EBS相关文档
> PeopleSoft与JDE
> SAP R/3
> SAP Business One开发与快速实施
> SAP财务及CRM
> SAP后勤及HR
> mySAP ERP
> 系统开发及跨应用设置
> SAP相关文档
> 国外其它ERP产品
> 国内ERP产品
≡ 开发技术 ≡
> Java入门与认证版
> Java web开发及框架技术
> Java企业开发
> ASP.NET【已迁移到微软开发技术论坛】
> .Net企业开发与应用【已迁移到微软开发技术论坛】
> WEB程序开发
> WEB 2.0技术
> 动态语言
> 移动与游戏开发
≡ 系统设计与项目管理 ≡
> 系统分析与UML
> 系统分析与UML精华区
> 项目管理
> 项目过程
> 软件测试
> 算法讨论与研究
≡ IBM软件技术园地 ≡
> IBM数据库产品
> Lotus
> Tivoli
> Websphere
> Rational
> 与SOA相关的IBM产品与技术
> IBM软件技术精英协会
> 软件技术精英活动专版
≡ 操作系统与硬件 ≡
> AIX及IBM产品【已迁移到IXPUB】
> HP-UX及HP产品【已迁移到IXPUB】
> Solaris及SUN产品【已迁移到IXPUB】
> Linux及其应用 【已迁移到IXPUB】
> 其它UNIX系统【已迁移到IXPUB】
> windows系统及微软相关产品 【已迁移到IXPUB】
> 存储设备与容灾技术 【已迁移到IXPUB】
> 服务器 【已迁移到IXPUB】
≡ 行业纵向讨论区 ≡
> IT业界评论与展望
> 政府与教育事业
> 中国政府信息主管联盟
> 电信行业
> 金融行业
> 医卫行业
> 制造行业
> 电力行业
> 信息安全与审计
≡ 会员交流 ≡
> IT职业生涯
> 招聘求职商务信息
> 体育世界
> 体育博彩专版
> 旅游,驴友
> 汽车世界
> 外语角
> 数码摄影
> 你的故事我的歌
> 音乐推荐区
> 电子图书与IT文档资料
> 软件交流
> 软件交流精华区
≡ ITPUB产品与服务 ≡
> ITPUB地面活动专版
> BLOG天地
> WIKI世界
> 授权用户区
> 站务管理
≡ 微软开发技术 ≡
> 开发工具和语言
> .NET Framework 相关
> Visual Basic/VB.net
> Visual C#
> Visual C++/vc.net
> Visual Studio
> .NET软件架构与模式
> .NET开发辅助工具及框架
> Web开发
> ASP.NET与AJAX
> Web相关技术讨论(IIS等)
> Silverlight 技术
> 微软企业级产品技术
> SQL Server
> windows server
> SharePoint
> Exchange Server
> Biztalk
> 嵌入式及移动开发
> Windows Embedded 嵌入式技术
> Windows 移动设备
> Office开发
> Microsoft office system
> Office Business Application
> 微软产品用户交流区
> .Net电子书籍&&书籍介绍
> .Net人才交流
技术积分榜
社区积分榜
徽章
电子杂志
会员
团队
统计
邮箱
游乐场
帮助
TOP
CopyRight 1999-2006 itpub.net All Right Reserved.
北京皓辰广域网络信息技术有限公司. 版权所有
E-mail:Webmaster@itpub.net
京ICP证:010037号
联系我们
法律顾问
控制面板首页
编辑个人资料
积分交易
公众用户组
好友列表
升级个人空间
基本概况
论坛排行
主题排行
发帖排行
积分排行
在线时间
管理团队
管理统计