[转载]Asp.net MVC 2 + Castle + NHibernate 项目实战(1)

[转载]Asp.net MVC 2 + Castle + NHibernate 项目实战(1)

http://www.cnblogs.com/taven/archive/2010/09/28/1837478.html

本文将开始一步一步地使用Asp.net MVC 2 + Castle + NHibernate 开发一个项目。

 

在开始之前,我先对这三个组件做一个简单的介绍:

 Asp.net MVC 

它是微软提供的一个基于MVC标准的Web开发模式,其典型特点是有控制器和视图;在这之前,.NET下的Web开发模式大多是采用WebForm,其典型特点是服务端控件和后台触发事件; 

 NHibernate

它是一个ORM框架,使用Java的SSH做过项目开发的人就非常熟悉了,当前最新版本为NHibernate 3.0,听说完全支持Linq查询语句了,以前只支持HQL语句。 

 Castle

它是一个非常大的框架,包含IoC、MVC、ORM、AOP等,这次我只用到其IoC的功能。

 

关于IoC

IoC的英文是Inversion of Control,中文意思是控制反转,也有称其为Dependence Injection,中文意思是依赖注入,也就是DI;

除了Castle可以作IoC 外,还有著名的Spring.Net、Unity等,在我另外一个开源博客项目 http://rorowo.codeplex.com/ 中就用到了微软的Unity 2.0作为IoC。

 

好了,废话不多说,下面开始。 

 

首先,第一步是创建一个解决方案,项目结构如下:

[转载]Asp.net MVC 2 + Castle + NHibernate 项目实战(1) 

需要引用的DLL库文件,我们全部放在RoRoWo.Common.DependLib下:

 [转载]Asp.net MVC 2 + Castle + NHibernate 项目实战(1)

 根据表的结构,如图: 

 [转载]Asp.net MVC 2 + Castle + NHibernate 项目实战(1)

 

现在我们开始为表结构创建实体类,以BlogCategory表为例,我们根据该表的字段名称和数据类型创建一个POCO类,放在RoRoWo.Domain.Entities下,代码内容如下:

 ExpandedBlockStart.gif代码

namespace
 RoRoWo.Domain {

    
    

public
 
class
 Blogcategory {

        

public
 Blogcategory() { }
        

public
 
virtual
 
int
 Cateid { 
get

set
; }
        

public
 
virtual
 IList
<
Blogarticle
>
 Blogarticles { 
get

set
; }
        

public
 
virtual
 
string
 Catename { 
get

set
; }
        

public
 
virtual
 
int
 Parentid { 
get

set
; }
        

public
 
virtual
 
int
 State { 
get

set
; }
        

public
 
virtual
 
int
 Sortid { 
get

set
; }
        

public
 
virtual
 
int
 Articlecount { 
get

set
; }
        

public
 
virtual
 System.DateTime Createtime { 
get

set
; }
        

public
 
virtual
 
string
 Note { 
get

set
; }

    }

}

 

 在这里,我推荐一个可以生成NHibernate映射对象代码的工具(开源的),项目地址:NHibernate Mapping Generator(它同时支持 hbm.xml文件的方式、Fluent Mapping方式、Castle ActiveRecord方式)

接下来,要为这个类创建一个映射关系,我这里使用配置文件的方式,但是正式项目推荐使用Fluent方式。我们创建一个XML文件,文件名为“Blogcategory.hbm.xml”,放在RoRoWo.Domain.Mappings下,内容如下:

[转载]Asp.net MVC 2 + Castle + NHibernate 项目实战(1)
ExpandedBlockStart.gif
代码

<?
xml version
=

1.0

 encoding
=

utf-8

?>


<
hibernate

mapping assembly
=

RoRoWo.Domain

 
namespace
=

RoRoWo.Domain

 xmlns
=

urn:nhibernate-mapping-2.2

 
default

lazy
=

false

>

    

<
class
 name
=

Blogcategory

 table
=

BlogCategory

>

        

<
id name
=

Cateid

 column
=

CateID

>

            

<
generator 
class
=

native

 
/>

        

</
id
>

        

<
property name
=

Catename

 column
=

CateName

 
/>

        

<
property name
=

Parentid

 column
=

ParentID

 
/>

        

<
property name
=

State

 column
=

State

 
/>

        

<
property name
=

