[转]NHibernate之旅(5):探索Insert, Update, Delete操作

本节内容

  • 操作数据概述
  • 1.新建对象
  • 2.删除对象
  • 3.更新对象
  • 4.保存更新对象
  • 结语

操作数据概述

我们常常所说的一个工作单元,通常是执行1个或多个操作,对这些操作要么提交要么放弃/回滚。想想使用LINQ to SQL,一切的东西都在内存中操作,只有调用了DataContext.SubmitChanges()方法才把这些改变的数据提交到数据库中,LINQ to SQL那么提交要么回滚。

我们使用NHibernate也一样,如果只查询数据,不改变它的值,就不需要提交(或者回滚)到数据库。

注意:这节,我们在上一节源代码的基础上,在数据访问层中新建CRUD.cs类用于编写操作方法,在数据访问的测试层新建一CRUDFixture.cs类用于测试。

1.新建对象

简单描述:新建一个对象;调用ISession.Save();同步ISession。

例子:在数据访问层编写CreateCustomer()方法,把传过来的Customer对象保存在数据库中。

public int CreateCustomer(Customer customer)
{
    int newid = (int)_session.Save(customer);
    _session.Flush();
    return newid;
}

我们测试这个方法,新建一个Customer对象,调用CreateCustomer()方法返回新插入的CustomerId,再次根据CustomerId查询数据库是否存在这个对象。

[Test]
public void CreateCustomerTest()
{
    var customer = new Customer() { Firstname = "YJing", Lastname = "Lee" };
    int newIdentity = _crud.CreateCustomer(customer);
    var testCustomer = _crud.GetCustomerById(newIdentity);
    Assert.IsNotNull(testCustomer);
}

2.删除对象

简单描述:获取一个对象;调用ISession.Delete();同步ISession。

说明:使用ISession.Delete()会把对象的状态从数据库中移除。当然,你的应用程序可能仍然持有一个指向它的引用。所以,最好这样理解:Delete()的用途是把一个持久化实例变成临时实例。 你也可以通过传递给Delete()一个NHibernate 查询字符串来一次性删除很多对象。删除对象顺序没有要求,不会引发外键约束冲突。当然,有可能引发在外键字段定义的NOT NULL约束冲突。

例子:在数据访问层编写DeleteCustomer()方法,从数据库中删除Customer对象。

public void DeleteCustomer(Customer customer)
{
    _session.Delete(customer);
    _session.Flush();
}

我们测试这个方法,在数据库中查询CustomerId为2的Customer对象,调用DeleteCustomer()方法删除,再次根据CustomerId查询数据库是否存在这个对象。

[Test]
public void DeleteCustomerTest()
{
    var coutomer = _crud.GetCustomerById(2);
    _crud.DeleteCustomer(coutomer);
    var testCustomer = _crud.GetCustomerById(2);
    Assert.IsNull(testCustomer);
}

3.更新对象

简单描述:获取一个对象;改变它的一些属性;调用ISession.Update();同步ISession。

例子:在数据访问层编写UpdateCustomer()方法,修改Customer对象。

public void UpdateCustomer(Customer customer)
{
    _session.Update(customer);
    _session.Flush();
}

测试这个方法,在数据库中查询CustomerId为1的Customer对象并修改它的Firstname属性值,调用UpdateCustomer()方法更新,再次查询数据库中CustomerId为1的Customer对象的Firstname值为修改之后的值。

[Test]
public void UpdateCustomerTest()
{
    var customer = _crud.GetCustomerById(1);
    customer.Firstname = "liyongjing";
    _crud.UpdateCustomer(customer);
    var testCustomer = _crud.GetCustomerById(1);
    Assert.AreEqual("liyongjing", customer.Firstname);
}

4.保存更新对象

你会不会想出这个问题?哪些是刚刚创建的对象,哪些是修改过的对象?对于刚刚创建的对象我们需要保存到数据库中,对于修改过的对象我们需要更新到数据库中。

幸好,ISession可以识别出这不同的对象,并为我们提供了ISession.SaveOrUpdate(object)方法

ISession.SaveOrUpdate(object)方法完成如下工作:

  • 检查这个对象是否已经存在Session中。
  • 如果对象不在,调用Save(object)来保存。
  • 如果对象存在,检查这个对象是否改变了。
  • 如果对象改变,调用Update(object)来更新。

看看下面例子说明了这种情况,在数据访问层编写SaveOrUpdateCustomer()方法,保存更新Customer对象列表,依次遍历列表中的Customer对象,调用ISession.SaveOrUpdate(object)方法保存更新每个Customer对象。

public void SaveOrUpdateCustomer(IList<Customer> customer)
{
    foreach (var c in customer)
    {
        _session.SaveOrUpdate(c);
    }
    _session.Flush();
}

测试这个方法,先在数据库中查询Firstname为YJing的Customer对象并修改它的Lastname属性值,这些对象是数据库中存在的,并改变了,然后新建2个Customer对象,这两个对象在数据库中不存在,是新创建的。调用SaveOrUpdateCustomer()方法保存更新对象,即更新前面修改的对象和保存了后面新创建的2个对象。再次查询数据库中Firstname为YJing,Lastname为YongJing的Customer对象是否一致了。

