简述:C#中的库(libraries) M{xVkXc>
言归正传,我现在要说的是库(libraries),和大家一起学习如何用C#建立一个DLL文件.说起DLL,肯定是无人不知,无人不晓,这个WINDOWS的典型代表,同时也经常是大家功击的对象.呵呵,不管怎么样,学还是要学的.我们下面就开始,如何用命令行方式将一个C#程序编译成DLL,和如何在客户端使用他. ahtYSz_FM
这个例子包括两个文件,一个是Factorial.cs,作用是计算一个数字的阶乘.还有一个是DigitCounter.cs,作用是计算传过来的字符串参数中的数字的数目. ,<3uc
我们可以这样来建立库,在命令行方式下这样做: _IL2-c8
csc /target:library /out:Functions.dll Factorial.cs DigitCounter.cs p08kZ
下面讲一下各个参数的用法: ^%8qKC`Tt
/target:library:向系统指出输出的是一个DLL库,而不是一个EXE的可执行文件. y-#
/out:Functions.dll:指定输出的DLL的文件名,即Functions.dll,一般地,如果你省略了第一个参数,那么默认的文件名将是第一个文件的文件名,即Factorial.dll. "XNu-_$N<a
下面我们再来建立一个文件,即使用这个库的文件,叫客户端文件,FunctionClient.cs.建立好后,用下面的语名编译: =#(0)p$EC
csc /out:FunctionTest.exe /R:Functions.DLL FunctionClient.cs i7nL_N
下面说一下这个编译语句的用法: ole|J
/out:FunctionTest.exe:指出输出的文件名是FunctionTest.exe y?#9>S >:\
/R:Functions.DLL:指出要引用的库,如果不是在当前目录下,必须要指出其的完整路径.
Znta#G0
下面我就把这几个文件的代码写在下面: ^IGyuj0]jG
000: // Libraries\Factorial.cs %X9b=%'+
001: using System; \V^*44+
<!
002: jJVT_8J
003: namespace Functions &$c5~9p\B
004: { i<m$#6<Z
005: public class Factorial h}|6VJ@.
006: { 1s`)yu^`v
007: public static int Calc(int i) U,<]J*b(@4
008: { C]'g:93L
009: return((i <= 1) ? 1 : (i * Calc(i-1))); "#pzZ)Zh
010: } >+
]R4
011: } S= -M3fP~
012: } V5a?=vK9
这是Factorial.cs这个文件的代码.在003行中,namespace的意思是名字空间,据M$的介绍,库必须根据它的名字空间打包,以使.NET能够正确地载入你的类. sS2_-X[_
下面是DigitCounter.cs这个文件的内容: uuSR%KK]|
000: // Libraries\DigitCounter.cs 1OJ*wI*
001: using System; |mxNUo-
002: 3Q"F(uE v^
003: namespace Functions .G}k/`a
004: { w<65S
005: public class DigitCount PW%1xHLfk
006: { b,s Gq
007: public static int NumberOfDigits(string theString) wmo{YS3t|
008: { yGvDn' m
009: int count = 0; W|dpFh`
010: for ( int i = 0; i < theString.Length; i++ ) qO-C%p
[5
011: { 94|yvh.B
012: if ( Char.IsDigit(theString) ) PK6*}y
013: { ZBX
014: count++; '@TI48 J+
015: } 9?;@*x
016: } 5VR.o!h3I
017: F aFp_P?
018: return count; ~uI**{
019: } {'h_'Y`bOQ
020: } yGiP[d|tRc
021: } W]]q=c%2
注意,这个例子中的namespace应与第一个的一致,因为它们是同一个库中的.NumberOfDigits方法计算了参数中的数字的个数. g5#CN:%f
第三个文件是FunctionClient.cs Gg%tVQu
我们知道,一个库一旦建立,就可以被别的类利用(废话,要不然怎么叫库呢?).下面的C#程序就利用了我们刚才建立的库中的类. fcRj
000: // Libraries\FunctionClient.cs p jKt:R}
001: using System; X>8-`p
002: using Functions; M$Fth*q{GD
003: class FunctionClient MO[kr2T
004: { $!G` D=
005: public static void Main(string[] args) X JY5@I.
006: { r6`\d k
007: Console.WriteLine("Function Client"); m0A# 6=<
008: i&`!|X-=R
009: if ( args.Length == 0 ) fVe@YqNa
010: { AnNPTi
011: Console.WriteLine("Usage: FunctionTest ... "); Y4#y34We
012: return; &<au/^F
013: } _(C^[ :s
014: QDS0ejhp
015: for ( int i = 0; i < args.Length; i++ ) g nt45]@{
016: { L[9OVD
017: int num = Int32.Parse(args); iTh
xVD
018: Console.WriteLine( H]s4% 9T
019: "The Digit Count for String [{0}] is [{1}]", #?9Q{0e
020: args, <uZPqi||
021: DigitCount.NumberOfDigits(args)); S%kS#U${|
022: Console.WriteLine( McjS)4j&.
023: "The Factorial for [{0}] is [{1}]", &p5&=zV}
024: num, {j?7d; 'j
025: Factorial.Calc(num) ); RqXi1<6j#
026: } ]pnYvXf>!
027: } v~"Ef_`
028: } k6@b|
在002行中,一个using Functions指明了引用Functions.DLL这个类. J58#$NC
`'
如果我们在命令行中键入如下命令,就可以看到输出: 1otspOy
FunctionTest 3 5 10 =7 VCtd/
输出: :NuR>~
Function Client c/
_yMN
The Digit Count for String [3] is [1] -vV'Lw(
The Factorial for [3] is [6] 3DW3LYo{
The Digit Count for String [5] is [1] BCx!0v?9
The Factorial for [5] is [120] `<^*jB@P
The Digit Count for String [10] is [2] u_.HPA
The Factorial for [10] is [3628800] ]:&n-&@L
注意:当你运行这个.EXE文件时,它引用的DLL文件可以是在当前目录,子目录,或是CORPATH这个环境变量.CORPATH这个环境变量是在.NET环境中的类路径,用来指引系统寻找类.说白了,就是JAVA中的CLASSPATH,明白了吧,呵呵. ^'vIOq-1v
好了,又完了一篇,今天的任务完成了,可以休息了.唉,有什么好玩的呢? B7HQR{t
C#中的版本处理 >uTPjR[
现在我要说的是C#中的版本处理.其实这是任何一个软件必须要考虑的问题.每个软件都不只一个版本(除了我写的以外),因此版本处理显得非常地重要.JAVA很好地处理了这个问题,而我个人认为C#借鉴了JAVA的处理方法,所以,也做得很好. [Tb\woU
在C#中,如果你在声明一个方法的时候用了virtual这个关键字,那么,在派生类中,你就可以使用override或者new关键字来弃用它或是忽略它.如果你在父类中用了virtual这个关键字,而在其派生类中又没有用override或new关键字,而直接引用一个同名方法的话,编译器将会报错,并将以new方式,即忽略派生类中的方法的方式来运行.下面的例子可以帮助你来理解: 3 jF|Ic
000: // Versioning\versioning.cs -#aZF2z
001: public class MyBase 'M8aW!~
002: { Wr5 Q5s)c
003: public virtual string Meth1() EJLQ&oH[
004: { vU!8`x)
005: return "MyBase-Meth1"; :.$"kXm^
006: } ?;
[ T
007: public virtual string Meth2() 5`~mqqR5
008: { IaLMWoh
009: return "MyBase-Meth2"; V&i2L.{G)
010: } .+yW%~0
011: public virtual string Meth3() j0FW8!!-g
012: { R&#tSL
013: return "MyBase-Meth3"; 7^MX l
014: } d+6]u_J
015: } ;i\C]*
016: F$Q04Qw
017: class MyDerived : MyBase 5Z{_m;I.
018: { 4T`&Sl
019: public override string Meth1() }c%
pH{HI
020: { KiAcA]0
021: return "MyDerived-Meth1"; O8lFx_N7Q
022: } )iU^&@[S
023: public new string Meth2() FXahZW~Ol
024: { Uoji@
025: return "MyDerived-Meth2"; s<vs:jna
026: } t`5j4bdG
027: public string Meth3() // 系统在这里将会有一个警告,并且将会隐藏方法Meth3() vXdZmYrC
028: A59gIp*>
029: 9t K>gwb
030: { KE.Dt
031: return "MyDerived-Meth3"; NZk&JND
032: } ]JjK#eh
033: :.uk$jx
034: public static void Main() J02^i5l
035: { Es.nHN^]%K
036: MyDerived mD = new MyDerived(); 1fFj:p./l_
037: MyBase mB = (MyBase) mD; J}TfRrf
038: y+U83a[L*
039: System.Console.WriteLine(mB.Meth1()); q[d)e6
040: System.Console.WriteLine(mB.Meth2()); y-9+a7j
041: System.Console.WriteLine(mB.Meth3()); PKf:O
042: } exDkq0u]
043: } Hi7y(h?wj
输出: 81F,Y)x.
MyDerived-Meth1 dz%EM8
MyBase-Meth2 oNM?y:O
MyBase-Meth3 }`o?/!X
可以很明显地看出来,后两个new关键字的输出是父类中的方法的输出,所以可以看出,new这个关键字的作用是如果在以前的版本中有这个方法,就沿用以前的方法,而不用我现在方法内容.而virtual的方法的作用正好相反,它的作用是如果在父类中有这样一个方法,则用我现在写的方法内容,让以前的滚蛋!不过,这里用new好象不太好,让人误解(糟了,盖痴又要打我了!&*%$#@). y=a V=qD
如果你把第037行去掉,把039-041中的mB全部改为mD,输出又变为: K2rzhHfb
MyDerived-Meth1 T8XY fcc*h
MyDerived-Meth2 +E5=$`
MyDerived-Meth3 pSfYu=#f
这又说明了什么呢,说明了派生类的对象只有在被父类重塑的时候,override和new关键字才会生效.呵呵,这样说的确有点难以理解,大家只有自己动手,才能搞清楚这其中的机关,所谓"实践是检验C#的唯一标准",哈哈! f:woP7FP
在C#中,你可以自由地为在派生类中为加入一个方法,或者覆盖父类的方法,如下所示,非常地简单: S1bAu
<
class Base {} *Zbuq8>
class Derived: Base s0C:m
{ kl}Xmw{tJ
public void F() {} _xrwu;o0}
} ,9of(T(~
和: rzeLx Wt
class Base /ty?<24ko
{ B,vOsa"x6`
public void F() {} :%X Ls,
} }Qr6l/2
class Derived: Base x83a!9
{ )oU)}asY
public void F() {} 2.lgT|p
} 5`-UMz<]
好了,这一节又完了,明天见! PaO-J&<
C#中的结构(struct) qlsQ|/'D
我要说的是C#中的,注意,我在这里说的结构不是指的C#的语言结构.这里所说的是一种与类(class)相对的东西,下面我就与类相对比,来说一说这个struct. O1P=#l iYX
下面的这个例子讲述了如何建立一个具有属性,方法和一个字段的结构.并讲述如何使用他. qOy=O
[+9
000: // Structs\struct1.cs L}%dCe
001: using System; s B
20/F
002: struct SimpleStruct mdbp8,O
003: { +?m0Q;%b
004: private int xval; ]lBGyUJn
005: public int X g(hOg~S\E
006: { '#\1uXM1U?
007: get { h<6UC%'ac
008: return xval; 2/7_;_#vJ%
009: } TgfrI
010: set { \Kavw
011: if (value < 100) $uh z
012: xval = value; OCV+h'
013: } ~i~%~doa
014: } @jy41eIo
015: public void DisplayX() m:+8J,jW
016: { gfa[4
z
017: Console.WriteLine("The stored value is: {0}", xval); Q2|p\rO
018: } _\8qwDg"#e
019: } aP-<4uGx
020: S*
R,FKg
021: class TestClass 7 sFz?`-
022: { y$W|~ H
023: public static void Main() G"dS+,Q
024: { J
CGC
025: SimpleStruct ss = new SimpleStruct(); Y&.UIosWb
026: ss.X = 5; {b)~V3rsY
027: ss.DisplayX(); )2e#HBnH
028: } qu|i;WZE
029: } ,h]o>
这个例子的输出是: 'UU\4M
The stored value is: 5 <skajQQ
从上面的例子中我们可以看到结构和类似乎是一样的.的确,如果你用类去重亲写这个程序,结果是一样的.但是,很明显,两个一样的东西是不可能一起出现的. 结构(struct)是值(value)型的,而类是参考型的.这样,你就可以用结构建立像内建类型那样的对象了. Vw{*P2v)
还有就是如果你用一个new关键字建立一个类的实例的时候,它是以堆(heap)来分配的,而用new来建立一个结构的的实例的时候,它是以栈(stack)来分配.这会给我们提高很多性能(M$说的).好了,让我们再来看下面的例子吧: g);^NAA
000: // Structs\struct2.cs hJ;$A*Y
001: using System; B 0ee?VC
002: Wp0
Dq(
003: class TheClass ]wVk+%e
004: { YT#3n
005: public int x; ]lO h&Cz[
006: } /+]s.V.
007: s
+s" MI
008: struct TheStruct C.Uju`3
009: { NH A 5e<
010: public int x; b1#dz]
011: } e [h8}F
012: UUe#{6Jx_
013: class TestClass eU@Cr7@,|
014: { iq$$+y,
015: public static void structtaker(TheStruct s) ,m3e?j@;r
016: { -~{c
u47_
017: s.x = 5; K2)!h.W
018: } iBg3mc@OO
019: public static void classtaker(TheClass c) uQ1@b-e`5
020: { o{:xp r=(
021: c.x = 5; |*5 =_vF
022: } OhZgcUqQ8
023: public static void Main() u+m,b76
024: { NpP')m!`}
025: TheStruct a = new TheStruct(); <UP
m=Hb
026: TheClass b = new TheClass(); 7,
}
$u
027: a.x = 1; 8IQtz2
028: b.x = 1; feM6K!fL`
029: structtaker(a); ZP\M9Ja
030: classtaker(b); bm~W
EX
031: Console.WriteLine("a.x = {0}", a.x); C4$:mJ>y
032: Console.WriteLine("b.x = {0}", b.x); Sl2iz?
033: }
-fI`3#
034: } 7cDU2l
这个例子的输出是: {7hLsK[])
a.x = 1b.x = 5 BaI $S>/Q
从这个例子例子可以看出,当一个结构被传递到一个方法时,被传递的只不过是一个副本,而一个类被传递时,被传递的是一个参考.所以a.x=输出的是1,不变,而b.x却变了. y&~w2{a
还有的区别就是结构可以不用new来实例化,而类却要.如果你不用new来实例化一个结构,那么所有的字段将仍然处于未分配状态,直到所有的字段被初始化.和类一样,结构可以执行接口.更重要的是,结构没有继承性,一个结构不能从别的类继承,也不能是别的类的基类. ;*TIM%6#
例三: S[3iA~)Z-
interface IImage XN=67f$Hw
{ ,_.I\EY[
void Paint(); *iO u'
} en S}A*Io
struct Picture : IImage s8"8y`u
{ {P%9
public void Paint() yF}OfK?0f
{ ))kF<A_MK
// painting code goes here
zG }?
} f"G-
private int x, y, z; // other struct members z;f2*F
} 8`>h}Q$
好了,关于结构我就讲到这了,以后还会讲到的. 5zJj]A
C#中的ADO数据库访问 ^FmU_Q0
这一节我要讲的是大家非常关心的,肯定也是非常感兴趣的部分.嘿嘿,也是我写教程最拿手的部分----ADO数据库访问.想到这,我就想起我在去年写的"访问数据库"系列文章,嘿嘿!所以呢,如果你觉得对记录集之类的东西比较难理解的话,我推荐你先看一看我的那几篇文章.好了,让我们开始吧! >eQr<-8
什么是ADO(ActiveX Data Objects译作ActiveX数据对象),ADO是一个非常简单的思想,一种让你仅用一种方式去访问数据的思想.ADO不算一个新的思想,仅是采用现有的数据访问技术,将其融合起来.如果你不理解ADO,想一想ODBC吧!其实我们在搞ASP的时候,就用到了ADO,还记得吗,那个曾经被我们用过无数次的set conn=Server.CreateObject("ADODB.Connection")吗?是的,就是它.至于ADO的一些概念性的东西,请大家自行查阅资料,不过,其实不了解也没有关系,只把它想象成一个M$给我们的访问数据的工具吧! 5^<h}u9
OK,下面的例子是以一个M$ ACCESS 2000的数据库为基础的,它的结构如下,表名是Categories,文件名是BugTypes.mdb ,赶快建一个吧: \uqjs+
Category ID Category Name tsOrt3
1 Bugbash stuff MB^~%uZ2K
2 Appweek Bugs 1J=.N|(@Q
3 .NET Reports aimarU
4 Internal support qU2~fNY
好的,我先把所有的程序都写出来,然后我们来一句一句的品尝: k %e^kej
000: // ADO\adosample.cs {R<Ea
@LV+
001: using System; >zsid:
002: using System.Data;
/-_=nf}w
003: using System.Data.ADO; (
9!k#
004: H`bSYjgM!
005: public class MainClass K%<j=c
006: { g6@Fp7T
007: public static void Main () l:0s2
008: { [v7^i_d
009: // 设定好连接字符串和选择命令字符串010: string strAccessConn = "Provider=Microsoft.Jet.OLEDB.4.0;Data Source=BugTypes.MDB"; $E<Esf$
011: string strAccessSelect = "SELECT * FROM Categories"; fqX"Lus `=
012: y.5/?{GL
013: //建立记录集,并把Categories这个表填进去 }VS3L_
;}/
014: DataSet myDataSet = new DataSet(); oF9
-&
015: myDataSet.Tables.Add("Categories"); ]w T 7*( Y
016: S:4crI
017: //建立ADO实例018: ADOConnection myAccessConn = new ADOConnection(strAccessConn); WG*t::NN
019: ADODataSetCommand myAccessDataSetCmd = new ADODataSetCommand(); >^q7c8]~g
020: myAccessDataSetCmd.SelectCommand = new ADOCommand(strAccessSelect,myAccessConn); XZ&KR.C,
021: [${
QzO
022: myAccessConn.Open(); MObt,[^W
023: try Nk=JBIsKv
024: { ]V %.I_
025: myAccessDataSetCmd.FillDataSet(myDataSet,"Categories"); D0k
8^
026: } e0@6Pd
027: finally H1<>NWm!v7
028: { 3~,d+P
029: myAccessConn.Close(); 9CA^B2u
030: } LTNj| u
031: 0q>P~]Ow
032: try 0&qr
033: { bwVPtu`
034: // 一个记录集可以包含多个表,我们把它们放到一个数组中035: DataTable[] dta = myDataSet.Tables.All; yKYUsp
036: foreach (DataTable dt in dta) 5>3}_
037: { d(vsE%/!
038: Console.WriteLine("Found data table {0}", dt.TableName); 5w %_$x
039: } =U8a ?0
040: {Q+gZcu
041: //下面的两行程序展示了两种从一个记录集中得到这个数据集中的表格数的方法 swA+f
042: Console.WriteLine("{0} tables in data set", myDataSet.Tables.Count); Hsih[f
043: Console.WriteLine("{0} tables in data set", dta.Length); [A}rbD K
044: //下面的几行程序说明了如何从一个记录集中依靠表格的名称来取得信息 }X x(^Zh
045: Console.WriteLine("{0} rows in Categories table", myDataSet.Tables["Categories"].Rows.Count); A(?\>X
9g
046: //列的信息是自动从数据库中得到的,所以我们可以用以下的代码047: Console.WriteLine("{0} columns in Categories table", myDataSet.Tables["Categories"].Columns.Count); X37 L\e[c
048: DataColumn[] drc = myDataSet.Tables["Categories"].Columns.All; ]\/tVn.'
049: int i = 0; jV.g}F+1m
050: foreach (DataColumn dc in drc) 4}_O`Uxh
051: { Gl1jxxd
052: //打印出列的下标和列的名称和数据类型053: Console.WriteLine("Column name[{0}] is {1}, of type {2}",i++ , dc.ColumnName, dc.DataType); ,Jc m+Wb
054: } ^w ] /
055: DataRow[] dra = myDataSet.Tables["Categories"].Rows.All; lb'GXd %
056: foreach (DataRow dr in dra) vN2u34
057: { d(g^M1m
058: //打印出CategoryID和CategoryName059: Console.WriteLine("CategoryName[{0}] is {1}", dr[0], dr[1]); F+ E|r6'i
060: } uOa26kE4
061: } C6O8RHg
062: catch (Exception e) ??n*2s@t
063: { K;LZ-
064: Console.WriteLine("Oooops. Caught an exception:\n{0}", e.Message); N`)$[&NG]
065: } b-3*Nl _%
066: } TKk-;Y=N
067: } qwIa?!8o
看起来,这个例子是有一些复杂的,只怪我例子选的不好,呵呵.不过,细细分析一下,还是可以理解的.我现在说一下这个例子中几个特别的东东.第一就是不象在ASP中,一个命令字符串被须被当作一个命令对象. [((;+B
020做的正是这个事情.注意,在015行有一个myDataSet.Tables.Add("Categories")语句,这并不是把数据库中的Categories这个表填进去,只不过是建一个空表,而 wApMzZ(X2y
025才是真的填充数据. !61Pl/uQ
这个例子的输出是: !LkWzn3
Found data table Categories PW3GL3+
1 tables in data set |_omr&[_
1 tables in data set D;UV&.$'v
4 rows in Categories table S1D@vnZ3O\
2 columns in Categories table 8q1wHZ
Column name[0] is CategoryID, of type Int32 Vi4~`;|&b+
Column name[1] is CategoryName, of type System.String SP|<Tny
CategoryName[1] is Bugbash stuff hFiIW77s2
CategoryName[2] is Appweek Bugs piU/&
CategoryName[3] is .NET Reports c/_+o;Bc
CategoryName[4] is Internal support