Sortid

 column
=

SortID

 
/>

        

<
property name
=

Articlecount

 column
=

ArticleCount

 
/>

        

<
property name
=

Createtime

 column
=

CreateTime

 
/>

        

<
property name
=

Note

 column
=

Note

 
/>

    

</
class
>


</
hibernate

mapping
>

 

 

这里就是一个典型的映射关系了,如果有一对多,多对多的表关系,也是在这个配置中进行维护。

创建这个XML后,还有个很重要的操作,就是将其设为“嵌入的资源” ,鼠标右键查看“Blogcategory.hbm.xml”文件属性,将“生成操作”项的“内容”改为“嵌入的资源”,如图:

[转载]Asp.net MVC 2 + Castle + NHibernate 项目实战(1) 

 到此,一个映射关系的实体就建立好了,下面我们就要来实现NHibernate对该表的增删改查操作了,我们把这些操作的实现放在RoRoWo.Data中。

注意,RoRoWo.Data需要添加对 “Castle.Core.dll” 

Castle.DynamicProxy2.dll

 

NHibernate.dll

 

RoRoWo.Domain

 的引用。

 

在创建数据库操作类之前,我们需要创建一个 “SessionManager”类,它负责维护整个ORM中的上下文,这里我使用李永京的一个类,其代码如下:

[转载]Asp.net MVC 2 + Castle + NHibernate 项目实战(1)
ExpandedBlockStart.gif
代码

using
 System;

using
 System.Collections.Generic;

using
 System.Linq;

using
 System.Text;


using
 NHibernate;

using
 NHibernate.Cfg;


namespace
 RoRoWo.Data

{

    

public
 
class
 SessionManager

    {

        

private
 ISessionFactory _sessionFactory;
        

public
 SessionManager()

        {


            _sessionFactory 
=
 GetSessionFactory();

        }
        

private
 ISessionFactory GetSessionFactory()

        {

            

return
 (
new
 Configuration()).Configure().BuildSessionFactory();

        }
        

public
 ISession GetSession()

        {

            

return
 _sessionFactory.OpenSession();

        }

    }

}

 

 

我们创建一个“BlogCategoryRespository.cs”文件,其代码如下:

[转载]Asp.net MVC 2 + Castle + NHibernate 项目实战(1)
ExpandedBlockStart.gif
代码

using
 System;

using
 System.Collections.Generic;

using
 System.Linq;

using
 System.Text;


using
 NHibernate;

using
 NHibernate.Cfg;

using
 NHibernate.Criterion;

using
 RoRoWo.Domain;


namespace
 RoRoWo.Data

