NHibernate也是能够操作存储过程的,不过第一次配置可能会碰到很多错误。

一、删除

  首先,我们新建一个存储过程如下:

  CREATE PROC DeletePerson
  @Id int
  AS
  DELETE FROM Person WHERE PersonId = @Id;

  修改映射文件,添加删除对象的存储过程:

<?xml version="1.0" encoding="utf-8" ?>
<hibernate-mapping xmlns="urn:nhibernate-mapping-2.2">
<class name="Model.PersonModel, Model" table="Person">
<id name="PersonId" column="PersonId" type="Int32">
<generator class="native"/>
</id>
<property name="PersonName" column="PersonName" type="String"/>
<!--多对一关系:Person属于一个Country name是Person实体类里的-->
<many-to-one name="Country" column="CountryId" not-null="true" class="Model.CountryModel,Model" foreign-key="FK_Person_Country" />
<!-- 一个Country里面有多个Person -->
<set name="ListChild" table="Child" generic="true">
<key column="ParentId" foreign-key="FK_Child_Person"/>
<one-to-many class="Model.ChildModel,Model"/>
</set>
<sql-delete>DeletePerson ?</sql-delete>
</class>
</hibernate-mapping>

  执行存代码:

            using (ISession session = sessionFactory.OpenSession())
{
PersonModel p = session.Get<PersonModel>();
session.Delete(p);
session.Flush();
}

  从监控到,SQLServer执行的语句如下:

exec sp_executesql N'DeletePerson @p0',N'@p0 int',@p0=5

  明显已经看到哥写的存储过程名字。如果不是执行存储过程,则执行的语句如下:

exec sp_executesql N'DELETE FROM Person WHERE PersonId = @p0',N'@p0 int',@p0=5

  可以看到,如果你配置了存储过程,那么Delete()方法就执行存储过程,否则就执行NHibernate自己生成的SQL语句。

二、添加

   测试了一个NHibernate的添加存储过程之后,感觉就一个字,不爽,估计这种方式配置的存储也不会是好的选择,还是生成语句比较好。

  存储过程如下:

ALTER PROC InsertCountry
@CountryName nvarchar(50),
@CountryId int OUTPUT
AS
INSERT INTO Country VALUES(@CountryName);

  首先,添加将CountryId设置为自增的。

  然后配置文件如下(注意两个加粗的地方):

<?xml version="1.0" encoding="utf-8" ?>
<hibernate-mapping xmlns="urn:nhibernate-mapping-2.2">
<class name="Model.CountryModel,Model" table="Country">
<id name="CountryId" column="CountryId" type="Int32">
<generator class="increment"/>
</id>
<property name="CountryName" column="CountryName" type="String"/>
<!-- 一个Country里面有多个Person -->
<set name="ListPerson" table="Person" generic="true">
<key column="CountryId" foreign-key="FK_Person_Country"/>
<one-to-many class="Model.PersonModel,Model"/>
</set>
<sql-insert>InsertCountry ?,?</sql-insert>
</class>
</hibernate-mapping>

  设置increment的目的是主键,本处为CountryId由NHibernate生成。实际上不是一个什么好的方法,因为NHibernate是通过SQL语句:

SELECT MAX(CountryId) FROM Country

  获得的。并且,你还必须设置数据库主键自增。一句话,不爽。都说了没什么人会选择这种方式咯。

  执行代码如下:

        static void Main(string[] args)
{
ISessionFactory sessionFactory = new Configuration().Configure().BuildSessionFactory(); using (ISession session = sessionFactory.OpenSession())
{
CountryModel c = new CountryModel();
c.CountryName = "青州";
session.Save(c);
session.Flush();
}
Console.ReadKey();
}

  监控到SQL Server执行了:

exec sp_executesql N'InsertCountry @p0,@p1',N'@p0 nvarchar(4000),@p1 int',@p0=N'青州',@p1=8

三、更新

  更新的方式与Insert类似,不过好点过Insert了,还不错。

  存储过程:

CREATE PROC UpdateCountry
@CountryName nvarchar(50),
@CountryId int OUTPUT
AS
UPDATE Country SET CountryName = @CountryName WHERE CountryId = @CountryId;

  映射文件:

