Entity Framework 五

连接情景中的CRUD操作:

连接场景中的CRUD操作是一项相当简单的任务,因为默认情况下,上下文会自动跟踪实体在其生命周期中发生的更改,前提是AutoDetectChangesEnabled为true。

以下示例显示了如何在连接方案(在上下文的范围内)中添加,更新和删除实体,然后该实体又将在数据库上执行插入,更新和删除命令。上下文将自动检测更改并更新实体的状态

  

 

断开的实体:

   当我们得到一个断开的实体图,甚至一个断开的实体时,我们需要做两件事情。首先,我们需要使用新的上下文实例附加实体并使上下文知道这些实体其次,手动为这些实体设置合适的EntityStates,因为新的上下文实例不知道在断开的实体上执行的操作,所以新的上下文不能应用适当的EntityState。

示意图:

DbSet.Add():

DbSet.Add()方法将整个实体图形附加到新的上下文,并自动将添加的实体状态应用于所有实体

 

使用父DbSet实体的Add方法将整个实体图附加到具有Added状态的新的上下文实例。调用ctx.Savechanges()将为所有实体执行插入命令,这会将新行插入到适当的数据库表中。

 

DbSet.Attach():

DbSet.Attach方法将整个实体图形附加到具有不变实体状态的新上下文中

 

 Attach方法只会将实体图附加到上下文中,所以我们需要为每个实体找到适当的实体状态并手动应用。

 

DbContext.Entry():

DbContext的Entry方法返回指定实体的DbEntityEntry实例。DbEntityEntry可以用来改变一个实体的状态

 

 

在断开的情况下使用DBContext添加新实体:

 

 

在断开的情况下使用DBContext更新现有实体:

 

 

步骤:

从数据库获取现有的学生。

将学生姓名改为上下文范围(断开模式)

将修改的实体传递给Entry方法以获取其DBEntityEntry对象,然后将其状态标记为Modified

调用SaveChanges()方法将学生信息更新到数据库中。

 

在断开的情况下使用DBContext删除实体:

 

使用DbContext添加实体图:

我们可以使用DbSet.Add()方法将整个实体图形附加到上下文中,并将所有实体的状态设置为Added。

 

使用DbContext更新实体图:

在断开的情况下更新实体图是一项复杂的任务

在断开的场景中更新实体图形的问题在于,上下文不知道在客户端执行了哪些操作。根据下图,新的上下文不知道每个实体的状态:

 

在断开的情况下识别实体状态的模式

在断开的情况下,有几种方法(如下所示)来标识实体状态:

  1. 使用实体的PrimaryKey(Id)属性
  2. 在实体集中设置一个属性来保存状态 (学习的时候可以这样设计)

1)使用实体的PrimaryKey属性:

您可以使用每个实体的PrimaryKey(Id)属性来确定其实体状态。但是,您必须决定使用以下哪个架构规则:

  • 每种类型的实体都必须有Id属性(PK)
  • Id属性的默认值应为0

在断开的情况下,context2不知道每个实体的状态。它必须通过使用StandardId来确定Standard实体的状态,并使用TeacherId属性来确定Teacher实体的状态。如果StandardId和TeacherID的值为零,则表示它是一个新的实体,如果它不是零,那么它是一个修改的实体。(这个比较好理解)

 

优点:

  • 无需额外的编码/处理来确定实体状态。
  • 很好的表现。

缺点:

  • 每个实体类型都需要有一个Id属性。它不能确定没有Id属性的实体的状态。
  • 无法识别未更改的实体。即使实体没有被改变,它也被设置为Modified状态。因此,对于未更改的实体存在不必要的数据库更新语句。
  • 无法处理删除实体场景。它需要单独处理删除。

2)在每个实体拥有状态:

①创建一个接口,声明一个枚举

②让实体继承改接口

 

 

上面的代码修改了一个Teacher,新建了一个Teacher,和新建了一个Standar

 

优点:

  • 无需额外的编码/处理来确定实体状态
  • 已添加,已修改,已删除和未更改状态正确
  • 没有不必要的更新调用不变的实体。

坏处:

  • 需要在断开模式下设置每个实体的适当状态。所以在断线模式下需要格外小心。

 

实体框架中的并发:

  SQL里面处理并发则是使用多线程来处理,对有并发的数据实现同步,只能一个一个的去访问数据。再就是使用锁,在有人访问这个数据的时候别人是访问不了的。

①Student表里面有个RowVersion字段是 timestamp(时间戳)类型,在添加,更新的时候会自动变化

②打开设计器修改Student实体中的RowVersion属性,更改并发模式

 

 

③测试并发

   数据库中只修改了一次

 

至于怎么去处理并发这里就不做深入研究了。 

 

转载于:https://www.cnblogs.com/Sea1ee/p/7878539.html

