Compartilhar via


Alterações significativas incluídas no EF Core 3.x

As seguintes alterações de API e comportamento têm o potencial de interromper aplicativos existentes ao atualizá-los para 3.x. As alterações que esperamos que afetem apenas os provedores de banco de dados estão documentadas em alterações de provedor.

Resumo

Alterações da falha Impacto
As consultas LINQ não são mais avaliadas no cliente Alto
A ferramenta de linha de comando do EF Core, dotnet ef, não faz mais parte do SDK do .NET Core Alto
DetectChanges respeita os valores de chave gerados pelo repositório Alto
FromSql, ExecuteSql e ExecuteSqlAsync foram renomeados Alto
Tipos de consulta são consolidados com tipos de entidade Alto
O Entity Framework Core não faz mais parte da estrutura compartilhada do ASP.NET Core Médio
Agora, as exclusões em cascata acontecem imediatamente por padrão Médio
O carregamento ansioso de entidades relacionadas agora acontece em uma única consulta Médio
DeleteBehavior.Restrict tem uma semântica mais limpa Médio
A API de configuração para relações de tipo de propriedade mudou Médio
Cada propriedade usa geração de chave de inteiro em memória independente Médio
As consultas sem acompanhamento não executam mais a resolução de identidade Médio
Alterações na API de metadados Médio
Alterações na API de metadados específicos do provedor Médio
UseRowNumberForPaging foi removido Médio
O método FromSql quando usado com o procedimento armazenado não pode ser composto Médio
Os métodos FromSql só podem ser especificados em raízes de consulta Baixo
Valores de chave temporários não estão mais definidos em instâncias de entidade Baixo
As entidades dependentes que compartilham a tabela com a entidade de segurança agora são opcionais Baixo
Todas as entidades que compartilham uma tabela com uma coluna de token de simultaneidade precisam mapeá-la para uma propriedade Baixo
Entidades de propriedade não podem ser consultadas sem que o proprietário use uma consulta de acompanhamento Baixo
Agora, as propriedades herdadas de tipos não mapeados são mapeadas para uma única coluna para todos os tipos derivados Baixo
A convenção da propriedade de chave estrangeira não corresponde mais ao mesmo nome que a propriedade de entidade de segurança Baixo
Agora, a conexão de banco de dados será fechada se não for mais usada antes da conclusão do TransactionScope Baixo
Os campos de suporte são usados por padrão Baixo
Gerar se vários campos de suporte compatíveis são encontrados Baixo
Os nomes de propriedade somente de campo devem corresponder ao nome de campo Baixo
AddDbContext/AddDbContextPool não chama mais AddLogging e AddMemoryCache Baixo
AddEntityFramework* adiciona IMemoryCache com um limite de tamanho Baixo
DbContext.Entry agora executa uma DetectChanges local Baixo
As chaves de matriz de byte e cadeia de caracteres não são geradas pelo cliente por padrão Baixo
ILoggerFactory agora é um serviço com escopo Baixo
Os proxies de carregamento lento não presumem mais que as propriedades de navegação estejam totalmente carregadas Baixo
Agora, a criação excessiva de provedores de serviço internos é um erro por padrão Baixo
Novo comportamento para HasOne/HasMany chamado com uma única cadeia de caracteres Baixo
O tipo de retorno para vários métodos assíncronos foi alterado de Task para ValueTask Baixo
A anotação Relational:TypeMapping agora é apenas TypeMapping Baixo
ToTable em um tipo derivado gera uma exceção Baixo
O EF Core não envia mais pragma para imposição do FK SQLite Baixo
Microsoft.EntityFrameworkCore.Sqlite agora depende de SQLitePCLRaw.bundle_e_sqlite3 Baixo
Os valores de Guid agora são armazenados como TEXTO no SQLite Baixo
Os valores Char agora são armazenados como TEXTO no SQLite Baixo
As IDs de migração agora são geradas usando o calendário da cultura invariável Baixo
As informações de extensão/metadados foram removidas do IDbContextOptionsExtension Baixo
LogQueryPossibleExceptionWithAggregateOperator foi renomeado Baixo
Esclarecer a API para nomes da restrição de chave estrangeira Baixo
IRelationalDatabaseCreator.HasTables/HasTablesAsync foram tornados públicos Baixo
Microsoft.EntityFrameworkCore.Design agora é um pacote de DevelopmentDependency Baixo
SQLitePCL.raw atualizado para a versão 2.0.0 Baixo
NetTopologySuite atualizado para a versão 2.0.0 Baixo
Microsoft.Data.SqlClient é usado em vez de System.Data.SqlClient Baixo
Várias relações ambíguas de autorreferência devem ser configuradas Baixo
DbFunction.Schema sendo nulo ou cadeia de caracteres vazia configura-o para estar no esquema padrão do modelo Baixo
O EF Core 3.0 tem como destino o .NET Standard 2.1 em vez do .NET Standard 2.0 Revertido
A execução de consulta é registrada no nível da Depuração Revertida

Alterações de alto impacto

As consultas LINQ não são mais avaliadas no cliente

Acompanhamento de problema nº 14935Confira também o problema nº 12795

Comportamento antigo

Antes da 3.0, quando o EF Core não podia converter uma expressão que fazia parte de uma consulta para SQL ou parâmetro, ela automaticamente avaliava a expressão no cliente. Por padrão, a avaliação do cliente de expressões potencialmente dispendiosas apenas disparava um aviso.

Novo comportamento

A partir da 3.0, o EF Core só permite expressões na projeção de nível superior (a última chamada Select() na consulta) a ser avaliada no cliente. Quando expressões em qualquer outra parte da consulta não podem ser convertidas em SQL ou parâmetro, uma exceção é lançada.

Por que

A avaliação automática de consultas do cliente permite que várias consultas sejam executadas, mesmo que partes importantes delas não possam ser convertidas. Esse comportamento pode resultar em comportamento inesperado e potencialmente perigoso que pode se tornar evidente apenas em produção. Por exemplo, uma condição em uma chamada Where() que não pode ser convertida pode fazer todas as linhas da tabela serem transferidas do servidor de banco de dados e o filtro ser aplicado no cliente. Essa situação pode facilmente passar despercebidas se a tabela contém apenas algumas linhas no desenvolvimento, mas ter consequências sérias quando o aplicativo passa para a produção, em que a tabela pode conter milhões de linhas. Avisos de avaliação do cliente também se mostraram muito fáceis de ignorar durante o desenvolvimento.

Além disso, a avaliação automática do cliente pode levar a problemas em que melhorar a conversão da consulta para expressões específicas causa alterações significativas não intencionais entre versões.

Atenuações

Se não for possível converter totalmente uma consulta, reescreva a consulta em uma forma que possa ser convertida ou use AsEnumerableAsync(), ToListAsync() ou semelhante para explicitamente trazer dados de volta para o cliente em um local em que possam ser processados usando o LINQ to Objects.

Alterações de impacto médio

O Entity Framework Core não faz mais parte da estrutura compartilhada do ASP.NET Core

Anúncios de acompanhamento de problema nº 325

Comportamento antigo

Antes do ASP.NET Core 3.0, quando você adicionava uma referência de pacote a Microsoft.AspNetCore.App ou Microsoft.AspNetCore.All, ela incluiria o EF Core e alguns provedores de dados do EF Core, como o provedor do SQL Server.

Novo comportamento

A partir da 3.0, a estrutura compartilhada do ASP.NET Core não inclui o EF Core nem provedores de dados do EF Core.

Por que

Antes dessa alteração, obter o EF Core exigia diferentes etapas, dependendo de se o aplicativo tem como destino o ASP.NET Core e o SQL Server ou não. Além disso, atualizar o ASP.NET Core forçou a atualização do EF Core e do provedor do SQL Server, o que nem sempre é desejável.

Com essa alteração, a experiência de obter o EF Core é a mesmo em todos os provedores, implementações de .NET com suporte e tipos de aplicativos. Os desenvolvedores agora também podem controlar exatamente quando o EF Core e os provedores de dados do EF Core são atualizados.

Atenuações

Para usar o EF Core em um aplicativo ASP.NET Core 3.0 ou qualquer outro aplicativo com suporte, adicione explicitamente uma referência de pacote ao provedor de banco de dados do EF Core que seu aplicativo usará.

A ferramenta de linha de comando do EF Core, dotnet ef, não faz mais parte do SDK do .NET Core

Acompanhamento de problema n. 14016

Comportamento antigo

Antes da 3.0, a ferramenta dotnet ef foi incluída no SDK do .NET Core e ficou prontamente disponível para uso na linha de comando de qualquer projeto sem a necessidade de etapas adicionais.

Novo comportamento

Da 3.0 em diante, o SDK do .NET não inclui a ferramenta dotnet ef, portanto, antes que você possa usá-la, você precisa instalá-la explicitamente como uma ferramenta local ou global.

Por que

Essa alteração permite distribuir e atualizar dotnet ef como uma ferramenta de CLI do .NET regular no NuGet, consistente com o fato de que o EF Core 3.0 também é sempre distribuído como um pacote do NuGet.

Atenuações

Para conseguir gerenciar migrações ou scaffold no DbContext, instale dotnet-ef como uma ferramenta global:

dotnet tool install --global dotnet-ef

Você também pode obtê-lo como uma ferramenta local quando você restaura as dependências de um projeto que o declara como uma dependência de ferramentas usando um arquivo de manifesto de ferramenta.

Alterações de baixo impacto

FromSql, ExecuteSql e ExecuteSqlAsync foram renomeados

Acompanhamento de questões nº 10996

Importante

ExecuteSqlCommand e ExecuteSqlCommandAsync são preteridos. Em vez disso, use esses métodos.

Comportamento antigo

Em versões do EF Core anteriores à 3.0, esses nomes de método eram sobrecarregados para trabalhar com uma cadeia de caracteres normal ou uma cadeia de caracteres que deveria ser interpolada em SQL e parâmetros.

Novo comportamento

Da versão 3.0 do EF Core em diante, use FromSqlRaw, ExecuteSqlRaw, e ExecuteSqlRawAsync para criar uma consulta parametrizada em que os parâmetros são passados separadamente da cadeia de consulta. Por exemplo:

context.Products.FromSqlRaw(
    "SELECT * FROM Products WHERE Name = {0}",
    product.Name);

Use FromSqlInterpolated, ExecuteSqlInterpolated, e ExecuteSqlInterpolatedAsync para criar uma consulta parametrizada em que os parâmetros são passados como parte de uma cadeia de consulta interpolada. Por exemplo:

context.Products.FromSqlInterpolated(
    $"SELECT * FROM Products WHERE Name = {product.Name}");

Observe que ambas as consultas acima produzirão o mesmo SQL parametrizado com os mesmos parâmetros SQL.

Por que

Sobrecargas de método como essa tornam muito fácil chamar acidentalmente o método de cadeia de caracteres bruta quando a intenção era para chamar o método de cadeia de caracteres interpolada e vice-versa. Isso pode resultar em consultas não parametrizadas, quando deveriam ter sido.

Atenuações

Mude para usar os novos nomes de método.

O método FromSql quando usado com o procedimento armazenado não pode ser composto

Problema de acompanhamento nº 15392

Comportamento antigo

Antes do EF Core 3.0, o método FromSql tentou detectar se o SQL passado pode ser composto. Ele fez a avaliação do cliente quando o SQL não era composável como um procedimento armazenado. A consulta a seguir funcionou executando o procedimento armazenado no servidor e fazendo FirstOrDefault no lado do cliente.

context.Products.FromSqlRaw("[dbo].[Ten Most Expensive Products]").FirstOrDefault();

Novo comportamento

A partir do EF Core 3.0, o EF Core não tentará analisar o SQL. Portanto, se você estiver compondo depois de FromSqlRaw/FromSqlInterpolated, o EF Core redigirá o SQL causando a subconsulta. Portanto, se você estiver usando um procedimento armazenado com composição, obterá uma exceção para sintaxe SQL inválida.

Por que

O EF Core 3.0 não dá suporte à avaliação automática do cliente, pois ele era propenso a erros, conforme explicado aqui.

Atenuações

Se você estiver usando um procedimento armazenado em FromSqlRaw/FromSqlInterpolated, você saberá que ele não pode ser composto, portanto, você pode adicionar AsEnumerable/AsAsyncEnumerable logo após a chamada do método FromSql para evitar qualquer composição no lado do servidor.

context.Products.FromSqlRaw("[dbo].[Ten Most Expensive Products]").AsEnumerable().FirstOrDefault();

Os métodos FromSql só podem ser especificados em raízes de consulta

Acompanhamento de problema nº 15704

Comportamento antigo

Antes do EF Core 3.0, o método FromSql podia ser especificado em qualquer lugar na consulta.

Novo comportamento

A partir do EF Core 3.0, os novos métodos FromSqlRaw e FromSqlInterpolated (que substituíram o FromSql) só podem ser especificados em raízes de consultas, ou seja, diretamente no DbSet<>. A tentativa de especificá-los em qualquer outro lugar resultará em um erro de compilação.

Por que

Especificar FromSql em qualquer lugar em vez de DbSet não tinha nenhum significado adicional ou valor agregado e poderia causar ambiguidade em determinados cenários.

Atenuações

As invocações de FromSql devem ser movidas para estarem diretamente no DbSet ao qual se aplicam.

As consultas sem acompanhamento não executam mais a resolução de identidade

Problema de acompanhamento nº 13518

Comportamento antigo

Antes do EF Core 3.0, a mesma instância de entidade era usada para cada ocorrência de uma entidade com uma ID e um tipo especificados. Isso corresponde ao comportamento das consultas de acompanhamento. Por exemplo, esta consulta:

var results = await context.Products.Include(e => e.Category).AsNoTracking().ToListAsync();

retornará a mesma instância de Category para cada Product associado à categoria especificada.

Novo comportamento

No EF Core 3.0 em diante, diferentes instâncias de entidade serão criadas quando uma entidade com uma ID e um tipo especificados for encontrada em locais diferentes no grafo retornado. Por exemplo, a consulta acima agora retornará uma nova instância de Category para cada Product, mesmo quando dois produtos estiverem associados à mesma categoria.

Por que

A resolução de identidade (ou seja, determinar que uma entidade tem o mesmo tipo e a mesma ID de uma entidade encontrada anteriormente) adiciona sobrecarga extra de desempenho e memória. Isso geralmente executa um contador para descobrir o motivo pelo qual as consultas sem acompanhamento são usadas em primeiro lugar. Além disso, embora a resolução de identidade às vezes possa ser útil, ela não é necessária se as entidades são serializadas e enviadas a um cliente, o que é comum para consultas sem acompanhamento.

Atenuações

Use uma consulta de acompanhamento se a resolução de identidade for necessária.

Valores de chave temporários não estão mais definidos em instâncias de entidade

Acompanhamento de problema nº 12378

Comportamento antigo

Antes do EF Core 3.0, valores temporários eram atribuídos a todas as propriedades de chave que mais tarde teriam um valor real gerado pelo banco de dados. Normalmente, esses valores temporários eram grandes números negativos.

Novo comportamento

A partir da 3.0, o EF Core armazena o valor de chave temporária como parte das informações de acompanhamento da entidade e deixa a propriedade de chave em si inalterada.

Por que

Essa alteração foi feita para impedir que valores de chave temporários erroneamente se tornem permanente quando uma entidade anteriormente controlada por alguma instância DbContext for movida para outra instância DbContext.

Atenuações

Aplicativos que atribuem valores de chave primária em chaves estrangeiras para formar associações entre entidades poderão depender do comportamento antigo se as chaves primárias forem geradas pelo repositório e pertencerem a entidades no estado Added. Isso pode ser evitado:

  • Não usando chaves geradas pelo repositório.
  • Definindo propriedades de navegação para formar relações, em vez de definir valores de chave estrangeira.
  • Obter os valores de chave temporários reais de informações de acompanhamento de entidade. Por exemplo, context.Entry(blog).Property(e => e.Id).CurrentValue retornará o valor temporário, embora blog.Id em si não tenha sido definido.

DetectChanges respeita os valores de chave gerados pelo repositório

Acompanhamento de problema nº 14616

Comportamento antigo

Antes do EF Core 3.0, uma entidade não controlada encontrada pelo DetectChanges seria controlada no estado Added e inserida como uma nova linha quando SaveChanges fosse chamado.

Novo comportamento

A partir do EF Core 3.0, se uma entidade estiver usando valores de chave gerados e algum valor de chave estiver definido, a entidade será rastreada no estado Modified. Isso significa que se presume que uma linha para a entidade exista e ela será atualizada quando SaveChanges for chamado. Se o valor da chave não for definido ou se o tipo de entidade não estiver usando as chaves geradas, a nova entidade ainda será rastreada como Added como nas versões anteriores.

Por que

Essa alteração foi feita para tornar mais fácil e consistente trabalhar com grafos de entidade desconectados ao usar chaves geradas pelo repositório.

Atenuações

Essa alteração poderá interromper um aplicativo se um tipo de entidade estiver configurado para usar chaves geradas, mas os valores de chave estiverem explicitamente definidos para novas instâncias. A correção é configurar explicitamente as propriedades de chave para não usarem valores gerados. Por exemplo, com a API fluente:

modelBuilder
    .Entity<Blog>()
    .Property(e => e.Id)
    .ValueGeneratedNever();

Ou com anotações de dados:

[DatabaseGenerated(DatabaseGeneratedOption.None)]
public string Id { get; set; }

Exclusões em cascata acontecerão imediatamente por padrão

Acompanhamento de problema nº 10114

Comportamento antigo

Antes da 3.0, as ações em cascata aplicadas do EF Core (excluindo entidades dependentes quando uma entidade de segurança obrigatória é excluída ou quando a relação com uma entidade de segurança obrigatória é interrompida) não aconteciam até que SaveChanges fosse chamado.

Novo comportamento

A partir da 3.0, o EF Core aplica ações em cascata assim que a condição de disparo é detectada. Por exemplo, chamar context.Remove() para excluir uma entidade principal resultará em todos dependentes obrigatórios relacionados rastreados também serem definidos como Deleted imediatamente.

Por que

Essa alteração foi feita para melhorar a experiência em cenários de vinculação de dados e auditoria, nos quais é importante entender quais entidades serão excluídas antes de SaveChanges ser chamado.

Atenuações

O comportamento anterior pode ser restaurado por meio das configurações em context.ChangeTracker. Por exemplo:

context.ChangeTracker.CascadeDeleteTiming = CascadeTiming.OnSaveChanges;
context.ChangeTracker.DeleteOrphansTiming = CascadeTiming.OnSaveChanges;

Problema de acompanhamento nº 18022

Comportamento antigo

Antes da 3.0, carregar ansiosamente as navegações de coleção por meio Include de operadores fazia com que várias consultas fossem geradas no banco de dados relacional, uma para cada tipo de entidade relacionada.

Novo comportamento

A partir do 3.0, o EF Core gera uma única consulta com JOINs em bancos de dados relacionais.

Por que

A emissão de várias consultas para implementar uma única consulta LINQ causou vários problemas, incluindo o desempenho negativo, pois várias viagens de ida e volta de banco de dados eram necessárias e problemas de coerência de dados, pois cada consulta podia observar um estado diferente do banco de dados.

Atenuações

Embora tecnicamente essa não seja uma alteração significativa, ela pode ter um efeito considerável no desempenho do aplicativo quando uma única consulta contém um grande número de operadores nas navegações de coleção Include. Consulte este comentário para obter mais informações e para reescrever consultas de forma mais eficiente.

**

DeleteBehavior.Restrict tem uma semântica mais limpa

Acompanhamento de questões nº 12661

Comportamento antigo

Antes do 3.0, DeleteBehavior.Restrict criava chaves estrangeiras no banco de dados com a semântica Restrict, mas também alterava a correção interna de maneira não óbvia.

Novo comportamento

No 3.0 em diante, DeleteBehavior.Restrict garante que as chaves estrangeiras são criadas com a semântica Restrict – ou seja, nenhuma cascata é gerada em violação de restrição – sem afetar também a correção interna do EF.

Por que

Essa alteração foi feita para melhorar a experiência do uso de DeleteBehavior de maneira intuitiva, sem efeitos colaterais inesperados.

Atenuações

O comportamento anterior pode ser restaurado usando DeleteBehavior.ClientNoAction.

Tipos de consulta são consolidados com tipos de entidade

Acompanhamento de problema nº 14194

Comportamento antigo

Antes do EF Core 3.0, tipos de consulta eram um meio de consultar dados que não definem uma chave primária de maneira estruturada. Ou seja, um tipo de consulta era usado para mapear os tipos de entidade sem chaves (mais provavelmente de uma exibição, mas possivelmente de uma tabela), enquanto um tipo de entidade normal era usado quando uma chave estava disponível (mais provavelmente de uma tabela, mas possivelmente de um modo de exibição).

Novo comportamento

Um tipo de consulta agora se torna apenas um tipo de entidade sem uma chave primária. Tipos de entidade sem chave têm a mesma funcionalidade que tipos de consulta nas versões anteriores.

Por que

Essa alteração foi feita para reduzir a confusão entre a finalidade dos tipos de consulta. Especificamente, são tipos de entidade sem chave e inerentemente somente leitura devido a isso, mas não devem ser usados apenas porque um tipo de entidade precisa ser somente leitura. Da mesma forma, geralmente são mapeados para modos de exibição, mas isso é só porque os modos de exibição geralmente não definem chaves.

Atenuações

As seguintes partes da API agora estão obsoletas:

  • ModelBuilder.Query<>() – Em vez disso, ModelBuilder.Entity<>().HasNoKey() precisa ser chamado para marcar um tipo de entidade como não tendo chaves. Isso ainda não será configurado por convenção para evitar erros de configuração quando uma chave primária for esperada, mas não corresponder à convenção.
  • DbQuery<> – Em vez disso, DbSet<> deve ser usado.
  • DbContext.Query<>() – Em vez disso, DbContext.Set<>() deve ser usado.
  • IQueryTypeConfiguration<TQuery> – Em vez disso, IEntityTypeConfiguration<TEntity> deve ser usado.

Observação

Devido a um problema no 3.x ao consultar entidades sem chave que tenham todas as propriedades definidas como null uma null será retornado em vez de uma entidade, se esse problema for aplicável ao seu cenário também adicionará lógica para lidar com null os resultados.

A API de configuração para relações de tipo de propriedade mudou

Acompanhamento de problema nº 12444Acompanhamento de problema nº 9148Acompanhamento de problema nº 14153

Comportamento antigo

Antes do EF Core 3.0, a configuração da relação de propriedade era realizada diretamente após a chamada OwnsOne ou OwnsMany.

Novo comportamento

A partir do EF Core 3.0, agora há uma API fluente para configurar uma propriedade de navegação para o proprietário usando WithOwner(). Por exemplo:

modelBuilder.Entity<Order>.OwnsOne(e => e.Details).WithOwner(e => e.Order);

A configuração relacionada à relação entre o proprietário e a propriedade agora deve ser encadeada após WithOwner() da mesma forma que como as outras relações são configuradas. No entanto, a configuração para o tipo de propriedade em si ainda é encadeada após OwnsOne()/OwnsMany(). Por exemplo:

modelBuilder.Entity<Order>.OwnsOne(e => e.Details, eb =>
    {
        eb.WithOwner()
            .HasForeignKey(e => e.AlternateId)
            .HasConstraintName("FK_OrderDetails");

        eb.ToTable("OrderDetails");
        eb.HasKey(e => e.AlternateId);
        eb.HasIndex(e => e.Id);

        eb.HasOne(e => e.Customer).WithOne();

        eb.HasData(
            new OrderDetails
            {
                AlternateId = 1,
                Id = -1
            });
    });

Além disso, chamar Entity(), HasOne() ou Set() com um destino de tipo próprio agora lançará uma exceção.

Por que

Essa alteração foi feita para criar uma separação mais clara entre configurar o tipo de propriedade em si e a relação com o tipo de propriedade. Isso, por sua vez, removerá a ambiguidade e a confusão quanto a métodos como HasForeignKey.

Atenuações

Altere a configuração de relações de tipo de propriedade para usar a nova superfície de API, conforme mostrado no exemplo acima.

As entidades dependentes compartilham a tabela com a entidade de segurança agora são opcionais

Acompanhamento de questões nº 9005

Comportamento antigo

Considere o seguinte modelo :

public class Order
{
    public int Id { get; set; }
    public int CustomerId { get; set; }
    public OrderDetails Details { get; set; }
}

public class OrderDetails
{
    public int Id { get; set; }
    public string ShippingAddress { get; set; }
}

Em versões do EF Core anteriores à 3.0, se OrderDetails fosse pertencente a Order ou explicitamente mapeado para a mesma tabela, uma instância OrderDetails seria sempre necessária ao adicionar um novo Order.

Novo comportamento

Da versão 3.0 em diante, o EF Core permite adicionar um Order sem um OrderDetails e mapeia todas as propriedades OrderDetails, exceto a chave primária para colunas que permitem valor nulo. Ao realizar consultas, o EF Core definirá OrderDetails para null se qualquer uma de suas propriedades requeridas não tiver um valor ou se ele não tiver nenhuma propriedade necessária além da chave primária e todas as propriedades forem null.

Atenuações

Se seu modelo tem uma tabela de compartilhamento de dependentes com todas as colunas opcionais, mas a navegação apontando para ele não deve ser null, então o aplicativo deve ser modificado para lidar com casos quando a navegação for null. Se isso não for possível, uma propriedade necessária deverá ser adicionada ao tipo de entidade ou pelo menos uma propriedade deverá ter um valor não null atribuído a ela.

Todas as entidades compartilhando uma tabela com uma coluna de token de simultaneidade precisam mapeá-lo para uma propriedade

Acompanhamento de questões nº 14154

Comportamento antigo

Considere o seguinte modelo :

public class Order
{
    public int Id { get; set; }
    public int CustomerId { get; set; }
    public byte[] Version { get; set; }
    public OrderDetails Details { get; set; }
}

public class OrderDetails
{
    public int Id { get; set; }
    public string ShippingAddress { get; set; }
}

protected override void OnModelCreating(ModelBuilder modelBuilder)
{
    modelBuilder.Entity<Order>()
        .Property(o => o.Version).IsRowVersion().HasColumnName("Version");
}

Em versões do EF Core anteriores à 3.0, se OrderDetails pertencer a Order ou for explicitamente mapeado para a mesma tabela, atualizar apenas OrderDetails não atualizará o valor Version no cliente e a próxima atualização falhará.

Novo comportamento

Da versão 3.0 em diante, o EF Core propaga o novo valor Version para Order se ele for proprietário de OrderDetails. Caso contrário, uma exceção é gerada durante a validação do modelo.

Por que

Essa alteração foi feita para evitar um valor de token de simultaneidade obsoleto quando apenas uma das entidades mapeadas para a mesma tabela é atualizada.

Atenuações

Todas as entidades que compartilham a tabela precisam incluir uma propriedade que é mapeada para a coluna do token de simultaneidade. É possível criar uma no estado de sombra:

protected override void OnModelCreating(ModelBuilder modelBuilder)
{
    modelBuilder.Entity<OrderDetails>()
        .Property<byte[]>("Version").IsRowVersion().HasColumnName("Version");
}

Entidades de propriedade não podem ser consultadas sem que o proprietário use uma consulta de acompanhamento

Problema de acompanhamento nº 18876

Comportamento antigo

Antes do EF Core 3.0, as entidades de propriedade podiam ser consultadas como qualquer outra navegação.

context.People.Select(p => p.Address);

Novo comportamento

A partir da 3.0, o EF Core será lançado se uma consulta de acompanhamento projetar uma entidade de propriedade sem o proprietário.

Por que

As entidades de propriedade não podem ser manipuladas sem o proprietário, portanto, na grande maioria dos casos, consultá-las dessa forma é um erro.

Atenuações

Se a entidade de propriedade deve ser controlada para ser modificada posteriormente, o proprietário deverá ser incluído na consulta.

Caso contrário, adicione uma chamada AsNoTracking():

context.People.Select(p => p.Address).AsNoTracking();

Agora, as propriedades herdadas de tipos não mapeados são mapeadas para uma única coluna para todos os tipos derivados

Acompanhamento de questões nº 13998

Comportamento antigo

Considere o seguinte modelo :

public abstract class EntityBase
{
    public int Id { get; set; }
}

public abstract class OrderBase : EntityBase
{
    public int ShippingAddress { get; set; }
}

public class BulkOrder : OrderBase
{
}

public class Order : OrderBase
{
}

protected override void OnModelCreating(ModelBuilder modelBuilder)
{
    modelBuilder.Ignore<OrderBase>();
    modelBuilder.Entity<EntityBase>();
    modelBuilder.Entity<BulkOrder>();
    modelBuilder.Entity<Order>();
}

Em versões do EF Core anteriores à 3.0, a propriedade ShippingAddress seria mapeada para separar as colunas para BulkOrder e Order por padrão.

Novo comportamento

Da versão 3.0 em diante, o EF Core cria apenas uma coluna para ShippingAddress.

Por que

O comportamento antigo era inesperado.

Atenuações

A propriedade ainda pode ser mapeada explicitamente para uma coluna separada sobre os tipos derivados:

protected override void OnModelCreating(ModelBuilder modelBuilder)
{
    modelBuilder.Ignore<OrderBase>();
    modelBuilder.Entity<EntityBase>();
    modelBuilder.Entity<BulkOrder>()
        .Property(o => o.ShippingAddress).HasColumnName("BulkShippingAddress");
    modelBuilder.Entity<Order>()
        .Property(o => o.ShippingAddress).HasColumnName("ShippingAddress");
}

A convenção da propriedade de chave estrangeira não corresponde mais ao mesmo nome que a propriedade de entidade de segurança

Acompanhamento de problema nº 13274

Comportamento antigo

Considere o seguinte modelo :

public class Customer
{
    public int CustomerId { get; set; }
    public ICollection<Order> Orders { get; set; }
}

public class Order
{
    public int Id { get; set; }
    public int CustomerId { get; set; }
}

Antes do EF Core 3.0, a propriedade CustomerId seria usada para a chave estrangeira por convenção. No entanto, se Order for um tipo de propriedade, isso também tornará CustomerId a chave primária, e essa normalmente não é a expectativa.

Novo comportamento

Da versão 3.0 em diante, o EF Core não tentará usar propriedades para chaves estrangeiras por convenção se elas tiverem o mesmo nome que a propriedade de entidade de segurança. Nome do tipo de entidade de segurança concatenado com o nome da propriedade de entidade de segurança e nome de navegação concatenado com padrões de nome de propriedade de entidade de segurança ainda são correspondidos. Por exemplo:

public class Customer
{
    public int Id { get; set; }
    public ICollection<Order> Orders { get; set; }
}

public class Order
{
    public int Id { get; set; }
    public int CustomerId { get; set; }
}
public class Customer
{
    public int Id { get; set; }
    public ICollection<Order> Orders { get; set; }
}

public class Order
{
    public int Id { get; set; }
    public int BuyerId { get; set; }
    public Customer Buyer { get; set; }
}

Por que

Essa alteração foi feita para evitar definir erroneamente uma propriedade de chave primária no tipo de propriedade.

Atenuações

Caso a propriedade se destine a ser a chave estrangeira e, portanto, parte da chave primária, configure-a explicitamente como tal.

A conexão de banco de dados agora será fechada se não for mais usada antes da conclusão do TransactionScope

Acompanhamento de questões nº 14218

Comportamento antigo

Em versões do EF Core anteriores à 3.0, se o contexto abrir a conexão dentro de um TransactionScope, a conexão permanecerá aberta enquanto o TransactionScope atual estiver ativo.

using (new TransactionScope())
{
    using (AdventureWorks context = new AdventureWorks())
    {
        context.ProductCategories.Add(new ProductCategory());
        await context.SaveChangesAsync();

        // Old behavior: Connection is still open at this point

        var categories = await context.ProductCategories().ToListAsync();
    }
}

Novo comportamento

Da versão 3.0 em diante, o EF Core fecha a conexão assim que termina de usá-la.

Por que

Essa alteração permite para usar vários contextos no mesmo TransactionScope. O novo comportamento também corresponde ao EF6.

Atenuações

Se a conexão precisar permanecer aberta, uma chamada explícita para OpenConnection() garantirá que o EF Core não a feche prematuramente:

using (new TransactionScope())
{
    using (AdventureWorks context = new AdventureWorks())
    {
        await context.Database.OpenConnectionAsync();
        context.ProductCategories.Add(new ProductCategory());
        await context.SaveChangesAsync();

        var categories = await context.ProductCategories().ToListAsync();
        await context.Database.CloseConnectionAsync();
    }
}

Cada propriedade usa geração de chave de inteiro em memória independente

Acompanhamento de problema nº 6872

Comportamento antigo

Antes do EF Core 3.0, um gerador de valor compartilhado era usado para todas as propriedades de chave de inteiro em memória.

Novo comportamento

Cada propriedade de chave de inteiro a partir do EF Core 3.0 obtém seu próprio gerador de valor ao usar o banco de dados em memória. Além disso, se o banco de dados for excluído, a geração de chave será redefinida para todas as tabelas.

Por que

Essa alteração foi feita para alinhar melhor a geração de chave em memória com a geração de chave do banco de dados real e para melhorar a capacidade de isolar testes uns dos outros ao usar o banco de dados em memória.

Atenuações

Isso pode interromper um aplicativo que depende da definição de valores específicos de chave em memória. Considere, em vez disso, não depender de valores de chave específicos ou atualizar para coincidir com o novo comportamento.

Campos de suporte são usados por padrão

Acompanhamento de problema nº 12430

Comportamento antigo

Antes da 3.0, mesmo que o campo de suporte para uma propriedade fosse conhecido, o EF Core ainda faria, por padrão, a leitura e a gravação do valor da propriedade usando os métodos getter e setter da propriedade. A exceção a isso era a execução da consulta, em que o campo de suporte seria definido diretamente, se fosse conhecido.

Novo comportamento

Do EF Core 3.0 em diante, se o campo de suporte para uma propriedade é conhecido, o EF Core sempre lê e grava essa propriedade usando o campo de suporte. Isso poderá causar uma interrupção de aplicativo se o aplicativo depender do comportamento adicional codificado para os métodos getter ou setter.

Por que

Essa alteração foi feita para impedir que o EF Core erroneamente acionasse a lógica de negócios por padrão, ao executar operações de banco de dados envolvendo as entidades.

Atenuações

O comportamento anterior a 3.0 pode ser restaurado pela configuração do modo de acesso da propriedade em ModelBuilder. Por exemplo:

modelBuilder.UsePropertyAccessMode(PropertyAccessMode.PreferFieldDuringConstruction);

Lançado se vários campos de suporte compatíveis forem encontrados

Acompanhamento de problema nº 12523

Comportamento antigo

Antes do EF Core 3.0, se vários campos correspondessem às regras para localizar o campo de suporte de uma propriedade, então um campo seria escolhido com base em uma ordem de precedência. Isso pode fazer o campo errado ser usado em casos ambíguos.

Novo comportamento

A partir do EF Core 3.0, se vários campos corresponderem à mesma propriedade, uma exceção será lançada.

Por que

Essa alteração foi feita para evitar usar silenciosamente um campo em detrimento de outro, quando apenas um puder ser correto.

Atenuações

As propriedades com campos de suporte ambíguos devem ter o campo a ser usado explicitamente especificado. Por exemplo, usando a API fluente:

modelBuilder
    .Entity<Blog>()
    .Property(e => e.Id)
    .HasField("_id");

Os nomes de propriedade somente de campo devem corresponder ao nome de campo

Comportamento antigo

Antes do EF Core 3.0, uma propriedade poderia ser especificada por um valor de cadeia de caracteres e, se nenhuma propriedade com esse nome fosse encontrada no tipo .NET, o EF Core tentaria correspondê-la a um campo usando regras de convenção.

private class Blog
{
    private int _id;
    public string Name { get; set; }
}
modelBuilder
    .Entity<Blog>()
    .Property("Id");

Novo comportamento

No EF Core 3.0 em diante, uma propriedade somente de campo deve corresponder exatamente ao nome de campo.

modelBuilder
    .Entity<Blog>()
    .Property("_id");

Por que

Essa alteração foi feita para evitar o uso do mesmo campo para duas propriedades nomeadas de forma semelhante; também torna as regras de correspondência para propriedades somente de campo as mesmas propriedades mapeadas para propriedades CLR.

Atenuações

As propriedades somente de campo devem ter o mesmo nome do campo para o qual são mapeadas. Em uma versão futura do EF Core após a 3.0, planejamos reabilitar explicitamente a configuração de um nome de campo diferente do nome da propriedade (consulte o problema nº 15307):

modelBuilder
    .Entity<Blog>()
    .Property("Id")
    .HasField("_id");

AddDbContext/AddDbContextPool não chama mais AddLogging e AddMemoryCache

Acompanhamento de problema nº 14756

Comportamento antigo

Antes do EF Core 3.0, chamar AddDbContext ou AddDbContextPool também registrar serviços de cache de memória e registro em log com DI por meio de chamadas para AddLogging e AddMemoryCache.

Novo comportamento

A partir do EF Core 3.0, AddDbContext e AddDbContextPool não registrarão mais esses serviços com a DI (injeção de dependência).

Por que

O EF Core 3.0 não requer que esses serviços estejam no contêiner de DI do aplicativo. No entanto, se ILoggerFactory estiver registrado no contêiner de DI do aplicativo, ele ainda será usado pelo EF Core.

Atenuações

Se seu aplicativo precisar desses serviços, registre-os explicitamente com o contêiner de DI usando AddLogging ou AddMemoryCache.

AddEntityFramework* adiciona IMemoryCache com um limite de tamanho

Problema de acompanhamento nº 12905

Comportamento antigo

Antes do EF Core 3.0, os métodos de chamada AddEntityFramework* também registravam serviços de cache de memória com DI sem um limite de tamanho.

Novo comportamento

A partir do EF Core 3.0, AddEntityFramework* registrará um serviço IMemoryCache com um limite de tamanho. Se outros serviços adicionados posteriormente dependerem de IMemoryCache, eles poderão atingir rapidamente o limite padrão, causando exceções ou desempenho degradado.

Por que

O uso de IMemoryCache sem um limite poderá resultar no uso de memória descontrolada se houver um bug na lógica de cache de consulta ou se as consultas forem geradas dinamicamente. Ter um limite padrão reduz um possível ataque do DoS.

Atenuações

Na maioria dos casos, a chamada AddEntityFramework* não é necessária se AddDbContext ou AddDbContextPool é chamada também. Portanto, a melhor mitigação é remover a AddEntityFramework* chamada.

Se o aplicativo precisar desses serviços, registre uma implementação IMemoryCache explicitamente com o contêiner de DI com antecedência usando AddMemoryCache.

DbContext.Entry agora executa uma DetectChanges local

Acompanhamento de problema nº 13552

Comportamento antigo

Antes do EF Core 3.0, chamar DbContext.Entry faria alterações serem detectadas para todas as entidades rastreadas. Isso garantia que o estado exposto em EntityEntry fosse atualizado.

Novo comportamento

A partir do EF Core 3.0, chamar DbContext.Entry agora tentará apenas detectar alterações na entidade determinada e quaisquer entidades principais rastreadas relacionadas a ela. Isso significa que as alterações em outro lugar talvez não tenham sido detectadas chamando esse método, o que podia ter implicações no estado do aplicativo.

Observe que, se ChangeTracker.AutoDetectChangesEnabled for definido como false, até mesmo essa detecção de alteração local será desabilitada.

Outros métodos que causam detecção de alteração – por exemplo ChangeTracker.Entries e SaveChanges – ainda causam uma DetectChanges completa de todas as entidades de controladas.

Por que

Essa alteração foi feita para melhorar o desempenho padrão de usar context.Entry.

Atenuações

Chame ChangeTracker.DetectChanges() explicitamente antes de chamar Entry para garantir o comportamento pré-3.0.

As chaves de matriz de byte e cadeia de caracteres não são geradas pelo cliente por padrão

Acompanhamento de problema nº 14617

Comportamento antigo

Antes do EF Core 3.0, as propriedades string e byte[] podiam ser usadas sem configurar explicitamente um valor não nulo. Nesse caso, o valor da chave será gerado no cliente como um GUID, serializado em bytes para byte[].

Novo comportamento

A partir do EF Core 3.0, uma exceção será gerada indicando que nenhum valor de chave foi definido.

Por que

Essa alteração foi feita porque os valores string/byte[] gerados pelo cliente costumam não ser úteis, e o comportamento padrão tornava difícil refletir sobre os valores de chave gerados de uma maneira comum.

Atenuações

O comportamento pré-3.0 pode ser obtido especificando explicitamente que as propriedades chave devem usar valores gerados caso nenhum outro valor não nulo seja definido. Por exemplo, com a API fluente:

modelBuilder
    .Entity<Blog>()
    .Property(e => e.Id)
    .ValueGeneratedOnAdd();

Ou com anotações de dados:

[DatabaseGenerated(DatabaseGeneratedOption.Identity)]
public string Id { get; set; }

ILoggerFactory agora é um serviço com escopo

Acompanhamento de problema nº 14698

Comportamento antigo

Antes do EF Core 3.0, ILoggerFactory era registrado como um serviço singleton.

Novo comportamento

A partir do EF Core 3.0, ILoggerFactory agora está registrado no escopo.

Por que

Essa alteração foi feita para permitir a associação de um agente com uma instância DbContext, que habilita outra funcionalidade e remove alguns casos de comportamento patológico como uma explosão de provedores de serviço interno.

Atenuações