{

    

public
 
class
 BlogCategoryRespository

    {

        

private
 ISession _session;
        

public
 ISession Session

        {

            

set


            {


                _session 
=
 value;

            }

        }

        
public
 BlogCategoryRespository(ISession session)

        {


            _session 
=
 session;

        }

        
public
 Blogcategory GetById(
int
 cateId)

        {

            
return
 _session.Get
<
Blogcategory
>
(cateId);

        }

        
public
 
void
 Create(Blogcategory dto)

        {


            _session.Save(dto);

            _session.Flush();

        }

        
public
 
int
 CreateAndReturn(Blogcategory dto)

        {

            

int
 newid 
=
 (
int
)_session.Save(dto);

            _session.Flush();
            

return
 newid;

        }

        
///
 
<summary>

        

///
 使用事务
        

///
 
</summary>

        

///
 
<param name=”dto”></param>

        

///
 
<returns></returns>


        
public
 
int
 CreateTransaction(Blogcategory dto)

        {

            

using
 (ITransaction tx 
=
 _session.BeginTransaction())

            {

                

try


                {

                    

int
 newId 
=
 (
int
)_session.Save(dto);

                    _session.Flush();

                    tx.Commit();
                    

return
 newId;

                }
                

catch
 (HibernateException)

                {


                    tx.Rollback();
                    

throw
;

                }

            }

        }

        
public
 
void
 UpdateCustomer(Blogcategory dto)

        {


            _session.Update(dto);

            _session.Flush();

        }

        
public
 
void
 SaveOrUpdate(IList
<
Blogcategory
>
 dtos)

        {

            

foreach
 (var c 
in
 dtos)

            {


                _session.SaveOrUpdate(c);

            }

            _session.Flush();

        }

        
public
 
void
 Delete(Blogcategory dto)

        {


            _session.Delete(dto);

            _session.Flush();

        }

        
public
 IList
<
Blogcategory
>
 From()

        {

            

//
返回所有Blogcategory类的实例


            
return
 _session.CreateQuery(

from Blogcategory

)

                .List
<
Blogcategory
>
();

        }

        
public
 IList
<
int
>
 Select()

        {

            

//
返回所有Blogcategory的CateId


            
return
 _session.CreateQuery(

select c.CateId from Blogcategory c

)

                .List
<
int
>
();

        }

        
public
 IList
<
Blogcategory
>
 Where()

        {

            

return
 _session.CreateQuery(

from Blogcategory c where c.CateId=’3′

)

                .List
<
Blogcategory
>
();

        }

        
public
 IList
<
Blogcategory
>
 GetGreaterThan(
int
 CateId)

        {

            

//
return _session.CreateQuery(“select from Blogcategory c where c.CateId > :cid”)
            

//
    .SetInt32(“cid”, CateId)
            

//
    .List<Blogcategory>();



            

return
 _session.CreateCriteria(
typeof
(Blogcategory))

            .Add(Restrictions.Gt(

CateId

, CateId))

            .List
<
Blogcategory
>
();


        }

        
public
 IList
<
Blogcategory
>
 CreateCriteria()

        {


            ICriteria crit 
=
 _session.CreateCriteria(
typeof
(Blogcategory));

            crit.SetMaxResults(
50
);

            IList
<
Blogcategory
>
 blogcategorys 
=
 crit.List
<
Blogcategory
>
();
            

return
 blogcategorys;

        }

        
public
 IList
<
Blogcategory
>
 Narrowing()

        {


            IList
<
Blogcategory
>
 blogcategorys 
=
 _session.CreateCriteria(
typeof
(Blogcategory))

                .Add(Restrictions.Like(

Catename



s%

))

                .Add(Restrictions.Between(

Parentid



1



3

))

                .List
<
Blogcategory
>
();
            

return
 blogcategorys;

        }

        
public
 IList
<
Blogcategory
>
 Query()

        {


            Blogcategory dtoSample 
=
 
new
 Blogcategory() { Catename 
=
 

sss

, Parentid 
=
 
0
 };
            

return
 _session.CreateCriteria(
typeof
(Blogcategory))

                .Add(Example.Create(dtoSample))

                .List
<
Blogcategory
>
();

        }

        
public
 IList
<
Blogcategory
>
 UseQueryByExample_Get(Blogcategory dtoSample)

        {


            Example example 
=
 Example.Create(dtoSample)

                .IgnoreCase()

                .EnableLike()

                .SetEscapeCharacter(

&

);
            

return
 _session.CreateCriteria(
typeof
(Blogcategory))

               .Add(example)

               .List
<
Blogcategory
>
();

        }


    }

}

 

 

该代码具有了对一个表的基本操作,现在我们就来创建一个单元测试,测试一下插入数据。

在创建插入操作时,我们要能使NHibernate正常工作,还需要创建一个配置文件,我们创建一个XML文件,文件名为“hibernate.cfg.xml”,放在RoRoWo.UnitTest下,其内容为:

[转载]Asp.net MVC 2 + Castle + NHibernate 项目实战(1)
ExpandedBlockStart.gif
代码

<?
xml version=”1.0″ encoding=”utf-8″
?>


<
hibernate-configuration  
xmlns
=”urn:nhibernate-configuration-2.2″
 
>

    

<
session-factory
>

        

<
property 
name
=”connection.driver_class”
>
NHibernate.Driver.SqlClientDriver
</
property
>

        

<
property 
name
=”connection.connection_string”
>


            Data Source=.;Initial Catalog=RoRoWoDB;User ID=sa;Password=2010;
        

</
property
>

        

<
property 
name
=”adonet.batch_size”
>
10
</
property
>

        

<
property 
name
=”show_sql”
>
true
</
property
>

        

<
property 
name
=”dialect”
>
NHibernate.Dialect.MsSql2005Dialect
</
property
>

        

<
property 
name
=”use_outer_join”
>
true
</
property
>

        

<
property 
name
=”command_timeout”
>
10
</
property
>

        

