Entity Framework Core无连接的场景更新数据

Entity Framework Core无连接的场景更新数据,EF Core API在数据库中为EntityState已修改的实体构建并执行UPDATE语句。在连接的场景中,DbContext跟踪所有实体,以便它知道哪些被修改,因此自动设置EntityState为Modified。

Entity Framework Core无连接的场景更新数据,EF Core API在数据库中为EntityState已修改的实体构建并执行UPDATE语句。在连接的场景中,DbContext跟踪所有实体,以便它知道哪些被修改,因此自动设置EntityState为Modified。

Entity Framework Core无连接的场景更新数据

在诸如Web应用程序之类的无连接的场景中,DbContext由于实体被修改为超出当前DbContext实例的范围,因此不知道实体。因此,首先我们需要将无连接的实体附加到DbContext具有Modified 的实例EntityState

Entity Framework Core无连接的场景更新数据
Entity Framework Core无连接的场景更新数据

下表列出了更新实体的方法DbContextDbSet方法:

DbContext方法DbSet方法描述
DbContext.UpdateDbSet.Update将实体附加到具有修改状态的DbContext。
DbContext.UpdateRangeDbSet.UpdateRange将实体集合附加到具有已修改状态的DbContext。

Entity Framework Core无连接,以下示例演示如何更新断开连接的实体。

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
// Disconnected Student entity
var stud = new Student(){ StudentId = 1, Name = "Bill" };
stud.Name = "Steve";
using (var context = new SchoolContext())
{
context.Update<Student>(stud);
// or the followings are also valid
// context.Students.Update(stud);
// context.Attach<Student>(stud).State = EntityState.Modified;
// context.Entry<Student>(stud).State = EntityState.Modified;
context.SaveChanges();
}
// Disconnected Student entity var stud = new Student(){ StudentId = 1, Name = "Bill" }; stud.Name = "Steve"; using (var context = new SchoolContext()) { context.Update<Student>(stud); // or the followings are also valid // context.Students.Update(stud); // context.Attach<Student>(stud).State = EntityState.Modified; // context.Entry<Student>(stud).State = EntityState.Modified; context.SaveChanges(); }
// Disconnected Student entity
var stud = new Student(){ StudentId = 1, Name = "Bill" };
stud.Name = "Steve"; 
using (var context = new SchoolContext())
{
    context.Update<Student>(stud);
    // or the followings are also valid
    // context.Students.Update(stud);
    // context.Attach<Student>(stud).State = EntityState.Modified;
    // context.Entry<Student>(stud).State = EntityState.Modified; 
    context.SaveChanges(); 
}

在上面的示例中,请考虑stud现有Student实体对象,因为它具有有效的Key属性值(StudentId = 1)。Entity Framework Core引入了DbContext.Update()将指定实体附加到上下文并将其设置EntityState为Modified的方法。或者,您也可以使用DbSet.Update()方法(context.Students.Update(stud))来执行相同的操作。

上面的示例在数据库中执行以下UPDATE语句。

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
exec sp_executesql N'SET NOCOUNT ON;
UPDATE [Students] SET [Name] = @p0
WHERE [StudentId] = @p1;
SELECT @@ROWCOUNT;
',N'@p1 int,@p0 nvarchar(4000)',@p1=1,@p0=N'Steve'
go
exec sp_executesql N'SET NOCOUNT ON; UPDATE [Students] SET [Name] = @p0 WHERE [StudentId] = @p1; SELECT @@ROWCOUNT; ',N'@p1 int,@p0 nvarchar(4000)',@p1=1,@p0=N'Steve' go
exec sp_executesql N'SET NOCOUNT ON;
UPDATE [Students] SET [Name] = @p0
WHERE [StudentId] = @p1;
SELECT @@ROWCOUNT;
',N'@p1 int,@p0 nvarchar(4000)',@p1=1,@p0=N'Steve'
go

更新多个实体