Essa alteração não deverá afetar o código do aplicativo, a menos que esteja registrando e usando os serviços personalizados no provedor de serviço interno do EF Core. Isso não é comum. Nesses casos, a maioria das coisas ainda funcionará, mas qualquer serviço singleton que dependa de ILoggerFactory precisará ser alterado para obter o ILoggerFactory de maneira diferente.

Se você enfrentar situações como essa, registre um problema no Rastreador de problemas do GitHub do EF Core para informar como você está usando ILoggerFactory para que possamos entender melhor como não interromper isso novamente no futuro.

Proxies de carregamento lento não presumem mais que as propriedades de navegação estejam totalmente carregadas

Acompanhamento de problema nº 12780

Comportamento antigo

Antes do EF Core 3.0, quando DbContext era descartado, não havia como saber se uma propriedade de navegação fornecida em uma entidade obtida daquele contexto estava totalmente carregada ou não. Os proxies, em vez disso, presumirão que uma navegação de referência foi carregada caso ela tenha um valor não nulo e que uma navegação de coleção foi carregada caso ela não esteja vazia. Nesses casos, a tentativa de realizar um carregamento lento seria inoperante.

Novo comportamento

A partir do EF Core 3.0, proxies controlam de se uma propriedade de navegação é carregada ou não. Isso significa tentar acessar uma propriedade de navegação carregada após o contexto ter sido descartado sempre será não operacional, mesmo quando a navegação carregada for nula ou vazia. Por outro lado, a tentativa de acessar uma propriedade de navegação que não está carregada gerará uma exceção se o contexto for descartado, mesmo que a propriedade de navegação seja uma coleção não vazia. Se essa situação ocorre, isso significa que o código do aplicativo está tentando usar o carregamento lento em uma hora inválida e o aplicativo deve ser alterado para não fazer isso.

Por que

Essa alteração foi feita para tornar o comportamento consistente e correto ao tentar realizar um carregamento lento em uma instância DbContext descartada.

Atenuações

Atualize o código do aplicativo para não tentar carregamento lento com um contexto descartado ou configure-o para ser inoperante, conforme descrito na mensagem de exceção.

Criação excessiva de provedores de serviço internos agora é um erro por padrão

Acompanhamento de problema nº 10236

Comportamento antigo

Antes do EF Core 3.0, um aviso era registrado para um aplicativo, criando um número patológico de provedores de serviço internos.

Novo comportamento

A partir do EF Core 3.0, esse aviso agora é considerado e um erro e uma exceção são lançados.

Por que

Essa alteração era feita para obter melhores códigos de aplicativo expondo esse caso patológico mais explicitamente.

Atenuações

A causa mais apropriada de ação ao encontrar esse erro é entender a causa raiz e parar de criar tantos provedores de serviço internos. No entanto, o erro pode ser convertido de volta em um aviso (ou ignorado) por meio da configuração no DbContextOptionsBuilder. Por exemplo:

protected override void OnConfiguring(DbContextOptionsBuilder optionsBuilder)
{
    optionsBuilder
        .ConfigureWarnings(w => w.Log(CoreEventId.ManyServiceProvidersCreatedWarning));
}

Novo comportamento para HasOne/HasMany chamado com uma única cadeia de caracteres

Acompanhamento de problema nº 9171

Comportamento antigo

Antes do EF Core 3.0, o código que chamava HasOne ou HasMany com uma única cadeia de caracteres era interpretado de forma confusa. Por exemplo:

modelBuilder.Entity<Samurai>().HasOne("Entrance").WithOne();

O código parece estar relacionando Samurai com algum outro tipo de entidade usando a propriedade de navegação Entrance, que pode ser privada.

Na realidade, esse código tenta criar um relacionamento com algum tipo de entidade chamado Entrance sem propriedade de navegação.

Novo comportamento

A partir do EF Core 3.0, o código acima agora faz o que deveria ter feito antes.

Por que

O comportamento antigo era muito confuso, especialmente ao ler o código de configuração e ao procurar erros.

Atenuações

Isso interromperá somente os aplicativos que estão explicitamente configurando relacionamentos usando cadeias de caracteres para nomes de tipos e não estão especificando a propriedade de navegação explicitamente. Isso não é comum. O comportamento anterior pode ser obtido explicitamente passando null para o nome da propriedade de navegação. Por exemplo:

modelBuilder.Entity<Samurai>().HasOne("Some.Entity.Type.Name", null).WithOne();

O tipo de retorno para vários métodos assíncronos foi alterado de Task para ValueTask

Acompanhamento de questões nº 15184

Comportamento antigo

Os seguintes métodos assíncronos anteriormente retornavam um Task<T>:

  • DbContext.FindAsync()
  • DbSet.FindAsync()
  • DbContext.AddAsync()
  • DbSet.AddAsync()
  • ValueGenerator.NextValueAsync() (e classes derivadas)

Novo comportamento

Os métodos mencionados anteriormente agora retornam um ValueTask<T> sobre o mesmo T que antes.

Por que

Essa alteração reduz o número de alocações de heap incorridas ao invocar esses métodos, melhorando o desempenho geral.

Atenuações

Aplicativos que estão apenas aguardando as APIs acima só precisam ser recompilados. Nenhuma alteração de fonte é necessária. Um uso mais complexo (por exemplo, passando o Task retornado a Task.WhenAny()) normalmente exigem que o ValueTask<T> retornado seja convertido em um Task<T> chamando AsTask() nele. Observe que isso anula a redução de alocação feita por essa alteração.

A anotação Relational:TypeMapping agora é apenas TypeMapping

Acompanhamento de problema nº 9913

Comportamento antigo

O nome da anotação para anotações de mapeamento de tipo era "Relational:TypeMapping".

Novo comportamento

O nome de anotação para anotações de mapeamento de tipo agora é "TypeMapping".

Por que

Mapeamentos de tipo agora são usados mais do que apenas para provedores de banco de dados relacional.

Atenuações

Isso interromperá apenas aplicativos que acessam o mapeamento de tipo diretamente como uma anotação, o que não é comum. A ação mais apropriada para a correção é usar a superfície de API para acessar mapeamentos de tipo, em vez de usar a anotação diretamente.

ToTable em um tipo derivado gera uma exceção

Acompanhamento de problema nº 11811

Comportamento antigo

Antes do EF Core 3.0, ToTable() chamado em um tipo derivado era ignorado, uma vez que a única estratégia de mapeamento de herança era TPH, em que isso não é válido.

Novo comportamento

A partir do EF Core 3.0 e em preparação para a adição de suporte a TPT e TPC em uma versão posterior, ToTable() chamado em um tipo derivado agora gerará uma exceção para evitar uma alteração de mapeamento inesperada no futuro.

Por que

Atualmente, não é válido mapear um tipo derivado para uma tabela diferente. Essa alteração evita interrupção no futuro, quando se torna algo válido a se fazer.

Atenuações

Remova quaisquer tentativas de mapear tipos derivados para outras tabelas.

ForSqlServerHasIndex substituído por HasIndex

Acompanhamento de problema nº 12366

Comportamento antigo

Antes do EF Core 3.0, ForSqlServerHasIndex().ForSqlServerInclude() fornecia uma forma de configurar as colunas usadas com INCLUDE.

Novo comportamento

A partir do EF Core 3.0, agora há suporte para usar Include em um índice no nível relacional. Use HasIndex().ForSqlServerInclude().

Por que

Essa alteração foi feita para consolidar a API para índices com Include em um local para todos os provedores de banco de dados.

Atenuações

Use a nova API, conforme mostrado acima.

Alterações na API de metadados

Acompanhamento de questões nº 214

Novo comportamento

As seguintes propriedades foram convertidas em métodos de extensão:

  • IEntityType.QueryFilter ->GetQueryFilter()
  • IEntityType.DefiningQuery ->GetDefiningQuery()
  • IProperty.IsShadowProperty ->IsShadowProperty()
  • IProperty.BeforeSaveBehavior ->GetBeforeSaveBehavior()
  • IProperty.AfterSaveBehavior ->GetAfterSaveBehavior()

Por que

Essa alteração simplifica a implementação das interfaces mencionados anteriormente.

Atenuações

Use os novos métodos de extensão.

Alterações na API de metadados específicos do provedor

Acompanhamento de questões nº 214

Novo comportamento

Os métodos de extensão específicos do provedor serão nivelados:

  • IProperty.Relational().ColumnName ->IProperty.GetColumnName()
  • IEntityType.SqlServer().IsMemoryOptimized ->IEntityType.IsMemoryOptimized()
  • PropertyBuilder.UseSqlServerIdentityColumn() ->PropertyBuilder.UseIdentityColumn()

Por que

Essa alteração simplifica a implementação dos métodos de extensão mencionados anteriormente.

Atenuações

Use os novos métodos de extensão.

O EF Core não envia mais pragma para imposição do FK SQLite

Acompanhamento de problema nº 12151

Comportamento antigo

Antes do EF Core 3.0, o EF Core enviava PRAGMA foreign_keys = 1 quando uma conexão para o SQLite era aberta.

Novo comportamento

A partir do EF Core 3.0, o EF Core não envia mais PRAGMA foreign_keys = 1 quando uma conexão para o SQLite é aberta.

Por que

Essa alteração foi feita porque o EF Core usa SQLitePCLRaw.bundle_e_sqlite3 por padrão, o que, por sua vez, significa que a imposição de FK é ativada por padrão e não precisa ser habilitada explicitamente sempre que uma conexão é aberta.

Atenuações

Chaves estrangeiras são habilitadas por padrão no SQLitePCLRaw.bundle_e_sqlite3, que é usado por padrão para o EF Core. Para outros casos, chaves estrangeiras podem ser habilitadas especificando Foreign Keys=True em sua cadeia de conexão.

Microsoft.EntityFrameworkCore.Sqlite agora depende de SQLitePCLRaw.bundle_e_sqlite3

Comportamento antigo

Antes do EF Core 3.0, o EF Core era usado SQLitePCLRaw.bundle_green.

Novo comportamento

A partir do EF Core 3.0, o EF Core usa SQLitePCLRaw.bundle_e_sqlite3.

Por que

Essa alteração foi feita de modo que a versão do SQLite usada no iOS fosse consistente com outras plataformas.

Atenuações

Para usar a versão do SQLite nativa no iOS, configure Microsoft.Data.Sqlite para usar um pacote SQLitePCLRaw diferente.

Os valores de Guid agora são armazenados como TEXTO no SQLite

Acompanhamento de problema nº 15078

Comportamento antigo

Os valores de Guid foram previamente armazenados como valores de BLOB no SQLite.

Novo comportamento

Agora os valores de Guid são armazenados como TEXTO.

Por que

O formato binário de Guids não é padronizado. O armazenamento dos valores como TEXTO permite que o banco de dados seja mais compatível com outras tecnologias.

Atenuações

Você pode migrar os bancos de dados existentes para o novo formato executando um código SQL semelhante ao seguinte.

UPDATE MyTable
SET GuidColumn = hex(substr(GuidColumn, 4, 1)) ||
                 hex(substr(GuidColumn, 3, 1)) ||
                 hex(substr(GuidColumn, 2, 1)) ||
                 hex(substr(GuidColumn, 1, 1)) || '-' ||
                 hex(substr(GuidColumn, 6, 1)) ||
                 hex(substr(GuidColumn, 5, 1)) || '-' ||
                 hex(substr(GuidColumn, 8, 1)) ||
                 hex(substr(GuidColumn, 7, 1)) || '-' ||
                 hex(substr(GuidColumn, 9, 2)) || '-' ||
                 hex(substr(GuidColumn, 11, 6))
WHERE typeof(GuidColumn) == 'blob';

No EF Core, você pode continuar usando o comportamento anterior, configurando um conversor de valor nessas propriedades.

modelBuilder
    .Entity<MyEntity>()
    .Property(e => e.GuidProperty)
    .HasConversion(
        g => g.ToByteArray(),
        b => new Guid(b));

O Microsoft.Data.Sqlite ainda pode ler valores de Guid das colunas BLOB e TEXTO; no entanto, como o formato padrão para parâmetros e constantes foi alterado, é provável que você precise tomar medidas para a maioria dos cenários que envolvem Guids.

Os valores de char agora são armazenados como TEXTO no SQLite

Problema de acompanhamento nº 15020

Comportamento antigo

Anteriormente, os valores de char eram armazenados como valores INTEIROS no SQLite. Por exemplo, o valor de char A era armazenado como o valor inteiro 65.

Novo comportamento

Agora, os valores de char são armazenados como TEXTO.

Por que

O armazenamento dos valores como TEXTO é mais natural e permite que o banco de dados seja mais compatível com outras tecnologias.

Atenuações

Você pode migrar os bancos de dados existentes para o novo formato executando um código SQL semelhante ao seguinte.

UPDATE MyTable
SET CharColumn = char(CharColumn)
WHERE typeof(CharColumn) = 'integer';

No EF Core, você pode continuar usando o comportamento anterior, configurando um conversor de valor nessas propriedades.

modelBuilder
    .Entity<MyEntity>()
    .Property(e => e.CharProperty)
    .HasConversion(
        c => (long)c,
        i => (char)i);

O Microsoft.Data.Sqlite também continua capaz de ler os valores de caractere das colunas de INTEIRO e de TEXTO, para que determinados cenários não exijam nenhuma ação.

As IDs de migração agora são geradas usando o calendário da cultura invariável

Problema de acompanhamento nº 12978

Comportamento antigo

As IDs de migração eram geradas inadvertidamente, usando o calendário da cultura atual.

Novo comportamento

Agora, as IDs de migração sempre são geradas usando o calendário da cultura invariável (gregoriano).

Por que

A ordem das migrações é importante para a atualização do banco de dados ou a resolução de conflitos de mesclagem. O uso do calendário invariável evita os problemas de ordenação que podem ocorrer quando os membros da equipe têm calendários de sistemas diferentes.

Atenuações

Essa alteração afeta todos aqueles que usam um calendário não gregoriano no qual o ano é maior do que o do calendário gregoriano (como o calendário budista tailandês). As IDs de migração existentes precisarão ser atualizadas para que as novas migrações sejam ordenadas após as migrações existentes.

A ID da migração pode ser encontrada no atributo Migração nos arquivos de designer das migrações.

 [DbContext(typeof(MyDbContext))]