<
property 
name
=”query.substitutions”
>
true 1, false 0, yes ‘Y’, no ‘N’
</
property
>

        

<
property 
name
=”proxyfactory.factory_class”
>


            NHibernate.ByteCode.Castle.ProxyFactoryFactory,

            NHibernate.ByteCode.Castle
        

</
property
>

        

<
mapping 
assembly
=”RoRoWo.Domain”
/>

    

</
session-factory
>


</
hibernate-configuration
>

 

 其中“Data Source=.;Initial Catalog=RoRoWoDB;User ID=sa;Password=2010;”是我的数据库连接,您可以改为您的。

 

现在我们添加一个单元测试(不明白如何添加的请参考相关资料),先测试插入方法,其代码如下:

 ExpandedBlockStart.gif代码

        
///
 
<summary>

        

///
Create 的测试
        

///
</summary>


        [TestMethod()]
        

public
 
void
 CreateTest()

        {


            SessionManager sessionManager 
=
 
new
 SessionManager();

            ISession session 
=
 sessionManager.GetSession();

            BlogCategoryRespository target 
=
 
new
 BlogCategoryRespository(session);

            Blogcategory dto 
=
 
new
 Blogcategory();


            dto.Catename 
=
 

新分类

 
+
 
new
 Random().Next(
100000

999999
).ToString();

            dto.Parentid 
=
 
0
;

            dto.State 
=
 
0
;

            dto.Createtime 
=
 DateTime.Now;

            
int
 newid 
=
 target.CreateAndReturn(dto);

            Assert.IsTrue(newid 
>
 
0
);

        }

 

 执行单元测试,可以通过测试,并且在数据库中已经成功插入了一条记录,结果完全正确;

我们接着对查询方法进行测试,我们将测试代码如下:

 

[转载]Asp.net MVC 2 + Castle + NHibernate 项目实战(1)
ExpandedBlockStart.gif
代码

        
///
 
<summary>

        

///
GetById 的测试
        

///
</summary>


        [TestMethod()]
        

public
 
void
 GetByIdTest()

        {


            SessionManager sessionManager 
=
 
new
 SessionManager();

            ISession session 
=
 sessionManager.GetSession();

            BlogCategoryRespository target 
=
 
new
 BlogCategoryRespository(session);
            

int
 cateId 
=
 
3
;

            Blogcategory actual;

            actual 
=
 target.GetById(cateId);

            Assert.AreEqual(cateId, actual.Cateid);


        }

        
///
 
<summary>

        

///
From 的测试
        

///
</summary>


        [TestMethod()]
        

public
 
void
 FromTest()

        {


            SessionManager sessionManager 
=
 
new
 SessionManager();

            ISession session 
=
 sessionManager.GetSession();

            BlogCategoryRespository target 
=
 
new
 BlogCategoryRespository(session);

            IList
<
Blogcategory
>
 actual;

            actual 
=
 target.From();

            Assert.IsTrue(actual.Count 
>
 
0
);


        }

        
///
 
<summary>

        

///
Select 的测试
        

///
</summary>


        [TestMethod()]
        

public
 
void
 SelectTest()

        {


            SessionManager sessionManager 
=
 
new
 SessionManager();

            ISession session 
=
 sessionManager.GetSession();

            BlogCategoryRespository target 
=
 
new
 BlogCategoryRespository(session);

            IList
<
int
>
 actual;

            actual 
=
 target.Select();

            Assert.IsTrue(actual.Count 
>
 
0
);

        }

        
///
 
<summary>

        

///
Where 的测试
        

///
</summary>


        [TestMethod()]
        

public
 
void
 WhereTest()

        {


            SessionManager sessionManager 
=
 
new
 SessionManager();

            ISession session 
=
 sessionManager.GetSession();

            BlogCategoryRespository target 
=
 
new
 BlogCategoryRespository(session);


            IList
<
Blogcategory
>
 actual;

            actual 
=
 target.Where();

            Assert.IsTrue(actual.Count 
>
 
0
);

        }

        
///
 
<summary>

        

///
GetGreaterThan 的测试
        

///
</summary>


        [TestMethod()]
        

public
 
