Чтение онлайн

ЖАНРЫ

Язык программирования C#9 и платформа .NET5
Шрифт:

new { Color = "Pink", MakeId = 3, PetName = "Avon" },

new { Color = "Blue", MakeId = 4, PetName = "Blueberry" },

};

var carCount = Context.Cars.Count;

Context.Cars.AddRange(cars);

Context.SaveChanges;

var newCarCount = Context.Cars.Count;

Assert.Equal(carCount + 4, newCarCount);

}

}

Операторы

добавления пакетируются в единственное обращение к базе данных и запрашиваются все сгенерированные столбцы. Когда результаты запроса поступают в EF Core, сущности обновляются с использованием значений серверной стороны. Вот как выглядит выполняемый оператор SQL:

exec sp_executesql N'SET NOCOUNT ON;

DECLARE @inserted0 TABLE ([Id] int, [_Position] [int]);

MERGE [dbo].[Inventory] USING (

VALUES (@p0, @p1, @p2, 0),

(@p3, @p4, @p5, 1),

(@p6, @p7, @p8, 2),

(@p9, @p10, @p11, 3)) AS i ([Color], [MakeId], [PetName], _Position) ON 1=0

WHEN NOT MATCHED THEN

INSERT ([Color], [MakeId], [PetName])

VALUES (i.[Color], i.[MakeId], i.[PetName])

OUTPUT INSERTED.[Id], i._Position

INTO @inserted0;

SELECT [t].[Id], [t].[IsDrivable], [t].[TimeStamp] FROM [dbo].[Inventory] t

INNER JOIN @inserted0 i ON ([t].[Id] = [i].[Id])

ORDER BY [i].[_Position];',

N'@p0 nvarchar(50),@p1 int,@p2 nvarchar(50),@p3 nvarchar(50),

@p4 int,@p5 nvarchar(50),
@p6 nvarchar(50),@p7 int,@p8 nvarchar(50),

@p9 nvarchar(50),@p10 int,@p11 nvarchar(50)',
@p0=N'Yellow',@p1=1,

@p2=N'Herbie',@p3=N'White',@p4=2,@p5=N'Mach 5',@p6=N'Pink',@p7=3,

@p8=N'Avon',@p9=N'Blue',@p10=4,@p11=N'Blueberry'

Соображения относительно столбца идентичности при добавлении записей

Когда сущность имеет числовое свойство, которое определено как первичный ключ, то такое свойство (по умолчанию) отображается на столбец идентичности (

Identity
) в SQL Server. Исполняющая среда EF Core считает сущность со стандартным (нулевым) значением для свойства ключа новой, а сущность с нестандартным значением — уже присутствующей в базе данных. Если вы создаете новую сущность и устанавливаете свойство первичного ключа в ненулевое число, после чего пытаетесь добавить ее в базу данных, то EF Core откажется добавлять запись, поскольку вставка идентичности не разрешена. Включение вставки идентичности демонстрируется в коде инициализации данных.

Добавление объектного графа

При добавлении сущности в базу данных дочерние записи могут быть добавлены в том же самом обращении без их специального добавления в собственный экземпляр

DbSet<T>
, если они добавлены в свойство типа коллекции для родительской записи. Например, пусть создается новая сущность
Make
и в ее свойство
Cars
добавляется дочерняя запись
Car
. Когда сущность
Make
добавляется в свойство
DbSet<Make>
,
исполняющая среда EF Core автоматически начинает отслеживание также и дочерней записи
Car
без необходимости в ее явном добавлении в свойство
DbSet<Car>
. Выполнение метода
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<Car> из IEnumerable<Car>.

((List<Car>)make.Cars).Add(car);

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<T>
как отслеживаемой сущности, их изменения посредством кода и вызова метода
SaveChanges
контекста. При выполнении
SaveChanges
объект
ChangeTracker
сообщает обо всех модифицированных сущностях и исполняющая среда EF Core (наряду с поставщиком баз данных) создает надлежащий оператор SQL для обновления записи (или операторы SQL, если записей несколько).

Поделиться с друзьями: