Добавление объектного графа
При добавлении сущности в базу данных дочерние записи могут быть добавлены в том же самом обращении без их специального добавления в собственный экземпляр DbSet, если они добавлены в свойство типа коллекции для родительской записи. Например, пусть создается новая сущность Make и в ее свойство Cars добавляется дочерняя запись Car. Когда сущность Make добавляется в свойство DbSet, исполняющая среда EF Core автоматически начинает отслеживание также и дочерней записи Car без необходимости в ее явном добавлении в свойство DbSet. Выполнение метода SaveChanges приводит к совместному сохранению Make и Car, что демонстрируется в следующем тесте:
[Fact]
public void ShouldAddAnObjectGraph
{
ExecuteInATransaction(RunTheTest);
void RunTheTest
{
var make = new Make {Name = "Honda"};
var car = new Car { Color = "Yellow", MakeId = 1, PetName = "Herbie" };
// Привести свойство Cars к List
((List
Context.Makes.Add(make);
var carCount = Context.Cars.Count;
var makeCount = Context.Makes.Count;
Context.SaveChanges;
var newCarCount = Context.Cars. Count;
var newMakeCount = Context.Makes. Count;
Assert.Equal(carCount+1,newCarCount);
Assert.Equal(makeCount+1,newMakeCount);
}
}
Операторы добавления не пакетируются из-за наличия менее двух операторов, а в SQL Server пакетирование начинается с четырех операторов. Ниже показаны выполняемые операторы SQL:
exec sp_executesql N'SET NOCOUNT ON;
INSERT INTO [dbo].[Makes] ([Name])
VALUES (@p0);
SELECT [Id], [TimeStamp]
FROM [dbo].[Makes]
WHERE @@ROWCOUNT = 1 AND [Id] = scope_identity;
',N'@p0 nvarchar(50)',@p0=N'Honda'
exec sp_executesql N'SET NOCOUNT ON;
INSERT INTO [dbo].[Inventory] ([Color], [MakeId], [PetName])
VALUES (@p1, @p2, @p3);
SELECT [Id], [IsDrivable], [TimeStamp]
FROM [dbo].[Inventory]
WHERE @@ROWCOUNT = 1 AND [Id] = scope_identity;
',N'@p1 nvarchar(50),@p2 int,@p3 nvarchar(50)',@p1=N'Yellow',@p2=7,@p3=N'Herbie'
Обновление записей
Записи обновляются за счет их загрузки в DbSet как отслеживаемой сущности, их изменения посредством кода и вызова метода SaveChanges контекста. При выполнении SaveChanges объект ChangeTracker сообщает обо всех модифицированных сущностях и исполняющая среда EF Core (наряду с поставщиком баз данных) создает надлежащий оператор SQL для обновления записи (или операторы SQL, если записей несколько).
Состояние сущности
Когда сущность редактируется, EntityState устанавливается в Modified. После успешного сохранения изменений состояние возвращается к Unchanged.
Обновление отслеживаемых сущностей
Обновление одиночной записи очень похоже на добавление одной записи. Вам понадобится загрузить запись из базы данных, внести в нее какие-то изменения и вызвать метод SaveChanges. Обратите внимание, что вам не нужно вызывать Update/UpdateRange на экземпляре DbSet, поскольку сущности отслеживаются. Представленный ниже тест обновляет только одну запись, но при обновлении и сохранении множества отслеживаемых сущностей процесс будет таким же:
[Fact]
public void ShouldUpdateACar
{
ExecuteInASharedTransaction(RunTheTest);
void RunTheTest(IDbContextTransaction trans)
{
var car = Context.Cars.First(c => c.Id == 1);
Assert.Equal("Black",car.Color);
car.Color = "White";
// Вызывать Update не нужно, т.к. сущность отслеживается.
// Context.Cars.Update(car);
Context.SaveChanges;
Assert.Equal("White", car.Color);