-[Migration("25620318122820_MyMigration")]
+[Migration("20190318122820_MyMigration")]
 partial class MyMigration
 {

A tabela de histórico de Migrações também precisa ser atualizada.

UPDATE __EFMigrationsHistory
SET MigrationId = CONCAT(LEFT(MigrationId, 4)  - 543, SUBSTRING(MigrationId, 4, 150))

UseRowNumberForPaging foi removido

Acompanhamento de problema nº 16400

Comportamento antigo

Antes do EF Core 3.0, o UseRowNumberForPaging podia ser usado para gerar SQL para paginação compatível com SQL Server 2008.

Novo comportamento

A partir do EF Core 3.0, o EF só gerará SQL para paginação que seja compatível apenas com as versões posteriores do SQL Server.

Por que

Estamos fazendo essa alteração porque o SQL Server 2008 não é mais um produto com suporte e a atualização desse recurso para trabalhar com as alterações de consulta feitas no EF Core 3.0 gera um trabalho significativo.

Atenuações

É recomendável atualizar para uma versão mais recente do SQL Server ou usar um nível de compatibilidade superior para que haja suporte para o SQL gerado. Assim, se isso não for possível, comente no rastreamento do problema com detalhes. Poderemos rever essa decisão com base nos comentários.

As informações de extensão/metadados foram removidas do IDbContextOptionsExtension

Acompanhamento de problema nº 16119

Comportamento antigo

IDbContextOptionsExtension continha métodos para fornecer metadados sobre a extensão.

Novo comportamento

Esses métodos foram movidos para uma nova classe base abstrata DbContextOptionsExtensionInfo, que é retornada da nova propriedade IDbContextOptionsExtension.Info.

Por que

Nas versões 2.0 a 3.0, precisamos adicionar ou alterar esses métodos várias vezes. Dividi-los em uma nova classe base abstrata facilitará a realização desse tipo de alteração sem interromper as extensões existentes.

Atenuações

Atualize as extensões para que sigam o novo padrão. Os exemplos são encontrados em várias implementações de IDbContextOptionsExtension em diferentes tipos de extensões no código-fonte do EF Core.

LogQueryPossibleExceptionWithAggregateOperator foi renomeado

Acompanhamento de problema nº 10985

Alterar

RelationalEventId.LogQueryPossibleExceptionWithAggregateOperator foi renomeado para RelationalEventId.LogQueryPossibleExceptionWithAggregateOperatorWarning.

Por que

Alinha a nomeação deste evento de aviso com todos os outros eventos de aviso.

Atenuações

Use o novo nome. (Observe que o número da ID do evento não foi alterado.)

Esclarecer a API para nomes de restrição de chave estrangeira

Acompanhamento de problema nº 10730

Comportamento antigo

Antes do EF Core 3.0, os nomes de restrição de chave estrangeira eram chamados simplesmente de "nome". Por exemplo:

var constraintName = myForeignKey.Name;

Novo comportamento

A partir do EF Core 3.0, os nomes de restrição de chave estrangeira são chamados de "nome de restrição". Por exemplo:

var constraintName = myForeignKey.ConstraintName;

Por que

Essa alteração traz consistência à nomeação nessa área e também esclarece que esse é o nome de restrição de chave estrangeira, e não o nome da coluna ou da propriedade na qual a chave estrangeira está definida.

Atenuações

Use o novo nome.

IRelationalDatabaseCreator.HasTables/HasTablesAsync tornaram-se públicos

Acompanhamento de problema nº 15997

Comportamento antigo

Antes do EF Core 3.0, esses métodos eram protegidos.

Novo comportamento

A partir do EF Core 3.0, esses métodos são públicos.

Por que

Esses métodos são usados pelo EF para determinar se um banco de dados foi criado, mas está vazio. Isso também pode ser útil fora do EF ao determinar se as migrações serão ou não aplicadas.

Atenuações

Altere a acessibilidade de todas as substituições.

Agora Microsoft.EntityFrameworkCore.Design é um pacote de DevelopmentDependency

Acompanhamento de problema nº 11506

Comportamento antigo

Antes do EF Core 3.0, o Microsoft.EntityFrameworkCore.Design era um pacote regular do NuGet cujo assembly podia ser referenciado por projetos que dependiam dele.

Novo comportamento

A partir do EF Core 3.0, ele se tornou um pacote de DevelopmentDependency. Isso significa que a dependência não fluirá transitivamente para outros projetos e que você não pode mais, por padrão, referenciar seu assembly.

Por que

Esse pacote destina-se para uso somente em tempo de design. Os aplicativos implantados não devem fazer referência a ele. Tornar o pacote um DevelopmentDependency reforça essa recomendação.

Atenuações

Se você precisar fazer referência a esse pacote para substituir o comportamento de tempo de design do EF Core, atualize os metadados do item PackageReference em seu projeto.

<PackageReference Include="Microsoft.EntityFrameworkCore.Design" Version="3.0.0">
  <PrivateAssets>all</PrivateAssets>
  <!-- Remove IncludeAssets to allow compiling against the assembly -->
  <!--<IncludeAssets>runtime; build; native; contentfiles; analyzers; buildtransitive</IncludeAssets>-->
</PackageReference>

Se o pacote estiver sendo referenciado transitivamente por meio de Microsoft.EntityFrameworkCore.Tools, você precisará adicionar um PackageReference explícito ao pacote para alterar seus metadados. Essa referência explícita deve ser adicionada a qualquer projeto em que os tipos do pacote sejam necessários.

SQLitePCL.raw atualizado para a versão 2.0.0

Acompanhamento de problema nº 14824

Comportamento antigo

Anteriormente, Microsoft.EntityFrameworkCore.Sqlite dependia da versão 1.1.12 do SQLitePCL.raw.

Novo comportamento

Atualizamos nosso pacote para depender da versão 2.0.0.

Por que

A versão 2.0.0 do SQLitePCL.raw é voltada para o .NET Standard 2.0. Anteriormente, era voltada ao .NET Standard 1.1, o que exigia um fechamento grande de pacotes transitivos para funcionar.

Atenuações

O SQLitePCL.raw versão 2.0.0 inclui algumas alterações de falha. Confira as notas sobre a versão para obter mais detalhes.

NetTopologySuite atualizado para a versão 2.0.0

Acompanhamento do problema n.º 14825

Comportamento antigo

Os pacotes espaciais anteriormente dependiam da versão 1.15.1 do NetTopologySuite.

Novo comportamento

Atualizamos nosso pacote para depender da versão 2.0.0.

Por que

O objetivo da versão 2.0.0 do NetTopologySuite é abordar vários problemas de usabilidade encontrados pelos usuários do EF Core.

Atenuações

O NetTopologySuite versão 2.0.0 inclui algumas alterações significativas. Confira as notas sobre a versão para obter mais detalhes.

Microsoft.Data.SqlClient é usado em vez de System.Data.SqlClient

Problema de acompanhamento nº 15636

Comportamento antigo

Microsoft.EntityFrameworkCore.SqlServer anteriormente dependia de System.Data.SqlClient.

Novo comportamento

Atualizamos nosso pacote para depender de Microsoft.Data.SqlClient.

Por que

Microsoft.Data.SqlClient é o principal driver de acesso a dados do SQL Server no futuro e System.Data.SqlClient não é mais o foco do desenvolvimento. Alguns recursos importantes, como o Always Encrypted, só estão disponíveis em Microsoft.Data.SqlClient.

Atenuações

Se o código usa uma dependência direta em System.Data.SqlClient, você deve alterá-lo para fazer referência a Microsoft.Data.SqlClient; como os dois pacotes mantêm um grau muito alto de compatibilidade de API, isso deve ser apenas um pacote simples e uma alteração de namespace.

Várias relações ambíguas de autorreferência devem ser configuradas

Acompanhamento de Problema nº 13573

Comportamento antigo

Um tipo de entidade com várias propriedades de navegação unidirecionais de autorreferência e correspondência de FKs foi configurado incorretamente como uma relação única. Por exemplo:

public class User
{
        public Guid Id { get; set; }
        public User CreatedBy { get; set; }
        public User UpdatedBy { get; set; }
        public Guid CreatedById { get; set; }
        public Guid? UpdatedById { get; set; }
}

Novo comportamento

Esse cenário agora é detectado na criação de modelo e uma exceção é gerada indicando que o modelo é ambíguo.

Por que

O modelo resultante era ambíguo e provavelmente, em geral, estará errado para esse caso.

Atenuações

Use a configuração completa da relação. Por exemplo:

modelBuilder
     .Entity<User>()
     .HasOne(e => e.CreatedBy)
     .WithMany();

 modelBuilder
     .Entity<User>()
     .HasOne(e => e.UpdatedBy)
     .WithMany();

DbFunction.Schema sendo nulo ou cadeia de caracteres vazia configura-o para estar no esquema padrão do modelo

Problema de acompanhamento nº 12757

Comportamento antigo

Um DbFunction configurado com o esquema como uma cadeia de caracteres vazia foi tratado como uma função interna sem um esquema. Por exemplo, o código a seguir mapeará DatePart a função CLR para a DATEPART função interna no SqlServer.

[DbFunction("DATEPART", Schema = "")]
public static int? DatePart(string datePartArg, DateTime? date) => throw new Exception();

Novo comportamento

Todos os mapeamentos DbFunction são considerados mapeados para funções definidas pelo usuário. Portanto, o valor de cadeia de caracteres vazio colocaria a função dentro do esquema padrão do modelo. Que pode ser o esquema configurado explicitamente por meio da API modelBuilder.HasDefaultSchema() fluente ou dbo de outra forma.

Por que

Anteriormente, o esquema estava vazio era uma maneira de tratar que a função é interna, mas essa lógica só é aplicável ao SqlServer em que as funções internas não pertencem a nenhum esquema.

Atenuações

Configure a tradução de DbFunction manualmente para mapeá-la para uma função interna.

modelBuilder
    .HasDbFunction(typeof(MyContext).GetMethod(nameof(MyContext.DatePart)))
    .HasTranslation(args => SqlFunctionExpression.Create("DatePart", args, typeof(int?), null));

O EF Core 3.0 tem como destino o .NET Standard 2.1 em vez do .NET Standard 2.0 Revertido

Problema de acompanhamento nº 15498

O EF Core 3.0 tem como destino o .NET Standard 2.1, que é uma alteração significativa que exclui aplicativos do .NET Framework. O EF Core 3.1 reverteu isso e direciona o .NET Standard 2.0 novamente.

A execução de consulta é registrada no nível da Depuração Revertida

Acompanhamento de problema nº 14523

Revertemos essa alteração porque a nova configuração do EF Core 3.0 permite que o nível de log de qualquer evento seja especificado pelo aplicativo. Por exemplo, para alternar o registro em log do SQL para Debug, configure explicitamente o nível em OnConfiguring ou AddDbContext:

protected override void OnConfiguring(DbContextOptionsBuilder optionsBuilder)
    => optionsBuilder
        .UseSqlServer(connectionString)
        .ConfigureWarnings(c => c.Log((RelationalEventId.CommandExecuting, LogLevel.Debug)));