使用DbContext.UpdateRangeor DbSet.UpdateRange方法将实体的集合或数组附加到其中DbContext,并将它们EntityState一次性设置为Modified。

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
var modifiedStudent1 = new Student()
{
StudentId = 1,
Name = "Bill"
};
var modifiedStudent2 = new Student()
{
StudentId = 3,
Name = "Steve"
};
var modifiedStudent3 = new Student()
{
StudentId = 3,
Name = "James"
};
IList<Student> modifiedStudents = new List<Student>()
{
modifiedStudent1,
modifiedStudent2,
modifiedStudent3
};
using (var context = new SchoolContext())
{
context.UpdateRange(modifiedStudents);
// or the followings are also valid
//context.UpdateRange(modifiedStudent1, modifiedStudent2, modifiedStudent3);
//context.Students.UpdateRange(modifiedStudents);
//context.Students.UpdateRange(modifiedStudent1, modifiedStudent2, modifiedStudent3);
context.SaveChanges();
}
var modifiedStudent1 = new Student() { StudentId = 1, Name = "Bill" }; var modifiedStudent2 = new Student() { StudentId = 3, Name = "Steve" }; var modifiedStudent3 = new Student() { StudentId = 3, Name = "James" }; IList<Student> modifiedStudents = new List<Student>() { modifiedStudent1, modifiedStudent2, modifiedStudent3 }; using (var context = new SchoolContext()) { context.UpdateRange(modifiedStudents); // or the followings are also valid //context.UpdateRange(modifiedStudent1, modifiedStudent2, modifiedStudent3); //context.Students.UpdateRange(modifiedStudents); //context.Students.UpdateRange(modifiedStudent1, modifiedStudent2, modifiedStudent3); context.SaveChanges(); }
var modifiedStudent1 = new Student()
{
    StudentId = 1,
    Name = "Bill"
};

var modifiedStudent2 = new Student()
{
    StudentId = 3,
    Name = "Steve"
};

var modifiedStudent3 = new Student()
{
    StudentId = 3,
    Name = "James"
};

IList<Student> modifiedStudents = new List<Student>()
{
    modifiedStudent1,
    modifiedStudent2,
    modifiedStudent3
};

using (var context = new SchoolContext())
{
    context.UpdateRange(modifiedStudents);
    // or the followings are also valid
    //context.UpdateRange(modifiedStudent1, modifiedStudent2, modifiedStudent3);
    //context.Students.UpdateRange(modifiedStudents);
    //context.Students.UpdateRange(modifiedStudent1, modifiedStudent2, modifiedStudent3);
                
    context.SaveChanges();
}

如您所见,该UpdateRange方法有两个重载。一个重载采用实体集合,第二个重载object[]作为参数。该DbSet.UpdateRange方法与该方法的工作方式相同DbContext.UpdateRange

EF Core通过为上例中的所有实体构建UPDATE语句来提高性能,并在单个数据库往返中执行它。

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
exec sp_executesql N'SET NOCOUNT ON;
UPDATE [Students] SET [Name] = @p0
WHERE [StudentId] = @p1;
SELECT @@ROWCOUNT;
UPDATE [Students] SET [Name] = @p2
WHERE [StudentId] = @p3;
SELECT @@ROWCOUNT;
UPDATE [Students] SET [Name] = @p4
WHERE [StudentId] = @p5;
SELECT @@ROWCOUNT;
',N'@p1 int,@p0 nvarchar(4000),@p3 int,@p2 nvarchar(4000),@p5 int,@p4 nvarchar(4000)',
@p1=1,@p0=N'Bill',@p3=2,@p2=N'Steve',@p5=3,@p4=N'James'
go
exec sp_executesql N'SET NOCOUNT ON; UPDATE [Students] SET [Name] = @p0 WHERE [StudentId] = @p1; SELECT @@ROWCOUNT; UPDATE [Students] SET [Name] = @p2 WHERE [StudentId] = @p3; SELECT @@ROWCOUNT; UPDATE [Students] SET [Name] = @p4 WHERE [StudentId] = @p5; SELECT @@ROWCOUNT; ',N'@p1 int,@p0 nvarchar(4000),@p3 int,@p2 nvarchar(4000),@p5 int,@p4 nvarchar(4000)', @p1=1,@p0=N'Bill',@p3=2,@p2=N'Steve',@p5=3,@p4=N'James' go
exec sp_executesql N'SET NOCOUNT ON;
UPDATE [Students] SET [Name] = @p0
WHERE [StudentId] = @p1;
SELECT @@ROWCOUNT;

UPDATE [Students] SET [Name] = @p2
WHERE [StudentId] = @p3;
SELECT @@ROWCOUNT;

UPDATE [Students] SET [Name] = @p4
WHERE [StudentId] = @p5;
SELECT @@ROWCOUNT;

',N'@p1 int,@p0 nvarchar(4000),@p3 int,@p2 nvarchar(4000),@p5 int,@p4 nvarchar(4000)',
@p1=1,@p0=N'Bill',@p3=2,@p2=N'Steve',@p5=3,@p4=N'James'
go

更改实体状态EntityState

Update方法EntityState基于key属性的值设置。如果根或子实体的key属性为空,指定数据类型的null或默认值,则该Update()方法将其视为新实体,并将其设置EntityStateEntity Framework Core 2.x中的Added。

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
public static void Main()
{
var newStudent = new Student()
{
Name = "Bill"
};
var modifiedStudent = new Student()
{
StudentId = 1,
Name = "Steve"
};
using (var context = new SchoolContext())
{
context.Update<Student>(newStudent);
context.Update<Student>(modifiedStudent);
DisplayStates(context.ChangeTracker.Entries());
}
}
private static void DisplayStates(IEnumerable<EntityEntry> entries)
{
foreach (var entry in entries)
{
Console.WriteLine($"Entity: {entry.Entity.GetType().Name},
State: {entry.State.ToString()} ");
}
}
public static void Main() { var newStudent = new Student() { Name = "Bill" }; var modifiedStudent = new Student() { StudentId = 1, Name = "Steve" }; using (var context = new SchoolContext()) { context.Update<Student>(newStudent); context.Update<Student>(modifiedStudent); DisplayStates(context.ChangeTracker.Entries()); } } private static void DisplayStates(IEnumerable<EntityEntry> entries) { foreach (var entry in entries) { Console.WriteLine($"Entity: {entry.Entity.GetType().Name}, State: {entry.State.ToString()} "); } }
public static void Main()
{
    var newStudent = new Student()
    {
        Name = "Bill"
    };

    var modifiedStudent = new Student()
    {
        StudentId = 1,
        Name = "Steve"
    };

    using (var context = new SchoolContext())
    {
        context.Update<Student>(newStudent);
        context.Update<Student>(modifiedStudent);

        DisplayStates(context.ChangeTracker.Entries());
    }
}
private static void DisplayStates(IEnumerable<EntityEntry> entries)
{
    foreach (var entry in entries)
    {
        Console.WriteLine($"Entity: {entry.Entity.GetType().Name},
                 State: {entry.State.ToString()} ");
    }
}
Output:
Entity: Student, State: Added 
Entity: Student, State: Modified

在上面的示例中,newStudent没有Key属性值(StudentId)。因此,该Update()方法将其标记为已添加,而modifiedStudent具有值,因此它将标记为已修改。

异常:

UpdateUpdateRange方法抛出InvalidOperationException,如果实例DbContext已经跟踪具有相同键的属性值的实体。请考虑以下示例:

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
var student = new Student()
{
StudentId = 1,
Name = "Steve"
};
using (var context = new SchoolContext())
{
// loads entity in a conext whose StudentId is 1
context.Students.First<Student>(s => s.StudentId == 1);
// throws an exception as it already tracking entity with StudentId=1
context.Update<Student>(student);
context.SaveChanges();
}
var student = new Student() { StudentId = 1, Name = "Steve" }; using (var context = new SchoolContext()) { // loads entity in a conext whose StudentId is 1 context.Students.First<Student>(s => s.StudentId == 1); // throws an exception as it already tracking entity with StudentId=1 context.Update<Student>(student); context.SaveChanges(); }
var student = new Student()
{
    StudentId = 1,
    Name = "Steve"
};

using (var context = new SchoolContext())
{
    // loads entity in a conext whose StudentId is 1
    context.Students.First<Student>(s => s.StudentId == 1); 
    // throws an exception as it already tracking entity with StudentId=1
    context.Update<Student>(student); 
    context.SaveChanges();
}

在上面的示例中,context对象加载的Student实体StudentId为1并开始跟踪它。因此,附加具有相同键值的实体将引发以下异常:

无法跟踪实体类型“Student”的实例,因为已经跟踪了另一个具有{‘StudentId’}相同键值的实例。附加现有实体时,请确保仅附加具有给定键值的一个实体实例。考虑使用’DbContextOptionsBuilder.EnableSensitiveDataLogging’来查看冲突的键值。

下一章将学习在无连接中删除数据。

加入电报群

【江湖人士】(jhrs.com)原创文章,作者:江小编,如若转载,请注明出处:https://jhrs.com/2019/27162.html

扫码加入电报群,让你获得国外网赚一手信息。

文章标题:Entity Framework Core无连接的场景更新数据

(0)
江小编的头像江小编
上一篇 2019-03-24 12:42
下一篇 2019-03-25 13:19

热门推荐

Leave a Reply

Sending

国外老牌便宜域名服务商Namecheap注册com域名大优惠,抢到就赚到,优惠码:NEWCOM698
$5.98/年
直达官网