<?xml version="1.0" encoding="utf-8" ?>
<hibernate-mapping xmlns="urn:nhibernate-mapping-2.2">
<class name="Model.CountryModel,Model" table="Country">
<id name="CountryId" column="CountryId" type="Int32">
<generator class="increment"/>
</id>
<property name="CountryName" column="CountryName" type="String"/>
<!-- 一个Country里面有多个Person -->
<set name="ListPerson" table="Person" generic="true">
<key column="CountryId" foreign-key="FK_Person_Country"/>
<one-to-many class="Model.PersonModel,Model"/>
</set>
<sql-insert>InsertCountry ?,?</sql-insert>
<sql-update>UpdateCountry ?,?</sql-update>
</class>
</hibernate-mapping>

  执行代码如下:

        static void Main(string[] args)
{
ISessionFactory sessionFactory = new Configuration().Configure().BuildSessionFactory(); using (ISession session = sessionFactory.OpenSession())
{
CountryModel c = session.Get<CountryModel>();
c.CountryName = "修改测试";
session.Update(c);
session.Flush();
} Console.ReadKey();
}

  监控到SQL执行的语句如下:

exec sp_executesql N'UpdateCountry @p0,@p1',N'@p0 nvarchar(4000),@p1 int',@p0=N'修改测试',@p1=4

四、查询

  首先,创建一个存储过程如下:

CREATE PROC EntityCountry
@CountryId int
AS
SELECT * FROM Country WHERE CountryId =@CountryId

  在NHibernate中,通过命名查询来映射存储过程,使用return返回具体的实体类,使用<return-property>告诉NHibernate使用哪些属性值,允许我们来选择如何引用字段以及属性。

  映射文件如下:

<?xml version="1.0" encoding="utf-8" ?>
<hibernate-mapping xmlns="urn:nhibernate-mapping-2.2">
<class name="Model.CountryModel,Model" table="Country">
<id name="CountryId" column="CountryId" type="Int32">
<generator class="increment"/>
</id>
<property name="CountryName" column="CountryName" type="String"/>
<!-- 一个Country里面有多个Person -->
<set name="ListPerson" table="Person" generic="true">
<key column="CountryId" foreign-key="FK_Person_Country"/>
<one-to-many class="Model.PersonModel,Model"/>
</set>
</class>
<sql-query name="Cc">
<return class="Model.CountryModel,Model" />
EXEC EntityCountry :CountryId
</sql-query>

</hibernate-mapping>

  执行操作:

        static void Main(string[] args)
{
ISessionFactory sessionFactory = new Configuration().Configure().BuildSessionFactory(); using (ISession session = sessionFactory.OpenSession())
{
CountryModel c = session.GetNamedQuery("Cc").SetInt32("CountryId", ).UniqueResult<CountryModel>();
Console.WriteLine(c.CountryName);
} Console.ReadKey();
}

  监控到SQL Server执行的语句如下:

exec sp_executesql N'EXEC EntityCountry @p0',N'@p0 int',@p0=1

  考虑到在NHibernate中使用存储过程并不是一个很好的选择,也无谓学习得太深入了,这个真心不想用。

五、总结

  <sql-query>配置在<class>后面。

  其他增删改要遵循以下顺序:

  •   <sql-insert>
  •   <sql-update>
  •   <sql-delete>

  OK,NHibernate执行存储过程就这么多了,以后用到的时候,再更新。

