目录
写在前面
前面的文章介绍了在nhibernate中使用存储过程进行增删改的操作,当然查询也是可以的,在nhibernate中也可以执行任意的存储过程。本篇文章将介绍如何使用查询的存储过程的方式。
文档与系列文章
[NHibernate]持久化类(Persistent Classes)
[NHibernate]集合类(Collections)映射
[NHibernate]缓存(NHibernate.Caches)
[NHibernate]NHibernate.Tool.hbm2net
[NHibernate]Nhibernate如何映射sqlserver中image字段
[NHibernate]条件查询Criteria Query
查询
使用节点<sql-query>根据用户id进行查询
添加存储过程
create proc [dbo].[ps_Search]
@CustomerID uniqueidentifier
as
begin
select * from TB_Customer
where CustomerID=@CustomerID
end
在映射文件中使用<sql-query>并定义<sql-query>查询的名称
<!--需要和class节点同一级别-->
<sql-query name="ps_Search" >
<return class="Wolfy.Shop.Domain.Entities.Customer,Wolfy.Shop.Domain" />
exec ps_Search :CustomerID
</sql-query>
测试
在数据访问层中,使用ISession接口提供的GetNamedQuery方法来调用带命名的存储过程,并传递一个整形参数。代码如下:
/// <summary>
/// 使用存储过程,进行查询
/// </summary>
/// <returns></returns>
public IList<Customer> SearchCustomerByIDUsingProc(Guid customerID)
{
ISession session = NHibernateHelper.GetSession();
//参数为映射文件中为<sql-query>节点指定的name return session.GetNamedQuery("ps_Search")
.SetGuid("CustomerID", customerID)
.List<Customer>();
}
结果
生成的sql语句
exec sp_executesql N'exec ps_Search @p0',N'@p0 uniqueidentifier',@p0='DDF63750-3307-461B-B96A-7FF356540CB8'
如果就想返回一个实体的一部分属性怎么办?
修改存储过程,只选择需要的字段
ALTER proc [dbo].[ps_Search]
@CustomerID uniqueidentifier
as
begin
select CustomerName from TB_Customer
where CustomerID=@CustomerID
end
修改映射文件
<!--需要和class节点同一级别-->
<sql-query name="ps_Search" >
<!--<return class="Wolfy.Shop.Domain.Entities.Customer,Wolfy.Shop.Domain" />-->
<return-scalar column="CustomerName" type="String"/>
exec ps_Search :CustomerID
</sql-query>
修改方法
/// <summary>
/// 使用存储过程,进行查询
/// </summary>
/// <returns></returns>
public string SearchCustomerNameUsingProc(Guid customerID)
{
ISession session = NHibernateHelper.GetSession();
//参数为映射文件中为<sql-query>节点指定的name
return session.GetNamedQuery("ps_Search")
.SetGuid("CustomerID", customerID).UniqueResult().ToString();
}
结果
因为执行的是存储过程,sql语句与上面相同(在原来的存储过程基础上修改的)。
最常见的bug
遇到这样的异常,可以忽视它,我找了很久也没找到解决方案,然后就直接.List<Customer>()将结果返回了。虽然得到结果了总有那么点不太满意(完美主义者)。
总结
这篇文章介绍了nhibernate中如何使用查询存储过程的方法及需注意的地方,比如映射文件中<sql-query>节点需和<class>同级别才行。到这里nhibernate中使用存储过程增删改查就算结束了,说个题外话,使用orm为了少写sql,所以在实际项目中使用nhibernate的存储过程比较少见。就算用也没有直接在映射文件中添加<sql-**>的方式。因为你知道,不一定别人就知道,所以造成一定的维护难度,不过知道这种高大上的写法,遇到的时候知道咋回事就ok了。
参考文章:http://www.cnblogs.com/lyj/archive/2008/11/07/1328782.html