QQ登录

只需要一步,快速开始

 注册地址  找回密码
查看: 3435|回复: 1
打印 上一主题 下一主题

通过.NET访问 Oracle数据库

[复制链接]
字体大小: 正常 放大
韩冰        

823

主题

3

听众

4048

积分

我的地盘我做主

该用户从未签到

发帖功臣 元老勋章

跳转到指定楼层
1#
发表于 2004-10-3 21:16 |只看该作者 |倒序浏览
|招呼Ta 关注Ta
长期以来,我一直用的是 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=clsid27CDB6E-AE6D-11cf-96B8-444553540000&gt;200309/guangli_320.swf"&gt;200309/guangli_320.swf"&gt; 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"&gt;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
转播转播0 分享淘帖0 分享分享0 收藏收藏0 支持支持0 反对反对0 微信微信
ilikenba 实名认证       

1万

主题

49

听众

2万

积分

  • TA的每日心情
    奋斗
    2024-6-23 05:14
  • 签到天数: 1043 天

    [LV.10]以坛为家III

    社区QQ达人 新人进步奖 优秀斑竹奖 发帖功臣

    群组万里江山

    群组sas讨论小组

    群组长盛证券理财有限公司

    群组C 语言讨论组

    群组Matlab讨论组

    回复

    使用道具 举报

    您需要登录后才可以回帖 登录 | 注册地址

    qq
    收缩
    • 电话咨询

    • 04714969085
    fastpost

    关于我们| 联系我们| 诚征英才| 对外合作| 产品服务| QQ

    手机版|Archiver| |繁體中文 手机客户端  

    蒙公网安备 15010502000194号

    Powered by Discuz! X2.5   © 2001-2013 数学建模网-数学中国 ( 蒙ICP备14002410号-3 蒙BBS备-0002号 )     论坛法律顾问:王兆丰

    GMT+8, 2026-6-11 14:44 , Processed in 0.494524 second(s), 57 queries .

    回顶部