Spring.NET 与 NHibernate
回到 Spring.NET & NHibernate of C#.NET 技术论坛
实战C#.NET编程----Spring.NET & NHibernate从入门到精通
您可以从以下网址下载最新版:http://tech.bokeecn.com 本文中的源代码下载地址:http://tech.bokeecn.com
作 者:李绿华 (William Lee) 制作时间:2006-11-30 版 本 号:V 0.50 |
版权声明:未经作者本人许可,任何公司、团体、个人都不得以任何方式复制或抄袭本文内容的部分或全部。
转载本文请通过以下方式联系作者获得许可。
邮件:it1630@163.com
目 录
实战C#.NET编程----Spring.NET & NHibernate从入门到精通-- 1
第一章 Visual C# .NET 入门指南-- 3
一、 C# 简介-- 3
二、 使用Visual Studio 开始C# 项目-- 3
步骤 1. 开始项目-- 4
步骤 2. Hello, World!- 5
步骤 3. 程序结构-- 7
步骤 4. 控制台输入-- 8
步骤 5. 使用数组-- 9
步骤 6. 文件输入/输出-- 10
步骤 7. 创建函数-- 13
步骤 8. 使用调试器-- 15
小结-- 17
第二章 面向对像ORM- 18
一、 什么是ORM- 18
二、 为什么需要ORM- 19
三、 流行的ORM框架简介-- 19
第三章 Spring.NET入门-- 20
一、 Spring.NET概览-- 20
二、 第一个Spring.NET的程序-- 22
第四章 NHibernate入门-- 25
一、 什么是Nhibernate- 25
二、 Nhibernate概述-- 25
三、 第一个NHibernate 程序-- 28
第五章 Spring.NET 与 NHibernate 的整合-- 34
一、 建立新的项目(SpringNHibernateSample)- 35
二、 添加NHibernate程序-- 35
三、 添加Spring.NET的程序-- 35
四、 添加Spring.NET为NHibernate的容器配置-- 38
五、 编写测试程序代码-- 41
六、 测试并查看结果-- 43
第六章 深入Spring.NET 与 NHibernate开发-- 43
第七章 项目实战----办公自动化系统-- 43
第八章 结束语-- 43
第一章 Visual C# .NET 入门指南
一、 C# 简介
Visual C# .NET 是 Visual Studio 系列中的最新成员。这种新语言基于 C/C++,但它深化了更容易地使用面向组件编程的发展方向。C/C++ 程序员应该非常熟悉它的语法。
下面的示例应用程序示范了如何构建一个简单的实现 QuickSort 算法的 C# 项目。它包括了 C# 程序的基本组成部分:读/写控制台和文件、创建函数和使用基本数组。
这些入门指南并不打算涵盖该编程语言的所有方面。它们只是您探索这种语言的一个起点。我们鼓励您按照本教程的说明执行,因为它包括了 QuickSort 应用程序的各个不同部分。您还可以获得完整的源代码和项目文件。
建议的要求
编译此示例应用程序需要 Visual Studio.NET 2003/2005。关于 C/C++ 的知识是有帮助的但不是必需的。
二、 使用Visual Studio 开始C# 项目
Visual C# .NET 入门指南通过实现一个简单的 QuickSort 算法,带您领略如何构建 Visual C# .NET 项目。
本节将按以下的步骤让大家一步一步了解Visual C#:
n 步骤 1. 开始项目
n 步骤 2. Hello, World!
n 步骤 3. 程序结构
n 步骤 4. 控制台输入
n 步骤 5. 使用数组
n 步骤 6. 文件输入/输出
n 步骤 7. 创建函数
n 步骤 8. 使用调试器
n 小结
你可以下载 Quicksort_Visual_CSharp_.NET.exe 。里面包含了下面的代码。
步骤 1. 开始项目
Visual Studio 中的开发工作以解决方案的形式进行组织,每个解决方案包含一个或多个项目。在本教程中,我们创建的解决方案包含一个 C# 项目。
n 创建一个新项目
在 Visual Studio .NET 环境中,从菜单中选择 File | New | Project。
在左侧选择 Visual C#Projects,然后在右侧选择 Console Application。
指定项目的名称,然后输入创建项目的位置。Visual Studio 会自动创建项目目录。
单击 OK,那么现在就正式开始了
n Visual C# 解决方案
Visual Studio.NET 已经创建了含有一个简单 Visual C# 项目的解决方案。该项目包含两个文件:assemblyinfo.cs 和 class1.cs。
接下来的几步骤将讨论这些不同的文件以及如何编译该项目。
步骤 2. Hello, World!
很遗憾,但我们仍然无法抵御这种诱惑……我们还是不得不完成一个基于 C# 的经典"Hello, World!"应用程序,这个应用程序最初是用 C 语言编写的。
n 修改源代码
在 Solution Explorer 中双击文件"class1.cs"。可以通过"View"菜单来显示 Solution Explorer。
更改预生成的模板 (class1.cs),如下面以斜体突出显示的 代码所示。
using System; namespace quicksort { /// /// Summary description for Class1. /// class Class1 { static void Main(string[] args) { // // TODO: Add code to start application here // Console.WriteLine ("Hello, C#.NET World!"); } } } |
注意,当您键入代码时,Visual Studio 将为您提示类和函数的名称(因为 .NET 框架发布了这种类型信息)。
n 编译应用程序
既然您已经完成了修改,就可以通过在 Build 菜单中简单地选择 Build 来编译 Visual C# 项目。
来自 C# 编译器的错误和消息会在 Output 窗口中显示。如果没有错误,则可以通过单击 Debug 菜单下的 Start without Debugging 来运行 Hello World 应用程序。
n 程序输出
在 Visual C# 中运行 Hello World 示例应用程序时,输出结果的屏幕截图如下:
n 理解更改
System.Console 类的 WriteLine() 函数打印传递给它的字符串,其后紧跟一行新的字符。此函数可以接受许多其他数据类型(包括整型和浮点型)的参数。
在程序加载完成后,控制就传递给 Main() 函数。这就是我们在该过程中插入对 WriteLine() 调用的原因。
步骤 3. 程序结构
既然我们已经构建了一个简单的 Hello World 应用程序,那么就让我们停下来分析一下 Visual C# 应用程序的基本组成部分。
n 源代码注释
字符 “//” 将行的剩余部分标记为一个注释,这样 C# 编译器就会忽略它。另外,/* 和 */ 之间的代码也会被当作注释。
// This line is ignored by the compiler. /* This block of text is also ignored by the Visual C# compiler. */ |
n Using 指令
.NET 框架为开发人员提供了许多有用的类。例如,Console 类处理对控制台窗口的输入和输出。这些类是按照层次树的形式组织的。Console 类的完全限定名实际上是 System.Console。其他的类包括 System.IO.FileStream 和 System.Collections.Queue。
using 指令允许您在不使用完全限定名的情况下引用命名空间中的类。以斜体突出显示的代码应用了 using 指令。
using System; class Class1 { static void Main(string[] args) { System.Console.WriteLine ("Hello, C#.NET World!"); Console.WriteLine ("Hello, C#.NET World!"); } } |
n 类声明
与 C++ 或 Visual Basic 不同,Visual C# 中的所有函数都必须封装在一个类中。class 语句声明一个新的 C# 类。就 Hello World 应用程序来说,Class1 类包含一个函数,即 Main() 函数。如果用一个 namespace 块将类的定义括起来,就可以把类组织为诸如 MsdnAA.QuickSortApp 这样的层次。
在本入门指南中,我们并不打算深入地介绍类,但是我们将为您简要概述为什么类是我们的示例应用程序的一部分。
n Main() 函数
在应用程序加载到内存之后,Main() 函数就会接收控制,因此,应该将应用程序启动代码放在此函数中。传递给程序的命令行参数存储在 args 字符串数组中。
步骤 4. 控制台输入
现在,我们将继续编写 QuickSort 应用程序。我们需要做的第一件事就是提示用户提供输入和输出文件。
n 修改源代码
更改 C# 源文件 (class1.cs),如下面以斜体突出显示的代码所示。其他的差异(如类名)可忽略不计。
// Import namespaces using System; // Declare namespace namespace MsdnAA { // Declare application class class QuickSortApp { // Application initialization static void Main (string[] szArgs) { // Describe program function Console.WriteLine ("QuickSort C#.NET Sample Application\n"); // Prompt user for filenames Console.Write ("Source: "); string szSrcFile = Console.ReadLine (); Console.Write ("Output: "); string szDestFile = Console.ReadLine (); } } } |
n 从控制台进行读取
Console 类的 ReadLine() 方法提示用户输入,并返回输入的字符串。它会自动地为字符串处理内存分配,由于使用了 .NET 垃圾回收器,您不需要做任何释放内存的工作。
n 程序输出
从菜单中选择 Debug | Start Without Debugging 来运行程序。这是到此为止来自 QuickSort 应用程序的输出的屏幕截图。
步骤 5. 使用数组
在对从输入读取的行进行排序之前,程序需要将其存储到一个数组中。我们将简要讨论可实现对象数组的 .NET 基类的用法。
n 修改源代码
更改 C# 源文件 (class1.cs),如下面以斜体突出显示的代码所示。其他的差异(如类名)可忽略不计。
// Import namespaces using System; using System.Collections; // Declare namespace namespace MsdnAA { // Declare application class class QuickSortApp { // Application initialization static void Main (string[] szArgs) { // Describe program function Console.WriteLine ("QuickSort C#.NET Sample Application\n"); // Prompt user for filenames Console.Write ("Source: "); string szSrcFile = Console.ReadLine (); Console.Write ("Output: "); string szDestFile = Console.ReadLine (); // TODO: Read contents of source file ArrayList szContents = new ArrayList (); } } } |
n 使用 ArrayList 类
我们将导入 System.Collections 命名空间,这样我们就可以直接引用 ArrayList。此类实现大小可动态调整的对象数组。要插入新的元素,可以简单地将对象传递到 ArrayList 类的 Add() 方法。新的数组元素将引用原始的对象,而垃圾回收器将处理它的释放。
string szElement = "insert-me"; ArrayList szArray = new ArrayList (); szArray.Add (szElement); |
要检索现有的元素,请将所需元素的索引传递给 Item() 方法。另外,作为一种简写形式,还可以使用方括号 operator [],它实际上映射到 Item() 方法。
Console.WriteLine (szArray[2]); Console.WriteLine (szArray.Item (2)); |
ArrayList 类中还有许多其他方法,但是插入和检索都是我们需要在此示例中使用的。请查阅 MSDN 库以获得完整的参考指南。
步骤 6. 文件输入/输出
现在,让我们来实现读取输入文件和写入输出文件。我们将每一行读取到一个字符串数组中,然后输出该字符串数组。在下一步中,我们将使用 QuickSort 算法来对该数组进行排序。
n 修改源代码
更改 C# 源文件 (class1.cs),如下面以斜体突出显示的代码所示。其他的差异(如类名)可忽略不计。
// Import namespaces using System; using System.Collections; using System.IO; // Declare namespace namespace MsdnAA { // Declare application class class QuickSortApp { // Application initialization static void Main (string[] szArgs) { ... ... ... // Read contents of source file string szSrcLine; ArrayList szContents = new ArrayList (); FileStream fsInput = new FileStream (szSrcFile, FileMode.Open, FileAccess.Read); StreamReader srInput = new StreamReader (fsInput); while ((szSrcLine = srInput.ReadLine ()) != null) { // Append to array szContents.Add (szSrcLine); } srInput.Close (); fsInput.Close (); // TODO: Pass to QuickSort function // Write sorted lines FileStream fsOutput = new FileStream (szDestFile, FileMode.Create, FileAccess.Write); StreamWriter srOutput = new StreamWriter (fsOutput); for (int nIndex = 0; nIndex < szContents.Count; nIndex++) { // Write line to output file srOutput.WriteLine (szContents[nIndex]); } srOutput.Close (); fsOutput.Close (); // Report program success Console.WriteLine ("\nThe sorted lines have been written.\n\n"); } } } |
n 从源文件进行读取
使用 FileStream 类打开源文件,然后加入 StreamReader 类,这样我们就可以使用它的 ReadLine() 方法了。现在,我们调用 ReadLine() 方法,直到它返回 null,这表示到达文件结尾。在循环过程中,我们将读取的行存储到字符串数组中,然后关闭这两个对象。
n 写入输出文件
假设已经用 QuickSort 对字符串数组进行了排序,接下来要做的事情就是输出数组的内容。按照同样的方式,我们将 StreamWriter 对象附加到 FileStream 对象上。这使得我们可以使用 WriteLine() 方法,该方法能够很方便地模仿 Console 类的行为。一旦遍历了数组,我们便可以象前面一样关闭这两个对象。
步骤 7. 创建函数
最后一步就是创建一个函数来在字符串数组中运行 QuickSort。我们将此函数放到应用程序类 QuickSortApp 之中。
n 修改源代码
更改 C# 源文件 (class1.cs),如下面以斜体突出显示的 代码所示。其他的差异(如类名)可忽略不计。
// Import namespaces using System; using System.Collections; using System.IO; // Declare namespace namespace MsdnAA { // Declare application class class QuickSortApp { // Application initialization static void Main (string[] szArgs) { ... ... ... // Pass to QuickSort function QuickSort (szContents, 0, szContents.Count - 1); ... ... ... } // QuickSort implementation static void QuickSort (ArrayList szArray, int nLower, int nUpper) { // Check for non-base case if (nLower < nUpper) { // Split and sort partitions int nSplit = Partition (szArray, nLower, nUpper); QuickSort (szArray, nLower, nSplit - 1); QuickSort (szArray, nSplit + 1, nUpper); } } // QuickSort partition implementation static int Partition (ArrayList szArray, int nLower, int nUpper) { // Pivot with first element int nLeft = nLower + 1; string szPivot = (string) szArray[nLower]; int nRight = nUpper; // Partition array elements string szSwap; while (nLeft <= nRight) { // Find item out of place while (nLeft <= nRight) { if (((string) szArray[nLeft]).CompareTo (szPivot) > 0) break; nLeft = nLeft + 1; } while (nLeft <= nRight) { if (((string) szArray[nRight]).CompareTo (szPivot) <= 0) break; nRight = nRight - 1; } // Swap values if necessary if (nLeft < nRight) { szSwap = (string) szArray[nLeft]; szArray[nLeft] = szArray[nRight]; szArray[nRight] = szSwap; nLeft = nLeft + 1; nRight = nRight - 1; } } // Move pivot element szSwap = (string) szArray[nLower]; szArray[nLower] = szArray[nRight]; szArray[nRight] = szSwap; return nRight; } } } |
n QuickSort() 函数
这个函数需要三个参数:对数组的引用、下界和上界。它调用 Partition() 函数将数组分成两部分,其中一部分包含 Pivot 值之前的所有字符串,另一部分包含 Pivot 值之后的所有字符串。然后,它调用自身来对每个部分进行排序。
上面修改中的注释应该说明了每个代码块的作用。唯一的新概念就是 CompareTo() 方法的使用,该方法是 String 类的成员,并且应该是自说明的。
n 运行 QuickSort 应用程序
这一步完成 QuickSort C# 示例应用程序。现在,可以构建项目并运行应用程序。需要提供一个示例文本文件,以供其进行排序。将该文件放在与 EXE 文件相同的目录中。
n 程序输出
下面是已完成的 QuickSort C# .NET 示例应用程序的输出。可以查看示例输入文件 'example.txt' 和输出文件 'output.txt'。
步骤 8. 使用调试器
调试器是诊断程序问题的一个必不可少的工具。我们觉得有必要在本入门指南中对其进行介绍。这最后一步将向您展示如何走查程序和使用诸如 QuickWatch 这样的功能。
n 设置断点
当程序在调试器中运行时,断点会暂停程序的执行,从而使开发人员能够控制调试器。要设置断点,请右键单击您想要程序暂停的行,然后单击 InsertBreakpoint,如下所示。
注:带有断点的行以红色突出显示。通过再次右键单击该行并选择 Remove Breakpoint 可以删除断点。
n 单步调试程序
既然设置了断点(最好是在前面所示的行中),就让我们在调试器中运行程序。在 Debug 菜单中,选择 Start 而不是同前面一样选择 Start Without Debugging。这样就在调试器中启动了程序,并因而激活了断点。
一旦程序遇到断点,调试器便会接收程序的控制。这时会有一个箭头指向当前执行的行。
要单步调试一行代码,可以选择 Debug | Step Over 并观察光标是否移到下一行。Debug | Step Into 命令允许您单步执行将要调用的函数。进行两次 Step Over 之后的屏幕如下所示。
如果想要程序在遇到下一个断点、遇到异常或退出之前继续执行,请从菜单中选择 Debug | Continue。
n 检查变量值
当您可以控制调试器时,可将鼠标指针移到变量上以获得它的基本值。
您也可以右键单击变量,然后从上下文菜单中选择 QuickWatch。QuickWatch 将为您提供关于某些变量(如 ArrayList 对象)的更多详细信息。
n 其他调试器工具
Visual Studio 调试器具有许多其他工具(例如 Call Stack 查看器)的功能,可以使用此调试器来查看到此为止调用的函数。还可以获得内存转储和关于进程中线程的信息。我们鼓励您使用这些功能强大的调试工具。
小结
本入门指南旨在帮助您用 Visual Studio 构建一个简单的 C# 项目。它无法进行全面的介绍。我们鼓励您查询关于 C# 和 .NET 的其他资源,以便更多地学习这些技术。在完成本教程之后,您至少有了一个可用的项目,在您研究 Visual C# 时,可以从修改此这些代码开始。
为了方便起见,我们提供了完整的源程序和项目文件。您可以通过本文档顶部的目录来访问它们。
n 其他资源
我们强烈推荐下面这些关于 C# 和 .NET 平台的书籍。它们是开发人员尝试学习这些新技术的有益资源。
《C# 高级编程第四版》 英文名为《Professional C# 2005》
《C#.NET 技术内幕》 英文名为《Microsoft Visual C#.NET 2003 Unleashed》
第二章 面向对像ORM
ORM的全称是Object/Relation Mapping,即对象/关系映射。ORM也可理解是一种规范,具体的ORM框架可作为应用程序和数据库的桥梁。目前ORM的产品非常多,比如Apache组织下的OJB,Oracle的TopLink,JDO等等。
一、 什么是ORM
ORM并不是一种具体的产品,而是一类框架的总称,它概述了这类框架的基本特征:完成面向对象的程序设计语言到关系数据库的映射。基于ORM框架完成映射后,既可利用面向对象程序设计语言的简单易用性,又可利用关系数据库的技术优势。
面向对象程序设计语言与关系数据库发展不同步时,需要一种中间解决方案,ORM框架就是这样的解决方案。笔者认为,随着面向对象数据库的发展,其理论逐步完善,最终会取代关系数据库。只是这个过程不可一蹴而就,ORM框架在此期间内会蓬勃发展。但随着面向对象数据库的出现,ORM工具会自动消亡。
二、 为什么需要ORM
在上一节已经基本回答了这个问题,面向对象的程序设计语言,代表了目前程序设计语言的主流和趋势,其具备非常多的优势,比如:
n 面向对象的建模、操作。
n 多态、继承。
n 摒弃难以理解的过程。
n 简单易用,易理解性。
但数据库的发展并未与程序设计语言同步,而且,关系数据库系统的某些优势,也是面向对象的语言目前无法解决的。比如:
n 大量数据操作查找、排序。
n 集合数据连接操作、映射。
n 数据库访问的并发、事务。
n 数据库的约束、隔离。
面对这种面向对象语言与关系数据库系统并存的局面,采用ORM就变成一种必然。
三、 流行的ORM框架简介
目前ORM框架的产品非常多,除了各大著名公司、组织的产品外,甚至,其他一些小团队也都有推出自己的ORM框架。目前流行的ORM框架有如下这些产品。
n 大名鼎鼎的(N)Hibernate:出自Gavin King的手笔,目前最流行的开源ORM框架,其灵巧的设计,优秀的性能,还有丰富的文档,都是其迅速风靡全球的重要因素。
n 传统的Entity EJB:Entity EJB实质上也是一种ORM技术,这是一种备受争议的组件技术,很多人说它非常优秀,也有人说它一钱不值。事实上,EJB为J2EE的蓬勃发展赢得了极高的声誉,就笔者的实际开发经验而言,EJB作为一种重量级、高花费的ORM技术上,具有不可比拟的优势。但由于其必须运行在EJB容器内,而且学习曲线陡峭,开发周期、成本相对较高,因而限制EJB的广泛使用。
n IBATIS:Apache软件基金组织的子项目。与其称它是一种ORM框架,不如称它是一种 “Sql Mapping”框架。相对Hibernate的完全对象化封装,iBATIS更加灵活,但开发过程中开发人员需要完成的代码量更大,而且需要直接编写SQL语句。
n Oracle的TopLink:作为一个遵循OTN协议的商业产品,TopLink在开发过程中可以自由下载和使用,但一旦作为商业产品使用,则需要收取费用。可能正是这一点,导致了TopLink的市场占有率。
n OJB:Apache软件基金组织的子项目。开源的ORM框架,但由于开发文档不是太多,而且OJB的规范一直并不稳定,因此并未在开发者中赢得广泛的支持。
第三章 Spring.NET入门
一、 Spring.NET概览
Spring.NET 是一个关注于.NET企业应用开发的应用程序框架。它能够提供宽广范围的功能,例如依赖注入、面向方面编程(AOP)、数据访问抽象, 以及ASP.NET集成等。基于java的spring框架的核心概念和价值已被应用到.NET。Spring.NET 1.0 包含一个完全功能的依赖注入容器和AOP库。后续的发布将包含对ASP.NET、Remoting和数据访问的支持。下图展现出了 Spring .NET的各个模块。具有黑色阴影的模块包含在1.0版本中,其他模块计划在将来的发布中推出。在很多情况下,你可以在我们的下载网站中发现可以工作的计划模块的实现。
Spring .NET框架概览
Spring.Core 库是框架的基础, 提供依赖注入功能。Spring.NET中大多数类库依赖或扩展了Spring.Core的功能。IObjectFactory接口提供了一个简单而优雅的工厂模式,移除了对单例和一些服务定位stub写程序的必要。允许你将真正的程序逻辑的配置和依赖的详细情况解耦。作为对IObjectFactory的扩展,IApplicationContext接口也在Spring.Core库中,并且添加了许多企业应用为中心的功能,例如利用资源文件进行文本本地化、事件传播、资源加载等等。
Spring.Aop 库提供对业务对象的面向方面编程(AOP) 的支持。Spring.Aop 库是对Spring.Core库的补充,可为声明性地建立企业应用和为业务对象提供服务提供坚实的基础。
Spring.Web 库扩展了ASP.NET,添加了一些功能,如对ASP.NET页面的依赖注入,双向数据绑定,针对 ASP.NET 1.1的Master pages以及改进的本地化支持。
Spring.Services库可让你将任何“一般”对象(即没有从其他特殊的服务基类继承的对象)暴露为企业服务或远程对象,使得.NET Web services 获得依赖注入的支持,并覆盖属性元数据。此外还提供了对Windows Service的集成。
Spring.Data 库提供了数据访问层的抽象,可以被多个数据访问提供者(从ADO.NET 到多个ORM 提供者)应用。它还包含一个对ADO.NET的抽象层,移除了为ADO.NET编写可怕的编码和声明性的事务管理的必要。
Spring.ORM库提供了对常见对象关系映射库的的集成,提供了一些功能,比如对声明性事务管理的支持。
二、 第一个Spring.NET的程序
n 建立项目
项目名称为:SpringSample,NameSpace为“OKEC.Sample.Spring”。
n 添加HelloTest类
HelloTest.cs
using System; namespace OKEC.Sample.Spring { /// <summary> /// HelloTest 的摘要说明。 /// </summary> public class HelloTest { public HelloTest() { // // TODO: 在此处添加构造函数逻辑 // } public void Test() { Console.WriteLine("This is Spring.NET Sample Test!"); Console.WriteLine("Please press Enter close the windows!"); Console.ReadLine();//让程序停留,回车关闭。 } } } |
n 添加Spring.NET的配置文件
文件名:Spring_bean.xml,属性设置为:嵌入的资源/ Embedded Resource
<?xml version="1.0" encoding="utf-8"?> <objects xmlns="http://www.springframework.net" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://www.springframework.net http://www.springframework.net/xsd/spring-objects.xsd"> <object id="Hello" type="OKEC.Sample.Spring.HelloTest,SpringSample" /> </objects> |
n 建立Spring.NET的容器初始化对像
SpringContext.cs
using System; using Spring.Core; using Spring.Aop; using System; using Spring.Core; using Spring.Aop; using Spring.Context; using Spring.Context.Support; namespace OKEC.Sample.Spring { /// <summary> /// SpringFactory 的摘要说明。 /// </summary> public class SpringContext { public SpringContext() { // // TODO: 在此处添加构造函数逻辑 // } private static bool isInit = false; private static IApplicationContext context; public static void init() { string[] xmlFiles = new string[1]; xmlFiles[0] = "assembly://SpringSample/OKEC.Sample.Spring/Spring_bean.xml"; context = new XmlApplicationContext(xmlFiles); isInit = true; } public static IApplicationContext Context { get{ if(!isInit) { init(); } return context; } } } } |
n 添加启动程序
StartMain.cs
using System; namespace OKEC.Sample.Spring { /// <summary> /// StartMain 的摘要说明。 /// </summary> public class StartMain { public StartMain() { // // TODO: 在此处添加构造函数逻辑 // } [STAThread] static void Main() { //Startup Spring Content SpringContext.init();
//Test Spring IOC HelloTest test = (HelloTest)SpringContext.Context.GetObject("Hello"); test.Test(); } } } |
n 运行程序
结果为:
This is Spring.NET Sample Test! Please press Enter close the windows! |
你的第一个Spring.NET的程序成功了!
第四章 NHibernate入门
一、 什么是Nhibernate
NHibernate 是一个基于.Net 的针对关系型数据库的对象持久化类库。Nhibernate 来源于非常优秀的基于Java的Hibernate 关系型持久化工具。
NHibernate 从数据库底层来持久化你的.Net 对象到关系型数据库。NHibernate 为你处理这些,远胜于你不得不写SQL去从数据库存取对象。你的代码仅仅和对象关联,NHibernat 自动产生SQL语句,并确保对象提交到正确的表和字段中去。
二、 Nhibernate概述
对NHibernate体系结构的非常高层的概览:
这幅图展示了NHibernate使用数据库和配置文件数据来为应用程序提供持久化服务(和持久化的对象)。
我们试图显示更多NHibernate运行时体系结构的细节。 但挺不幸的,NHibernate是比较灵活的并且提供了好几种不同的运行方式。我们展示一下两种极端情况。轻型体系中,应用程序自己提供ADO.NET连接,并且自行管理事务。这种方式使用了NHibernate API的一个最小子集。
全面解决体系中,对于应用程序来说,所有的底层ADO.NET API都被抽象了,NHibernate会替你照管所有的细节。
下面是图中一些对象的定义:
SessionFactory (NHibernate.ISessionFactory)
对属于单一数据库的编译过的映射文件的一个线程安全的,不可变的缓存快照。它是Session的工厂,是ConnectionProvider的客户。可以持有一个可选的(第二级)数据缓存,可以在进程级别或集群级别保存可以在事物中重用的数据。
会话,Session (NHibernate.ISession)
单线程,生命期短促的对象,代表应用程序和持久化层之间的一次对话。封装了一个ADO.NET连接。也是Transaction的工厂。保存有必需的(第一级)持久化对象的缓存,用于遍历对象图,或者通过标识符查找对象。
持久化对象(Persistent)及其集合(Collections)
生命期短促的单线程的对象,包含了持久化状态和商业功能。它们可能是普通的对象,唯一特别的是他们现在从属于且仅从属于一个Session。一旦Session被关闭,他们都将从Session中取消联系,可以在任何程序层自由使用(比如,直接作为传送到表现层的DTO,数据传输对象)。
临时对象(Transient Object)及其集合(Collection)
目前没有从属于一个Session的持久化类的实例。他们可能是刚刚被程序实例化,还没有来得及被持久化,或者是被一个已经关闭的Session所实例化的。
事务Transaction (NHibernate.ITransaction)
(可选) 单线程,生命期短促的对象,应用程序用它来表示一批工作的原子操作。是底层的ADO.NET事务的抽象。一个Session某些情况下可能跨越多个Transaction 事务。
ConnectionProvider (NHibernate.Connection.ConnectionProvider)
(可选)ADO.NET连接的工厂。从底层的IDbConnection抽象而来。对应用程序不可见,但可以被开发者扩展/实现。
TransactionFactory (net.sf.hibernate.TransactionFactory)
(可选)事务实例的工厂。对应用程序不可见,但可以被开发者扩展/实现。
在上面的轻型结构中,程序没有使用Transaction / TransactionFactory 或者ConnectionProvider API,直接和ADO.NET对话了
三、 第一个NHibernate 程序
任何熟悉Hibernate的人会发现这篇指南和Glen Smith 的 A Hitchhiker's Guide to Hibernate 非常相近。这里的内容正是基于他的指南,因此所有的感谢都应该给与他。
NHibernate的文档并非每处都和Hibernate的文档一致。然而,项目的相似应该能使读者通过读Hibernate的文档来很好的理解NHibernate如何工作。
这篇文档意在让你尽可能快的开始使用NHibernate。它将介绍如何持久化一个简单的对象到一张表里。想得到更多的复杂的例子,可以参考NUnit测试及附带代码。
我们将进行以下步骤。
1.新建一个将要持久化.Net对象的表
2.构建一个可以让NHibernate知道如何持久化对象属性的映射文件
3.构建一个需要被持久化的.Net类
4.构建一个存放NHibernater的配置文件的对像
5.使用NHibernate的API测试你的第一个NHibernate程序
n 新建项目
项目名称为:NHibernateSample,名字空间:OKEC.Sample.NHibernate
n 建立数据表
数据库为SQLServer2000,表名为:my_users
if exists (select * from dbo.sysobjects where id = object_id(N'[dbo].[my_users]') and OBJECTPROPERTY(id, N'IsUserTable') = 1) drop table [dbo].[my_users] GO CREATE TABLE [dbo].[my_users] ( [LogonId] [varchar] (50) COLLATE Chinese_PRC_CI_AS NOT NULL , [UserName] [varchar] (50) COLLATE Chinese_PRC_CI_AS NULL , [Password] [varchar] (50) COLLATE Chinese_PRC_CI_AS NULL , [EmailAddress] [varchar] (50) COLLATE Chinese_PRC_CI_AS NULL , [LastLogon] [datetime] NULL ) ON [PRIMARY] GO ALTER TABLE [dbo].[my_users] ADD CONSTRAINT [PK_my_users] PRIMARY KEY CLUSTERED ( [LogonId] ) ON [PRIMARY] GO |
n 建立XML对像映射文件
现在我们有数据表和需要去映射它的.Net类。我们需要一种方式去让NHibernate知道如何从一个映射到另一个。这个任务依赖于映射文件来完成。最易于管理的办法是为每一个类写一个映射文件,如果你命名它是User.hbm.xml并且把它放在和类的同一个目录里,NHiberante将会使得事情简单起来。下面是User.hbm.xml的例子:
<?xml version="1.0" encoding="utf-8" ?> <hibernate-mapping xmlns="urn:nhibernate-mapping-2.0"> <class name=" OKEC.Sample.NHibernate.NHibernateTest.User,NHibernateSample" table="my_users"> <id name="Id" column="LogonId" type="String" length="20"> <generator class="assigned" /> </id> <property name="UserName" column="UserName" type="String" length="40"/> <property name="Password" column="Password" type="String" length="20"/> <property name="EmailAddress" column="EmailAddress" type="String" length="40"/> <property name="LastLogon" column="LastLogon" type="DateTime"/> </class> </hibernate-mapping> |
注意事项:在Visual Studio 2003/2005中要将此文件的属性设置为“嵌入的资源”(Embedded Resource)
n 建立对像
对像定义:User.cs
using System; namespace OKEC.Sample.NHibernate.NHibernateTest { /// <summary> /// Summary description for User. /// </summary> public class User { private string id; private string userName; private string password; private string emailAddress; private DateTime lastLogon; public User() { } public string Id { get { return id; } set { id = value; } } public string UserName { get { return userName; } set { userName = value; } } public string Password { get { return password; } set { password = value; } } public string EmailAddress { get { return emailAddress; } set { emailAddress = value; } } public DateTime LastLogon { get { return lastLogon; } set { lastLogon = value; } } } } |
n 编写Nhibernate的初始化配置程序
程序名:MyConfiguration.cs
using System; using NHibernate.Cfg; namespace OKEC.Sample.NHibernate.NHibernateTest { /// <summary> /// MyConfiguration 的摘要说明。 /// </summary> public class MyConfiguration { public MyConfiguration() { // // TODO: 在此处添加构造函数逻辑 // } public Configuration GetConfig() { try { Configuration cfg = new Configuration(); cfg.SetProperty("hibernate.connection.provider","NHibernate.Connection.DriverConnectionProvider"); //请修改此行中的SQLServer的配置 cfg.SetProperty("hibernate.connection.connection_string","Data Source=192.168.88.15;Database=liluhua;User ID=sa;Password=sa;Trusted_Connection=False"); cfg.SetProperty("hibernate.dialect","NHibernate.Dialect.MsSql2000Dialect"); cfg.SetProperty("hibernate.connection.driver_class","NHibernate.Driver.SqlClientDriver"); cfg.AddAssembly("NHibernateSample"); return cfg; } catch(Exception ex) { Console.WriteLine(ex.Message); Console.WriteLine(ex.StackTrace); } return null; } } } |
n 编写调用程序
准备好上面的一切,我们就可以开始编辑启动程序,来测试你的第一个Nhibernate程序了。
程序名:UserFixture.cs
using System; using System.Collections; using NHibernate; using NHibernate.Cfg; using NHibernate.Expression; namespace OKEC.Sample.NHibernate.NHibernateTest { /// <summary> /// UserFixture 的摘要说明。 /// </summary> public class UserFixture { public UserFixture() { // // TODO: 在此处添加构造函数逻辑 // } public void ValidateQuickStart() { try { //得到NHibernate的配置 MyConfiguration config = new MyConfiguration(); Configuration cfg = config.GetConfig();
ISessionFactory factory = cfg.BuildSessionFactory(); ISession session = factory.OpenSession(); ITransaction transaction = session.BeginTransaction();
User newUser = null; try { newUser = (User)session.Load(typeof(User), "joe_cool"); } catch { } if(newUser==null) { newUser = new User(); newUser.Id = "joe_cool"; newUser.UserName = "Joseph Cool"; newUser.Password = "abc123"; newUser.EmailAddress = "joe@cool.com"; newUser.LastLogon = DateTime.Now;
// Tell NHibernate that this object should be saved session.Save(newUser); }
// commit all of the changes to the DB and close the ISession transaction.Commit(); session.Close();
// open another session to retrieve the just inserted user session = factory.OpenSession();
User joeCool = (User)session.Load(typeof(User), "joe_cool");
// set Joe Cool's Last Login property joeCool.LastLogon = DateTime.Now;
// flush the changes from the Session to the Database session.Flush();
IList recentUsers = session.CreateCriteria(typeof(User)) .Add(Expression.Gt("LastLogon", new DateTime(2004, 03, 14, 20, 0, 0))) .List(); foreach(User user in recentUsers) { //Assert.IsTrue(user.LastLogon > (new DateTime(2004, 03, 14, 20, 0, 0)) ); Console.WriteLine(user.UserName); Console.WriteLine(user.Password); } session.Close(); } catch(Exception ex) { Console.WriteLine(ex.Message); Console.WriteLine(ex.StackTrace); } Console.ReadLine(); } } } |
n 测试你的程序
如果运行后没有出错,显示了结果,说明你的第一个NHibernate程序成功了。
Joseph Cool abc123 |
第五章 Spring.NET 与 NHibernate 的整合
我们成功的运行了自己的第一个Spring.NET程序和第一个NHibernate程序。下面我们将上面的程序整合到一个项目中来。让Spring.NET的容器来管理NHibernate。
一、 建立新的项目(SpringNHibernateSample)
项目名称为:SpringNHibernateSample 名字空间:OKEC.Sample
二、 添加NHibernate程序
将NHibernateSample项目的User.cs、User.hbm.xml加入的新的项目中。
并修改User.hbm.xml,将其中的
<class name=" OKEC.Sample.NHibernate.NHibernateTest.User,NHibernateSample" table="my_users"> |
改为:
<class name=" OKEC.Sample.NHibernate.NHibernateTest.User,SpringNHibernateSample" table="my_users"> |
三、 添加Spring.NET的程序
n 首先,需要将SpringSample项目中的HelloTest.cs、Spring_bean.xml、SpringContext.cs加入到新的项目,并修改其中有用到程序集相关的地方。
如将Spring_bean.xml中的:
<object id="Hello" type="OKEC.Sample.Spring.HelloTest,SpringSample" /> |
改为:
<object id="Hello" type="OKEC.Sample.Spring.HelloTest,SpringNHibernateSample" /> |
n 然后,添加一个为NHibernate提供DbProvider的实现类,此类实现了Spring.Data.Common.IDbProvider接口,为NHibernate提供DbProvider所需的链接字串(ConnectionString)。
using System; using Spring.Data.Common; namespace OKEC.Sample.Spring { /// <summary> /// SQLPriv 的摘要说明。 /// </summary> public class SQLProvider:IDbProvider { public SQLProvider() { // // TODO: 在此处添加构造函数逻辑 // } #region IDbProvider 成员 public System.Data.IDbConnection CreateConnection() { // TODO: 添加 SQLPriv.CreateConnection 实现 return null; } public string CreateParameterName(string name) { // TODO: 添加 SQLPriv.CreateParameterName 实现 return null; } public System.Data.IDbDataParameter CreateParameter() { // TODO: 添加 SQLPriv.CreateParameter 实现 return null; } private string _connectionString=""; public string ConnectionString { get { // TODO: 添加 SQLPriv.ConnectionString getter 实现 return _connectionString; } set { _connectionString = value; // TODO: 添加 SQLPriv.ConnectionString setter 实现 } } public string ExtractError(Exception e) { // TODO: 添加 SQLPriv.ExtractError 实现 return null; } public System.Data.IDbDataAdapter CreateDataAdapter() { // TODO: 添加 SQLPriv.CreateDataAdapter 实现 return null; } public bool IsDataAccessException(Exception e) { // TODO: 添加 SQLPriv.IsDataAccessException 实现 return false; } public System.Data.IDbCommand CreateCommand() { // TODO: 添加 SQLPriv.CreateCommand 实现 return null; } public object CreateCommandBuilder() { // TODO: 添加 SQLPriv.CreateCommandBuilder 实现 return null; } public IDbMetadata DbMetadata { get { // TODO: 添加 SQLPriv.DbMetadata getter 实现 return null; } } #endregion } } |
n 最后,添加一个Spring.Data.NHibernate对NHibernate的封装对像,此对像实现对User对像数据操作,继承自
Spring.Data.NHibernate.Support.HibernateDaoSupport。
using System; using System.Collections; using Spring.Data.NHibernate.Support; namespace OKEC.Sample.NHibernate.NHibernateTest { /// <summary> /// UserDao 的摘要说明。 /// </summary> public class UserDao : HibernateDaoSupport { public UserDao() { // // TODO: 在此处添加构造函数逻辑 // } public bool SaveObject(User user) { HibernateTemplate.Save(user); return true; } public bool DeleteObject(User user) { HibernateTemplate.Delete(user); return true; } public bool UpdateObject(User user) { HibernateTemplate.Update(user); return true; } public IList GetAllObjectsList() { return HibernateTemplate.LoadAll(typeof(User)); } public User Load(Object ID) { return (User)HibernateTemplate.Load(typeof(User),ID); } } } |
四、 添加Spring.NET为NHibernate的容器配置
现在就可以在Spring.NET的容器中添加Nhibernate的配置了。
如下Spring_nhibernate.xml:
<?xml version="1.0" encoding="utf-8" ?> <objects xmlns='http://www.springframework.net'>
<!-- NHibernate初始化的 --> <object id="DbProvider" type="OKEC.Sample.Spring.SQLProvider,SpringNHibernateSample"> <property name="ConnectionString" value="Data Source=192.168.88.15;Database=liluhua;User ID=sa;Password=sa;Trusted_Connection=False"/> </object>
<object id="SessionFactory" type="Spring.Data.NHibernate.LocalSessionFactoryObject, Spring.Data.NHibernate"> <property name="DbProvider" ref="DbProvider"/> <property name="MappingAssemblies"> <list> <value>SpringNhibernateSample</value> </list> </property> <property name="HibernateProperties"> <dictionary> <entry key="hibernate.connection.provider" value="NHibernate.Connection.DriverConnectionProvider"/> <!--entry key="hibernate.connection.connection_string" value="Data Source=192.168.188.188;Database=Test;User ID=satest;Password=satest;Trusted_Connection=False"/--> <entry key="hibernate.dialect" value="NHibernate.Dialect.MsSql2000Dialect"/> <entry key="hibernate.connection.driver_class" value="NHibernate.Driver.SqlClientDriver"/> </dictionary> </property> </object>
<object id="HibernateTransactionManager" type="Spring.Data.NHibernate.HibernateTransactionManager, Spring.Data.NHibernate"> <property name="DbProvider" ref="DbProvider"/> <property name="sessionFactory" ref="SessionFactory"/> </object>
<object id="TransactionInterceptor" type="Spring.Transaction.Interceptor.TransactionInterceptor, Spring.Data"> <property name="TransactionManager" ref="HibernateTransactionManager"/> <property name="TransactionAttributeSource"> <object type="Spring.Transaction.Interceptor.AttributesTransactionAttributeSource, Spring.Data"/> </property> </object>
<!-- 以下是业务相关的 --> <object id="UserDao" type="OKEC.Sample.NHibernate.NHibernateTest.UserDao, SpringNHibernateSample"> <property name="SessionFactory" ref="SessionFactory"/> </object> </objects> |
我们现在对上面的加以细解:
下面这几行,是配置Nhibernate所需的数据库的DbProvider
<object id="DbProvider" type="OKEC.Sample.Spring.SQLProvider,SpringNHibernateSample"> <property name="ConnectionString" value="Data Source=192.168.88.15;Database=liluhua;User ID=sa;Password=sa;Trusted_Connection=False"/> </object> |
下面的是对Nhibernate的SessionFactory的封装的对像的定义
<object id="SessionFactory" type="Spring.Data.NHibernate.LocalSessionFactoryObject, Spring.Data.NHibernate"> <property name="DbProvider" ref="DbProvider"/> <property name="MappingAssemblies"> <list> <value>SpringNhibernateSample</value> </list> </property> <property name="HibernateProperties"> <dictionary> <entry key="hibernate.connection.provider" value="NHibernate.Connection.DriverConnectionProvider"/> <!--entry key="hibernate.connection.connection_string" value="Data Source=192.168.188.188;Database=Test;User ID=satest;Password=satest;Trusted_Connection=False"/--> <entry key="hibernate.dialect" value="NHibernate.Dialect.MsSql2000Dialect"/> <entry key="hibernate.connection.driver_class" value="NHibernate.Driver.SqlClientDriver"/> </dictionary> </property> </object> |
下面的是对Nhibernate中的Transaction封装对像
<object id="HibernateTransactionManager" type="Spring.Data.NHibernate.HibernateTransactionManager, Spring.Data.NHibernate"> <property name="DbProvider" ref="DbProvider"/> <property name="sessionFactory" ref="SessionFactory"/> </object>
<object id="TransactionInterceptor" type="Spring.Transaction.Interceptor.TransactionInterceptor, Spring.Data"> <property name="TransactionManager" ref="HibernateTransactionManager"/> <property name="TransactionAttributeSource"> <object type="Spring.Transaction.Interceptor.AttributesTransactionAttributeSource, Spring.Data"/> </property> </object> |
下面是对NHibernate业务操作对像的定义
<object id="UserDao" type="OKEC.Sample.NHibernate.NHibernateTest.UserDao, SpringNHibernateSample"> <property name="SessionFactory" ref="SessionFactory"/> </object> |
五、 编写测试程序代码
一切准备就绪,我们就可以编写测试程序的代码了。
如下:StartMain.cs
using System; using System.Collections; using OKEC.Sample.NHibernate.NHibernateTest; namespace OKEC.Sample.Spring { /// <summary> /// StartMain 的摘要说明。 /// </summary> public class StartMain { public StartMain() { // // TODO: 在此处添加构造函数逻辑 // } [STAThread] static void Main() { //Startup Spring & NHibernate Content SpringContext.init();
//Test Spring IOC HelloTest test = (HelloTest)SpringContext.Context.GetObject("Hello"); test.Test();
//Test Spring & NHibernate UserDao dao = SpringContext.Context.GetObject("UserDao") as UserDao; User newUser = null; try { newUser = dao.Load("joe_cool"); } catch {} if(newUser==null) { newUser = new User(); newUser.Id = "joe_cool"; newUser.UserName = "Joseph Cool"; newUser.Password = "abc123"; newUser.EmailAddress = "joe@cool.com"; newUser.LastLogon = DateTime.Now; // Tell NHibernate that this object should be saved dao.SaveObject(newUser); } User joeCool = dao.Load("joe_cool"); // set Joe Cool's Last Login property joeCool.LastLogon = DateTime.Now; // flush the changes from the Session to the Database dao.UpdateObject(joeCool); IList recentUsers = dao.GetAllObjectsList(); foreach(User user in recentUsers) { //Assert.IsTrue(user.LastLogon > (new DateTime(2004, 03, 14, 20, 0, 0)) ); Console.WriteLine(user.UserName); Console.WriteLine(user.Password); } Console.ReadLine();//让程序停留,回车关闭。 } } } |
六、 测试并查看结果
如果你看到了以下的输出结果,说明你已经成功了!
This is Spring.NET Sample Test! Please press Enter close the windows! Joseph Cool abc123 |
第六章 深入Spring.NET 与 NHibernate开发
待写
第七章 项目实战----办公自动化系统
待写
第八章 结束语
待写