- 在线时间
- 0 小时
- 最后登录
- 2007-9-23
- 注册时间
- 2004-9-10
- 听众数
- 3
- 收听数
- 0
- 能力
- 0 分
- 体力
- 9975 点
- 威望
- 7 点
- 阅读权限
- 150
- 积分
- 4048
- 相册
- 0
- 日志
- 0
- 记录
- 0
- 帖子
- 1893
- 主题
- 823
- 精华
- 2
- 分享
- 0
- 好友
- 0

我的地盘我做主
该用户从未签到
 |
长期以来,我一直用的是 MS SQL Server / Access 数据库,通过.NET 访问MS自家的东西几乎没碰到过什么麻烦。最近项目中要用 Oracle 作为数据库,学习研究了一些 .NET 访问Oracle 的东西,发现问题倒真的不少。 : h! e1 b- n. s4 J
< >& x4 p! t. g' n2 B
: t" \ o7 J5 x! s
flash/swflash.cab#version=5,0,0,0 height=280 width=320 classid=clsid 27CDB6E-AE6D-11cf-96B8-444553540000>200309/guangli_320.swf">200309/guangli_320.swf"> 200309/guangli_320.swf" width=320 height=280 type="application/x-shockwave-flash" pluginspage="http://www.macromedia.com/shockwave/download/index.cgi?P1_Prod_Version=ShockwaveFlash">1。System.Data.OracleClient 和 System.Data.OleDb 命名空间</P>* j( `! n7 K) Q) D- n2 c# v1 C
< > 虽然通过这两个命名空间的类都可以访问 Oracle 数据库,但和 SQL Server 类似的(System.Data.SqlClient 命名空间的类效率要比 System.Data.OleDb 命名空间中的类高一些),System.Data.OracleClient 命名空间中的类要比 System.Data.OleDb 命名空间的类效率高一些(这一点我没有亲自验证,但大多数地方都会这么说,而且既然专门为 Oracle 作的东西理论上也应该专门作过针对性的优化)。</P>
- |$ ~; O& l! z3 S6 ?: D0 j. t$ v0 u: w< > 当然还有另一点就是从针对性上说,System.Data.OracleClient 要更好一些:</P>
; n; P! Q, F& n1 ?% T< > 比如数据类型,System.Data.OleDb.OleDbType 枚举中所列的就没有 System.Data.OracleClient.OracleType 枚举中的那些有针对性;另外,Oracle 的Number 类型如果数字巨大,超出 .NET 数据类型范围的情况中,就必须使用System.Data.OracleClient 中的专门类 -- OracleNumber 类型。</P>+ X$ ?- R" p+ p
< > 好了,不再赘述这两个的比较,下面主要讨论System.Data.OracleClient 命名空间中的类型,即 ADO.NET for Oracle Data Provider (数据提供程序)。</P>4 t4 ~2 Q4 P s4 H* n. I2 @
< >2。数据库连接:</P>& F8 C7 L# k9 a" x( M
< > 无论是 System.Data.OleDb 还是 System.Data.OracleClient 访问 Oracle 都需要在 .NET 运行的机器(ASP.NET 中就是 Web 服务器)安装 Oracle 客户端组件。(这一点是和 MS 的两种数据库不同的,MS 的东西安装 MDAC: Microsoft Data Access Component 2.6 以上版本后,就无须再安装 SQL Server 客户端或者 Office 软件,就能访问。)</P>
9 [7 e- ~2 y+ c% s3 R( I& |6 _< >System Requirements:</P>
0 I/ r8 z x, i' Z! B, L< > (1)如用 System.Data.OracleClient 访问 Oracle,客户端组件版本应在 Oracle 8i Client Release 3 (8.1.7)以上版本。MS 只确保访问 Oracle 8.1.6、Oracle 8.1.7、Oracle 9i 服务器时的情况。MDAC 2.6 以上。</P>
, m: X0 t9 p* w0 {) x% N< > (2)如用 System.Data.OleDb 访问 Oracle,客户端组件版本 7.3.3.4.0 以上或 8.1.7.4.1 以上。MDAC 2.6 以上。</P>9 x" ~) ^- W: t( \
< > 如服务器为 Oracle8i 以上,客户端组件版本应为 8.0.4.1.1c。</P>8 f; K' i% l6 c+ E; n& o# J* y. `) H
< > 在 .NET 运行的机器中,安装 Oracle 客户端,然后打开 Net Manager (Oracle 9i) / Easy Config (Oracle 8i) 按你以前的经验设置本地服务的映射(这里的服务名将用于数据库连接串)。</P>
# L8 e( t# D" j# Y m& [, w< > System.Data.OracleClient 中访问 Oracle 数据库的连接串是:</P>2 L* V6 |7 S0 S4 n+ Z
< >User ID=用户名; Password=密码; Data Source=服务名</P>' }* H- D5 L: [% A5 d6 e
< > (上述为一般的连接串,详细的连接串项目可以在 System.Data.OracleClient.OracleConnection.ConnectionString 属性的文档中找到。)</P>
8 n _. z/ x1 W6 Z< > System.Data.OleDb 中的访问 Oracle 数据库的连接串是:</P>$ u( I5 ]3 T O" g5 G; F @
< > rovider=MSDAORA.1; User ID=用户名; Password=密码; Data Source=服务名</P>( e* r% b+ B$ t7 l O% B4 C
< >3。Oracle 中的数据类型:</P>
4 z1 D! L$ C0 g! d' ?< > Oracle 的数据类型和 SQL Server 相比,要“奇怪”一些:SQL Server 的大多数据类型很容易找到 .NET 中比较接近的类型,Oracle 中的类型就离 .NET 类型远了许多,毕竟 Oracle 是和 Java 亲近的数据库。</P>8 r4 O& G: o% K" g/ D: f
number: 数字类型,一般是 Number(M,N),M是有效数字,N是小数点后的位数(默认0),这个是按十进制说的。 6 Q1 A W8 ?: j3 d! a6 \
nvarchar2: 可变长字符型(Unicode),这个比较像 SQL Server 的 nvarchar(但不知 Oracle 为什么加了个“2”)。(去掉“n”为非 Unicode 的,下同。)
+ @: O+ _5 [2 [ s: o3 hnchar: 定长字符型(Unicode)。 % ]) ~. D$ l9 p; B1 k0 ~7 f
nclob: “写作文”的字段,存储大量字符(Unicode)时用。 ; V: `% f E0 V. h6 A
date: 日期类型,比较接近 SQL Server 的 datetime。
2 w7 i# `" }( z' h) R7 | H< > Oracle 中字段不能是 bit 或者 bool 之类的类型,一般是 number(1) 代替的。</P>' s# V% L$ x5 b' R: f7 M7 A; F6 z
< > 和 SQL Server 一样在 SQL 命令中,字符类型需要用单引号(')隔开,两个单引号('')是单引号的字符转义(比如: I'm fat. 写入一个 SQL 命令是: UPDATE ... SET ...='I''m fat.' ...)。</P> h1 ]2 S- }* `# @3 ~8 D: _
< > 比较特殊的是日期类型:比如要写入 2004-7-20 15:20:07 这个时刻需要如下写:</P>
- H1 ?. y; t- K1 u& k6 l+ E< >UPDATE ... SET ... = TIMESTAMP '2004-7-20 15:20:07' ...</P>
, \" M6 o; S- ?9 F) y5 W< >注意这里使用了 TIMESTAMP 关键字,并使用单引号隔开;另外请注意日期格式,上面的格式是可识别的,Oracle 识别的格式没有 SQL Server 那般多。这是和 SQL Server 不同的地方。</P>
: \0 L7 S% J: W k/ D* R< >顺便提一句:Access 中的日期类型是用井号(#)隔开的,UPDATE ... SET ... = #2004-7-20 15:20:07# ...</P>
3 |' U' c7 K1 k! s< >4。访问 Oracle 过程/函数(1)</P>
; p; `+ }5 N# a< > SQL Server 作程序时经常使用存储过程,Oracle 里也可以使用过程,还可以使用函数。Oracle 的过程似乎是不能有返回值的,有返回值的就是函数了(这点有些像 BASIC,函数/过程区分的很细致。SQL Server 存储过程是可以有返回值的)。</P>
5 S+ s9 b* C/ t- O( _% Y< >.NET 访问 Oracle 过程/函数的方法很类似于 SQL Server,例如:</P>
1 H4 e- t! a4 p7 V% K; T< >OracleParameter[] parameters = {) n7 ~1 j' u" t9 k8 m/ E7 v. J- d
new OracleParameter("ReturnValue", OracleType.Int32, 0, ParameterDirection.ReturnValue, true, 0, 0, "",
" f7 u! U5 O, j: @& G2 } DataRowVersion.Default, Convert.DBNull )
3 i. V/ M0 l, p new OracleParameter("参数1", OracleType.NVarChar, 10),
+ @1 r8 j6 y2 T new OracleParameter("参数2", OracleType.DateTime),2 r) g" o$ \. B# ~# _0 J( m9 N
new OracleParameter("参数3", OracleType.Number, 1)
9 F; a7 X: r R' J1 b& m1 p/ C };3 N( y3 `+ @+ L+ _- q
$ g$ U1 k, U$ Q/ o
parameters[1].Value = "test";
' Z) E9 \ D9 x9 Q/ R& kparameters[2].Value = DateTime.Now;( c. _: P! a, o- j# k8 O6 {
parameters[3].Value = 1; // 也可以是 new OracleNumber(1);</P>. S- F( r# H( N! p" p6 z
<P>OracleConnection connection = new OracleConnection( ConnectionString );6 _) U8 D( g; ?% i; H
OracleCommand command = new OracleCommand("函数/程名", connection);9 e7 r- L3 r7 ?' i- ~0 a1 \
command.CommandType = CommandType.StoredProcedure;+ J6 s; a' G: h4 d5 a6 M
- [' B: b! d3 ~8 `
foreach(OracleParameter parameter in parameters)
5 p4 ?; H8 E' Q$ d& R command.Parameters.Add( parameter );9 \/ }) H# z$ z, {+ G) Q
' k* C% G8 ~, g. d7 E" R) Nconnection.Open();5 N$ ]& L# W [2 |
command.ExecuteNonQuery();
( c! M! M7 j+ `. W9 F9 N# F1 \; kint returnValue = parameters[0].Value; //接收函数返回值
" n3 T1 B# t' C# Kconnection.Close();</P>
" J6 F- t z- ?" d<P> Parameter 的 DbType 设定请参见 System.Data.OracleClient.OracleType 枚举的文档,比如:Oracle 数据库中 Number 类型的参数的值可以用 .NET decimal 或 System.Data.OracleClient.OracleNumber 类型指定; Integer 类型的参数的值可以用 .NET int 或 OracleNumber 类型指定。等等。</P>& t! ^$ V0 H' `3 h3 B
<P> 上面例子中已经看到函数返回值是用名为“ReturnValue”的参数指定的,该参数为 ParameterDirection.ReturnValue 的参数。</P>" {! }% V; T! ]- O$ Q8 K8 k2 c
<P>5。访问 Oracle 过程/函数 (2)</P>
$ @, ]: i6 _* I" M* b0 y# x<P> 不返回记录集(没有 SELECT 输出)的过程/函数,调用起来和 SQL Server 较为类似。但如果想通过过程/函数返回记录集,在 Oracle 中就比较麻烦一些了。</P>
; ` N# H& |- q; E% w) ~! n9 \" g$ ~<P>在 SQL Server 中,如下的存储过程:</P>
6 H: y' v W/ z' ~<P>CREATE PROCEDURE GetCategoryBooks4 u; b! `1 q' j. c: k; S
(
) z& S5 G, D" T2 v @CategoryID int
0 x6 V8 u$ Y" v6 e5 H/ |9 C: n)# }% {$ f `4 f2 k
AS
* _4 V# q0 E" U: Z( b3 oSELECT * FROM Books: ]) [* L+ t, b! C5 r& f
WHERE CategoryID = @CategoryID9 }, s' w! A4 Z- a# I3 [7 ?& k
GO</P>' ~) |1 c2 F2 ]' \
<P> 在 Oracle 中,请按以下步骤操作:</P>
% v ^$ U! E9 }; w. D- x7 {7 Q<P>(1)创建一个包,含有一个游标类型:(一个数据库中只需作一次)</P>
5 v0 o& E+ I+ e7 k( Y! F! ?3 G<P>CREATE OR REPLACE PACKAGE Test: I! T# k' x7 ]$ p3 U+ \& H5 g
AS3 ?5 Q% }2 Z4 F5 o+ m
TYPE Test_CURSOR IS REF CURSOR;' W' h- N6 s2 x0 Z- }; }# K
END Test;</P>0 u/ ~2 j; @3 p
<P>(2)过程:</P>3 p" L! X/ h' y
<P>CREATE OR REPLACE PROCEDURE GetCategoryBooks
$ H! K6 v% z6 u8 O' Y; D s9 _(/ @+ w! @2 S3 i1 i1 D
p_CURSOR out Test.Test_CURSOR, -- 这里是上面包中的类型,输出参数' j% y( E/ E0 v
p_CatogoryID INTEGER
9 {2 c# a: }8 A$ p3 L: S/ ])
) k+ v/ k, b7 q4 pAS9 F# U$ b0 F! O
BEGIN
* _9 ]) `" M& ~( K @ OPEN p_CURSOR FOR
0 o# q: F: K, J( Q SELECT * FROM Books
) t" A0 \2 n- W& V WHERE CategoryID=p_CatogoryID;# X6 i7 S+ S0 R; n$ E& G, ^1 Y) J9 g! |
END GetCategoryBooks;</P>
+ h# B( X+ J! q& B0 \<P>(3).NET 程序中:</P>
# y( a; Y, T T" h( `: W( ~<P>OracleParameters parameters = {
j( l* W6 a% X! \# F6 `7 W new OracleParameter("p_CURSOR", OracleType.CURSOR, 2000, ParameterDirection.Output, true, 0, 0, "",
/ F. r$ s% L, n- ~ Y/ Q" T y DataRowVersion.Default, Convert.DBNull),1 C" Y' a6 B9 \# p3 Q, t+ x- J m5 G
new OracleParameter("p_CatogoryID", OracleType.Int32)
, a- P' y: x# a( i- C};7 K8 P$ ]3 L. T! P- P
! J9 e" B: q6 h1 u# C7 {parameters[1].Value = 22;( T% ?% I0 t8 V; [7 n1 s$ O% I6 R
" m# u: }- }, P( k
OracleConnection connection = new OracleConnection( ConnectionString );
2 k& L) M! @7 o/ ~OracleCommand command = new OracleCommand("GetCategoryBooks", connection);; S' k. ~. m3 |3 Q5 z+ B' Q- E% Q
command.CommandType = CommandType.StoredProcedure;
# B- A6 L9 J& Q% e9 P- g: ]
9 w6 X, k& g$ {( w9 Tforeach(OracleParameter parameter in parameters)
: Z. E$ G$ \% _& S2 N- _8 O5 X command.Parameters.Add( parameter );
! L5 _! y0 C( Z# x) U% t9 ~0 f' K' {9 A* P" c$ e9 m
connection.Open();% ]! [- J' t/ b" y6 y* F, |6 D
OracleDataReader dr = command.ExecuteReader();/ l& [& ]1 f! ]( L2 i* b, B* m
$ b, Q* n" G t, w8 I
while(dr.Read())9 m' L# M S9 Z) [
{
6 j! r' D; h" D. h9 G; r. \# { // 你的具体操作。这个就不需要我教吧?0 p! ~1 N' W/ n Y8 D J
}( y! \! Z5 q2 c* d( T( x
connection.Close();</P>
: y* Z, V! p3 Z" v1 J<P> 另外有一点需要指出的是,如果使用 DataReader 取得了一个记录集,那么在 DataReader 关闭之前,程序无法访问输出参数和返回值的数据。</P>
! Q! ], c6 ^1 m) X) W4 Y' l: e<P> 好了,先这些,总之 .NET 访问 Oracle 还是有很多地方和 SQL Server 不同的,慢慢学习了。</P>9 t: d+ O( h# b. f
<P> |
zan
|