创建了一个使用实体框架和SQL Server Express LocalDB来存储和显示数据的简单Web应用程序. 实现了基本的CRUD和排序、筛选、分页及分组功能. using System; using System.Collections.Generic; using System.Data; using System.Data.Entity; using System.Linq; using System.Net; using System.Web; using System.Web.Mvc; using ContosoUniversity.DAL; using ContosoUniversity.Models; using PagedList; namespace ContosoUniversity.Controllers { public class StudentsController : Controller { private SchoolContext db = new SchoolContext(); // GET: Students //public ActionResult Index() //{ // return View(db.Students.ToList()); //} /* step1 public ActionResult Index(string sortOrder) { ViewBag.NameSortParm = String.IsNullOrEmpty(sortOrder) ? "name_desc" : ""; ViewBag.DateSortParm = sortOrder == "Date" ? "date_desc" : "Date"; var students = from s in db.Students select s; switch (sortOrder) { case "name_desc": students = students.OrderByDescending(s => s.LastName); break; case "Date": students = students.OrderBy(s => s.EnrollmentDate); break; case "date_desc": students = students.OrderByDescending(s => s.EnrollmentDate); break; default: students = students.OrderBy(s => s.LastName); break; } return View(students.ToList()); } */ /*step2 public ViewResult Index(string sortOrder, string searchString) { ViewBag.NameSortParm = String.IsNullOrEmpty(sortOrder) ? "name_desc" : ""; ViewBag.DateSortParm = sortOrder == "Date" ? "date_desc" : "Date"; var students = from s in db.Students select s; if (!String.IsNullOrEmpty(searchString)) { students = students.Where(s => s.LastName.Contains(searchString) || s.FirstMidName.Contains(searchString)); } switch (sortOrder) { case "name_desc": students = students.OrderByDescending(s => s.LastName); break; case "Date": students = students.OrderBy(s => s.EnrollmentDate); break; case "date_desc": students = students.OrderByDescending(s => s.EnrollmentDate); break; default: students = students.OrderBy(s => s.LastName); break; } return View(students.ToList()); } */ public ViewResult Index(string sortOrder, string currentFilter, string searchString, int? page) { ViewBag.CurrentSort = sortOrder; ViewBag.NameSortParm = String.IsNullOrEmpty(sortOrder) ? "name_desc" : ""; ViewBag.DateSortParm = sortOrder == "Date" ? "date_desc" : "Date"; if (searchString != null) { page = 1; } else { searchString = currentFilter; } ViewBag.CurrentFilter = searchString; var students = from s in db.Students select s; if (!String.IsNullOrEmpty(searchString)) { students = students.Where(s => s.LastName.Contains(searchString) || s.FirstMidName.Contains(searchString)); } switch (sortOrder) { case "name_desc": students = students.OrderByDescending(s => s.LastName); break; case "Date": students = students.OrderBy(s => s.EnrollmentDate); break; case "date_desc": students = students.OrderByDescending(s => s.EnrollmentDate); break; default: // Name ascending students = students.OrderBy(s => s.LastName); break; } int pageSize = 6; int pageNumber = (page ?? 1); return View(students.ToPagedList(pageNumber, pageSize)); } // GET: Students/Details/5 public ActionResult Details(int? id) { if (id == null) { return new HttpStatusCodeResult(HttpStatusCode.BadRequest); } Student student = db.Students.Find(id); if (student == null) { return HttpNotFound(); } return View(student); } // GET: Students/Create public ActionResult Create() { return View(); } // POST: Students/Create // 为了防止“过多发布”攻击,请启用要绑定到的特定属性,有关 // 详细信息,请参阅 http://go.microsoft.com/fwlink/?LinkId=317598。 [HttpPost] [ValidateAntiForgeryToken] public ActionResult Create([Bind(Include = "LastName,FirstMidName,EnrollmentDate")] Student student) { try { if (ModelState.IsValid) { db.Students.Add(student); db.SaveChanges(); return RedirectToAction("Index"); } } catch(DataException /* dex */) { //Log the error (uncomment dex variable name and add a line here to write a log. ModelState.AddModelError("", "Unable to save changes. Try again, and if the problem persists see your system administrator."); } return View(student); } // GET: Students/Edit/5 public ActionResult Edit(int? id) { if (id == null) { return new HttpStatusCodeResult(HttpStatusCode.BadRequest); } Student student = db.Students.Find(id); if (student == null) { return HttpNotFound(); } return View(student); } // POST: Students/Edit/5 // 为了防止“过多发布”攻击,请启用要绑定到的特定属性,有关 // 详细信息,请参阅 http://go.microsoft.com/fwlink/?LinkId=317598。 [HttpPost] [ValidateAntiForgeryToken] public ActionResult Edit([Bind(Include = "ID,LastName,FirstMidName,EnrollmentDate")] Student student) { if (ModelState.IsValid) { db.Entry(student).State = EntityState.Modified; db.SaveChanges(); return RedirectToAction("Index"); } return View(student); } // GET: Students/Delete/5 public ActionResult Delete(int? id) { if (id == null) { return new HttpStatusCodeResult(HttpStatusCode.BadRequest); } //if (saveChangesError.GetValueOrDefault()) { // ViewBag.ErrorMessage = "Delete failed. Try again, and if the problem persists see your system administrator."; //} Student student = db.Students.Find(id); if (student == null) { return HttpNotFound(); } return View(student); } // POST: Students/Delete/5 [HttpPost, ActionName("Delete")] [ValidateAntiForgeryToken] public ActionResult DeleteConfirmed(int id) { Student student = db.Students.Find(id); db.Students.Remove(student); db.SaveChanges(); return RedirectToAction("Index"); } protected override void Dispose(bool disposing) { if (disposing) { db.Dispose(); } base.Dispose(disposing); } } }
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值