void
 GetGreaterThanTest()

        {


            SessionManager sessionManager 
=
 
new
 SessionManager();

            ISession session 
=
 sessionManager.GetSession();

            BlogCategoryRespository target 
=
 
new
 BlogCategoryRespository(session);
            

int
 CateId 
=
 
0

//
 TODO: 初始化为适当的值


            IList
<
Blogcategory
>
 actual;

            actual 
=
 target.GetGreaterThan(CateId);

            Assert.IsTrue(actual.Count 
>
 
0
);


        }

 

 

 测试结果全部绿灯通过:

 [转载]Asp.net MVC 2 + Castle + NHibernate 项目实战(1)

下一篇,我将谈谈如何配置一对多和多对多的表映射关系。

 

该例子的完整代码可以到开源平台下载,地址: http://rorowoproject.codeplex.com

文章中如有不足之处,望大家多指正。

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。

发布者:全栈程序员-用户IM,转载请注明出处:https://javaforall.cn/110505.html原文链接:https://javaforall.cn

【正版授权,激活自己账号】: Jetbrains全家桶Ide使用,1年售后保障,每天仅需1毛

【官方授权 正版激活】: 官方授权 正版激活 支持Jetbrains家族下所有IDE 使用个人JB账号...

(0)
blank

相关推荐

  • 蓝牙中的sbc_蓝牙sbc格式和aac差别

    蓝牙中的sbc_蓝牙sbc格式和aac差别在之前《小米的试用哲学:小米AirDots青春版和Redmi红米AirDots无线耳机体验》中,我提到了这两款耳机都支持aptXHD。不过在这两款耳机上,也仅仅支持了aptXHD这个技术,也需要配合高通骁龙手机使用,iOS也就别想了……经过实际测试,这两款耳机均不支持aptXHD技术,在此更正,并对造成的误解致歉!这两款耳机支持SBC、AAC两种编码。小米AirDots青春版和Redmi红…

  • 解决touchStart滑动时触发点击事件「建议收藏」

    解决touchStart滑动时触发点击事件「建议收藏」由于我们在将点击事件委托到div,span,font,p等元素标签上时会导致移动端的ios设备无法识别这个点击事件(一般指click事件),此时我们一般会使用touchstart来替换click事件,但是使用touchstart来代替click事件的弊端就立刻显示出来,每当我们滑动屏幕时就会立刻出发点击事件此时加入我这个封装的方法,这个弊端就会迎刃而解functiontap(el,fn…

  • tomcat宕机自动重启和每日定时启动tomcat

    tomcat宕机自动重启和每日定时启动tomcat在项目后期维护中会遇到这样的情况,tomcat在内存溢出的时候就出现死机的情况和遇到长时间不响应,需要人工手动关闭和重启服务,针对这样的突发情况,希望程序能自动处理问题而不需要人工关于,所以才有了目前的需求。一、设置tomcat定时启动1,首先将tomcat注册为服务,先打开tomcat的bin目录下service.bat文件,修改下面的值,这是sevvice的注册名称和显

  • 视频要不要开hdr_hdr在什么情况下使用

    视频要不要开hdr_hdr在什么情况下使用最近两年HDR这个概念可谓是铺天盖地而来,手机也好PC也好电视也好,都拼命往自己头上扣HDR的帽子。而在某些发烧友眼中,如果看片子不带HDR,堪比步兵变骑兵,一下子变得索然无味。然而,新事物往往也伴随着众多新坑,特别是在软硬件环境复杂的PC平台,稍有不慎就会摔得脸青鼻肿,播HDR的效果甚至不如播普通的片子。PC播HDR的大坑有几何?PC并不是专门为视频播放设计的机器,和专业的蓝光机等播放器相比…

  • vdbench配置说明[通俗易懂]

    vdbench配置说明[通俗易懂]一、前言1、介绍 vdbench是一个I/O工作负载生成器,通常用于验证数据完整性和度量直接附加(或网络连接)存储性能。它可以运行在windows、linux环境,可用于测试文件系统或块设备基准性能。2、版本包获取vdbench版本包下载链接:http://download.oracle.com/otn/utilities_drivers/vdbench/vdbench50406.zipjdk版本包下载链接(windows):https://download.oracle.com/

  • Spring Cloud-鸿鹄Cloud分布式微服务云系统—架构图[通俗易懂]

    Spring Cloud-鸿鹄Cloud分布式微服务云系统—架构图

发表回复

您的电子邮箱地址不会被公开。

关注全栈程序员社区公众号