问题
我们的一些查询遇到超时错误,经过调查和测试后,我们已将问题缩小到由Entity Framework Code First创建的动态Sql查询。
背景
我们正在使用Entity Framework v4.3.1,Code First模型 用VS 2010 Premium编写的代码
我们在数据库中有一个如下定义的表,ID为PK
CREATE TABLE [dbo].[Reference](
[DocumentID] [varchar](100) NOT NULL,
[DetailID] [varchar](50) NULL,
[TransactionSet] [char](5) NULL,
[Qualifier] [varchar](80) NULL,
[ReferenceID] [nvarchar](30) NULL,
[Description] [nvarchar](80) NULL,
[ID] [int] IDENTITY(1,1) NOT NULL
我们有一个如下定义的实体类来对应表:
public class Reference
{
public string DocumentID { get; set; }
public string DetailID { get; set; }
public string TransactionSet { get; set; }
public string Qualifier { get; set; }
public string ReferenceID { get; set; }
public string Description { get; set; }
public int ID { get; set; }
}
我们有一个数据上下文定义如下(为简洁起见,显示了部分类):
public class DataContext : DbContext
{
public DataContext() : base("Name=DataContext")
{
this.Configuration.AutoDetectChangesEnabled = false;
Database.SetInitializer<DataContext>(null);
}
public DbSet<Reference> References { get; set; }
protected override void OnModelCreating(DbModelBuilder modelBuilder)
{
modelBuilder.Conventions.Remove<PluralizingTableNameConvention>();
modelBuilder.Entity<Reference>().HasKey(pk => pk.ID);
}
}
最后调用代码如下:
private List<Reference> getDocumentReferences(List<string> documentIds)
{
List<Reference> result = null;
using (var context = new DataContext())
{
if (context != null)
{
var q = context.References.Where(r => documentIds.Contains(r.DocumentID) && r.Qualifier == "AFN");
Logger.WriteLogEntry("Sql query:\r\n {0}", q.ToString()); //Added to see the query generated by EF
result = q.ToList();
}
}
return result;
}
描述
执行代码时,以下查询将发送到Sql Server:
SELECT
[Extent1].[ID] AS [ID],
[Extent1].[DocumentID] AS [DocumentID],
[Extent1].[DetailID] AS [DetailID],
[Extent1].[TransactionSet] AS [TransactionSet],
[Extent1].[Qualifier] AS [Qualifier],
[Extent1].[ReferenceID] AS [ReferenceID],
[Extent1].[Description] AS [Description]
FROM [dbo].[Reference] AS [Extent1]
WHERE ([Extent1].[DocumentID] IN (N'A2014011300028343869701A020',N'A2014011300028343869701A021')) AND (N'AFN' = [Extent1].[Qualifier])
请注意,由于代码使用List的Contains方法,因此它由EF转换为sql查询中的IN子句。 IN子句中的每个字符串都以&#39; N&#39;为前缀。表示一个unicode字符串。由于表中的DocumentID列是VARCHAR(不是NVARCHAR),因此Sql必须执行隐式转换才能执行查询。经过调查和测试,我们发现在这种情况下,它不会选择导致表扫描的相关索引,而且通常会出现超时异常。
我们发现,通过从查询中删除 N前缀,使用了正确的索引,查询速度提高了几个数量级。
因此,为了使EF不在字符串前加上N,我将下面的两行添加到OnModelCreating方法的数据上下文类中:
modelBuilder.Entity<Reference>().Property(r => r.DocumentID).HasColumnType("VARCHAR");
modelBuilder.Entity<Reference>().Property(r => r.Qualifier).HasColumnType("VARCHAR");
当我在本地机器上测试时,它具有所需的效果。提交给Sql Server的查询如下。请注意,N前缀不包含在IN子句中的字符串中:
SELECT
[Extent1].[ID] AS [ID],
[Extent1].[DocumentID] AS [DocumentID],
[Extent1].[DetailID] AS [DetailID],
[Extent1].[TransactionSet] AS [TransactionSet],
[Extent1].[Qualifier] AS [Qualifier],
[Extent1].[ReferenceID] AS [ReferenceID],
[Extent1].[Description] AS [Description]
FROM [dbo].[Reference] AS [Extent1]
WHERE ([Extent1].[DocumentID] IN ('A2014011300028343869701A020','A2014011300028343869701A021')) AND ('AFN' = [Extent1].[Qualifier])
我认为我的问题已经解决,但是当我将此代码部署到测试环境时,N前缀未被删除,我无法弄清楚原因。
我可以通过手动创建Sql查询来解决这个问题,但是我在这个程序中还有一些需要类似更改的查询,如果我只需要更改OnModelCreating方法而不是手动创建每个查询就会更容易
有没有人想过为什么会这样?
顺便说一句,我的本地计算机使用Sql Server 2008,而测试环境(和生产)使用Sql 2005.我不确定这是否会影响EF生成的查询,因为我的印象是生成了查询在提交到数据库之前。
更新
如果我针对测试数据库执行我的框上的代码,它可以正常工作。如果代码在测试框上执行 ,即使它们都在同一个数据库中,它也不会。
更新2
代码根据执行的位置成功运行。
EXE Location | EXE Executed From | Database Server (Version) | Result
--------------------------------------------------------------------------
My Box | My Box | My Box (2008) | Success
Test Server | My Box | Test DB Server (2005) | Success
Test Server | Test Server | Test DB Server (2005) | Fail
基于上面的结果,似乎合乎逻辑的是,我的本地盒子上的某些东西与测试机器上的东西不同。但我不知道如何诊断它。我能想到的唯一区别是我的盒子安装了.Net 4.5而测试服务器没有。
答案 0 :(得分:0)
我已经能够使用DataAnnotations
更正此行为:
public class Reference
{
[Column(TypeName = "varchar"), MaxLength(100)]
public string DocumentID { get; set; }
[Column(TypeName = "varchar"), MaxLength(50)]
public string DetailID { get; set; }
[Column(TypeName = "char"), MaxLength(5)]
public string TransactionSet { get; set; }
[Column(TypeName = "varchar"), MaxLength(80)]
public string Qualifier { get; set; }
[MaxLength(30)]
public string ReferenceID { get; set; }
[MaxLength(80)]
public string Description { get; set; }
public int ID { get; set; }
}
EF6以后,您可以使用Custom Conventions
更改此DbContext
。