[Test]
public void SaveOrUpdateCustomerTest()
{
    IList<Customer> customers = _crud.GetCustomersByFirstname("YJing");
    foreach (var c in customers)
    {
        c.Lastname = "YongJing";
    }
    var c1 = new Customer() { Firstname = "YJing", Lastname = "YongJing"};
    var c2 = new Customer() { Firstname = "YJing", Lastname = "YongJing"};
    customers.Add(c1);
    customers.Add(c2);
    int initiaIListCount = customers.Count;

    _crud.SaveOrUpdateCustomer(customers);

    int testListCount = _crud.GetCustomersByFirstnameAndLastname("YJing", "YongJing").Count;
    Assert.AreEqual(initiaIListCount, testListCount);
}

结语

当然,这一节操纵对象操作,在NHibernate中涉及了对象的状态, 对象对一个特定的ISession来说,有三种状态分别是:瞬时(transient)对象、持久化(persistent)对象、游离(detached)对象。这一节没有说到了,以后在讨论Session的时候再介绍。

时间: 2024-11-08 20:29:11

[转]NHibernate之旅(5):探索Insert, Update, Delete操作的相关文章

LINQ体验(9)——LINQ to SQL语句之Insert/Update/Delete操作

我们继续讲解LINQ to SQL语句,这篇我们来讨论Insert/Update/Delete操作.这个在我们的程序中最为常用了.我们直接看例子. Insert/Update/Delete操作 插入(Insert) 1.简单形式 说明:new一个对象,使用InsertOnSubmit方法将其加入到对应的集合中,使用SubmitChanges()提交到数据库. NorthwindDataContext db = new NorthwindDataContext(); var newCustomer

数据库编程3 Oracle 子查询 insert update delete 事务 回收站 字段操作 企业级项目案例

[本文谢绝转载原文来自http://990487026.blog.51cto.com] <大纲> 数据库编程3 Oracle 子查询 insert update delete 事务 回收站 字段操作 企业级项目案例 实验所用数据表 子查询,解决一步不能求解 查询工资比scott高的员工信息: 子查询知识体系搭建: 解释3,查询部门是sales的员工信息: 方法1:子查询 [方法2]:多表: 优化考虑: 解释4[select],只能放单行子查询 解释4[from] 考题:显示员工姓名,薪水 解释

linux下mysql Insert update delete 事务 用户管理

linux下mysql Insert update delete  事务 用户管理 1.INSERT插入语句格式: INSERT INTO tb_name (字段1, 字段2, ...) VALUES (值1,值2, ...)[,(值1, 值2, ...),...]; INSERT INTO 表名 SET 字段1=值1,字段2=值2,...; INSERT INTO 表名 (字段1,字段2,...) SELECT (字段1,字段2,...) FROM 表名 [WHERE 条件]; 2.REPLA

mybatis insert update delete返回都是整型 0,1,增,删,改要提交事物

mybatis insert update delete返回都是整型 0,1, 没有扔 增,删,改要提交事物 原文地址:https://www.cnblogs.com/gzhbk/p/9499293.html

元素类型为 &quot;mapper&quot; 的内容必须匹配 &quot;(cache-ref|cache|resultMap*|parameterMap*|sql*|insert*|update*|delete*|select)

在配置ssm框架的时候,写mapper映射文件的时候会出现 元素类型为 "mapper" 的内容必须匹配 "(cache-ref|cache|resultMap*|parameterMap*|sql*|insert*|update*|delete*|select) 有时候编译的时候会出现这个bug,是因为在前面写了注释.导致了执行顺序的问题.需要把一些注释删掉,就可以正常执行了. /*mapper动态开发*/ <mapper namespace="com.ss

sqlserver 大数据量的insert、delete操作优化

http://blog.csdn.net/lanyuzhen/article/details/7547476 --大批量导出orders表:insert DBCC DROPCLEANBUFFERS  DBCC FREEPROCCACHE goSET NOCOUNT ON BEGIN TRANSACTION  INSERT INTO test.dbo.orders with(tablock) SELECT * FROM bak.dbo.OrdersWHERE ordertime BETWEEN '

mysql之insert,update,delete

测试数据 1.product表 CREATE table product( id INT(10) PRIMARY KEY NOT NULL, name VARCHAR(20) NOT NULL, function VARCHAR(50) DEFAULT NULL , company VARCHAR(20) NOT NULL, address VARCHAR(50) DEFAULT NULL ); 1.medicine表 CREATE table medicine( id INT(10) PRIM

oracle中execute immediate的使用(select/insert/update/delete)(转)

execute immediate的语法如下: execute immediate 'sql'; execute immediate 'sql_select' into var_1, var_2; execute immediate 'sql' using [in|out|in out] bind_var_1, [in|out|in out] bind_var_2; execute immediate 'sql_select' into var_1, var_2 using [in|out|in

MySQL之DML语句(insert update delete)

DML主要针对数据库表对象的数据而言的,一般DML完成: 插入新数据 修改已添加的数据 删除不需要的数据 1.insert into插入语句 //主键自增可以不插入,所以用null代替 insert into temp values(null, 'jack', 25); //指定列 insert into temp(name, age) values('jack', 22); 在表面后面带括号,括号中写列名,values中写指定列名的值即可.当省略列名就表示插入全部数据,注意插入值的顺序和列的顺