NHibernate 存储过程 第十四篇的更多相关文章

  1. 解剖SQLSERVER 第十四篇 Vardecimals 存储格式揭秘(译)

    解剖SQLSERVER 第十四篇    Vardecimals 存储格式揭秘(译) http://improve.dk/how-are-vardecimals-stored/ 在这篇文章,我将深入研究 ...

  2. 第十四篇 Integration Services:项目转换

    本篇文章是Integration Services系列的第十四篇,详细内容请参考原文. 简介在前一篇,我们查看了SSIS变量,变量配置和表达式管理动态值.在这一篇,我们使用SQL Server数据商业 ...

  3. Python之路【第十四篇】:AngularJS --暂无内容-待更新

    Python之路[第十四篇]:AngularJS --暂无内容-待更新

  4. 【译】第十四篇 Integration Services:项目转换

    本篇文章是Integration Services系列的第十四篇,详细内容请参考原文. 简介在前一篇,我们查看了SSIS变量,变量配置和表达式管理动态值.在这一篇,我们使用SQL Server数据商业 ...

  5. 跟我学SpringCloud | 第十四篇:Spring Cloud Gateway高级应用

    SpringCloud系列教程 | 第十四篇:Spring Cloud Gateway高级应用 Springboot: 2.1.6.RELEASE SpringCloud: Greenwich.SR1 ...

  6. SpringBoot第二十四篇:应用监控之Admin

    作者:追梦1819 原文:https://www.cnblogs.com/yanfei1819/p/11457867.html 版权声明:本文为博主原创文章,转载请附上博文链接! 引言   前一章(S ...

  7. Egret入门学习日记 --- 第十四篇(书中 5.4~5.6节 内容)

    第十四篇(书中 5.4~5.6节 内容) 书中内容: 总结 5.4节 内容重点: 1.如何编写自定义组件? 跟着做: 重点1:如何编写自定义组件? 文中提到了重要的两点. 好,我们来试试看. 第一步, ...

  8. Spring Cloud第十四篇 | Api网关Zuul

    ​ 本文是Spring Cloud专栏的第十四篇文章,了解前十三篇文章内容有助于更好的理解本文: Spring Cloud第一篇 | Spring Cloud前言及其常用组件介绍概览 Spring C ...

  9. Hibernate(十四篇)

    (一)Hibernate简介 (二)hibernate配置管理 (三)Hibernate对象-关系映射文件 (四)Hibernate API详解 (五)Hibernate一级缓存 (六)Hiberna ...

随机推荐

  1. Linux(Unix)密码策略问题导致root密码不能修改

    Linux(Unix)密码策略问题导致root密码不能修改 发布时间:  2016-01-19 浏览次数:  1034 下载次数:  5 用户修改了密码配置文件,导致root账户修改密码时报如下错误: ...

  2. bzoj 1179 Atm

    题目链接:http://www.lydsy.com/JudgeOnline/problem.php?id=1179 题解: 一道比较综合的图论题 直接讲正解: 如果这个图G中存在某个强连通分量,那么这 ...

  3. EOS.IO技术学习

    如今很火的项目EOS的学习,以下主要的内容是基于白皮书 参考: http://chainx.org/paper/index/index/id/20.html EOS.IO软件引入了一种新的块链架构,旨 ...

  4. API(选项/数据 选项/dom)

    选项/数据 data 类型: Object | Function 限制: 组件的定义只接受function var data = { a: 1 } // 直接创建一个实例 var vm = new V ...

  5. Github精选 – 适合移动端的HTML5 Datepicker

     2016-01-12 (updated: 2017-01-07) 15731 通过 HTML5 的 <input> 新属性可以简单方便地调用手机的原生 Datepicker,但功能较弱, ...

  6. Python+Selenium 自动化实现实例-打开浏览器模拟进行搜索数据并验证

    #导入模块 from selenium import webdriverfrom selenium.webdriver.common.keys import Keys #启动火狐浏览器driver = ...

  7. 自定义wordCount程序、

    1.MyWordCount代码: package com.hadoop.mr; import java.io.IOException; import org.apache.hadoop.conf.Co ...

  8. [水煮 ASP.NET Web API2 方法论](1-6)Model Validation

    问题 想要 ASP.NET Web API 执行模型验证,同时可以和 ASP.NET MVC 共享一些验证逻辑. 解决方案 ASP.NET Web API 与 ASP.NET MVC 支持一样的验证机 ...

  9. Django2.x版本路由系统的正则写法以及视图函数的返回问题

    一.关于url.py urlpatterns每个元素的不再用url(),而是path(),最重要的一点是,正则的使用需要你自己手动导入re_path,并且在每个使用正则匹配的的元素用re_path() ...

  10. bzoj 1485 卡特兰数 + 分解因子

    思路:打表可以看出是卡特兰数,但是模数不一定是素数,所以需要分解一下因数. #include<bits/stdc++.h> #define LL long long #define fi ...