我已经看到大多数人在使用
SingleOrDefault
时遇到此错误。然而,我正在使用FirstOrDefault
。有没有人见过这个异常?我正在使用存储库模式以使用依赖注入。
return context.Users.FirstOrDefault(p => p.Username.ToLower() == username.ToLower());
见下文:据我所知,错误来自 EntityFramework 的内部代码。
[InvalidOperationException: Sequence contains more than one matching element]
System.Linq.Enumerable.SingleOrDefault(IEnumerable`1 source, Func`2 predicate) +2668318
System.Data.Entity.ModelConfiguration.Conventions.IdKeyDiscoveryConventionImpl.MatchKeyProperty(EdmEntityType entityType, IEnumerable`1 primitiveProperties) +121
System.Data.Entity.ModelConfiguration.Conventions.KeyDiscoveryConvention.System.Data.Entity.ModelConfiguration.Conventions.IEdmConvention<System.Data.Edm.EdmEntityType>.Apply(EdmEntityType entityType, EdmModel model) +72
System.Data.Entity.ModelConfiguration.Conventions.IdKeyDiscoveryConvention.System.Data.Entity.ModelConfiguration.Conventions.IEdmConvention<System.Data.Edm.EdmEntityType>.Apply(EdmEntityType entityType, EdmModel model) +17
System.Data.Entity.ModelConfiguration.Configuration.EdmConventionDispatcher.Dispatch(TEdmDataModelItem item) +100
System.Data.Entity.ModelConfiguration.Configuration.EdmConventionDispatcher.VisitEdmEntityType(EdmEntityType item) +22
System.Data.Edm.Internal.DataModelItemVisitor.VisitCollection(IEnumerable`1 collection, Action`1 visitMethod) +138
System.Data.Edm.Internal.EdmModelVisitor.VisitEntityTypes(EdmNamespace edmNamespace, IEnumerable`1 entityTypes) +75
System.Data.Edm.Internal.EdmModelVisitor.VisitEdmNamespace(EdmNamespace item) +88
System.Data.Entity.ModelConfiguration.Configuration.EdmConventionDispatcher.VisitEdmNamespace(EdmNamespace item) +31
System.Data.Edm.Internal.DataModelItemVisitor.VisitCollection(IEnumerable`1 collection, Action`1 visitMethod) +138
System.Data.Edm.Internal.EdmModelVisitor.VisitNamespaces(EdmModel model, IEnumerable`1 namespaces) +75
System.Data.Edm.Internal.EdmModelVisitor.VisitEdmModel(EdmModel item) +56
System.Data.Entity.ModelConfiguration.Configuration.EdmConventionDispatcher.VisitEdmModel(EdmModel item) +44
System.Data.Entity.ModelConfiguration.Configuration.ConventionsConfiguration.ApplyModel(EdmModel model) +126
System.Data.Entity.DbModelBuilder.Build(DbProviderManifest providerManifest, DbProviderInfo providerInfo) +125
System.Data.Entity.DbModelBuilder.Build(DbConnection providerConnection) +165
System.Data.Entity.Internal.LazyInternalContext.CreateModel(LazyInternalContext internalContext) +61
System.Data.Entity.Internal.RetryLazy`2.GetValue(TInput input) +111
System.Data.Entity.Internal.LazyInternalContext.InitializeContext() +417
System.Data.Entity.Internal.InternalContext.GetEntitySetAndBaseTypeForType(Type entityType) +18
System.Data.Entity.Internal.Linq.InternalSet`1.Initialize() +63
System.Data.Entity.Internal.Linq.InternalSet`1.get_InternalContext() +15
System.Data.Entity.Infrastructure.DbQuery`1.System.Linq.IQueryable.get_Provider() +37
System.Linq.Queryable.FirstOrDefault(IQueryable`1 source, Expression`1 predicate) +63
Entities.User.GetCurrentPerson(String username, KmManagerDbContext context) in C:\Users\user\Documents\Visual Studio 2010\Projects\KmManager\Entities\User.cs:85
用户.cs
public class User
{
public long Id { get; set; }
public string FirstName { get; set; }
public string LastName { get; set; }
public string Username { get; set; }
// Custom Propreties
public string FullName
{
get
{
return FirstName + " " + LastName;
}
}
public string LastNameFirst
{
get
{
return LastName + ", " + FirstName;
}
}
public static string TableName
{
get
{
return "Users";
}
}
public static User GetCurrentPerson(string username, KmManagerDbContext context)
{
try
{
// find the person who has the ad name = username
return context.Users.FirstOrDefault(p => p.Username.ToLower() == username.ToLower());
}
catch (Exception ex)
{
throw new ApplicationException("There was an error retrieving the user from the database.", ex);
}
}
}
用户配置.cs
public UserConfiguration()
{
this.ToTable(User.TableName);
this.HasKey(x => x.Id);
this.Property(x => x.Id).HasDatabaseGeneratedOption(DatabaseGeneratedOption.Identity);
this.Property(x => x.FirstName).IsRequired();
this.Property(x => x.LastName).IsRequired();
this.Property(x => x.Username).IsRequired();
}
如果有人担心答案...
我使用 BaseEntity.cs 在我的 POCO 中拆分了一些共性
BaseEntity.cs
public class BaseEntity<T> where T : BaseEntity<T>
{
public long Id { get; set; }
public class Comparer : IEqualityComparer<T>
{
public bool Equals(T x, T y)
{
if (x.Id == y.Id)
{
return true;
}
return false;
}
public int GetHashCode(T obj)
{
return (int)obj.Id;
}
}
}
这导致配置有奇怪的行为。我将所有 POCO 更改为以前的状态,一切都按预期进行。抱歉浪费时间。
而用户 POCO 看起来像这样......
用户.cs
public class User : BaseEntity<User>
{
public string FirstName { get; set; }
public string LastName { get; set; }
public string Username { get; set; }
// Custom Propreties
public string FullName
{
get
{
return FirstName + " " + LastName;
}
}
public string LastNameFirst
{
get
{
return LastName + ", " + FirstName;
}
}
public static string TableName
{
get
{
return "Users";
}
}
public static User GetCurrentPerson(string username, KmManagerDbContext context)
{
try
{
// find the person who has the ad name = username
return context.Users.FirstOrDefault(p => p.Username.ToLower() == username.ToLower());
}
catch (Exception ex)
{
throw new ApplicationException("There was an error retrieving the user from the database.", ex);
}
}
}
这个错误不是查询的问题,而是映射配置的问题。似乎无论您调用 EF 代码的查询是什么,当在类上找到重复的匹配属性时,都会抛出错误。我猜它在类公共设置器中循环遍历 PropertyInfos。可能有重复的属性名称因为 c# 区分大小写所以“MyProperty”、“myproperty”都映射到同一列,或者在我的例子中是因为自定义类索引器。
所以对于类索引器: .net 框架将索引器反映为带有参数的属性。目前无法忽略 EntityFramework 代码优先映射文件中的自定义索引器,因为 lambda 表达式无法表达这些特殊属性。由于这还不够,我的班级有一个重载的索引器(因此出现错误“序列包含多个匹配元素”)
例子:
public decimal this[int month] {
get {
return _someArray[month];
}
set {
_someArray[month] = value;
}
}
public decimal this[int front, int month] {
get {
return _someArray2D[front - 1, month - 1];
}
set {
_someArray2D[front - 1, month - 1] = value;
}
}
几乎不可能追踪。花了我一整天的时间来反复试验。
希望这对某人有帮助。
不是答案,而是提示。基于 Tekerik JustDecompile,该方法看起来像 (EntityFramework.dll v4.2.0.0):
protected override EdmProperty MatchKeyProperty(EdmEntityType entityType, IEnumerable<EdmProperty> primitiveProperties)
{
return primitiveProperties.SingleOrDefault<EdmProperty>(delegate {
return string.Concat(entityType.Name, "Id").Equals(p.Name, stringComparison);
}) ?? primitiveProperties.SingleOrDefault<EdmProperty>(delegate {
return string.Concat(entityType.Name, "Id").Equals(p.Name, stringComparison);
});
}
这个函数似乎试图找到你实体上的哪个属性是关键属性。它期待 User -> UserId 映射。我无法解释为什么 SingleOrDefault 被调用两次。
升级到.net6?检查你的 nuget 包,你的代码可能没问题...... 由于现有答案表明此异常是由一些不寻常的编码问题引起的,我想补充一点,当目标 .net 版本更改时可能会发生这种情况.
我的 C# 代码也在调用
FirstOrDefault
。它没有仅在大小写(MyProperty 和 myProperty)或索引器或公共基类方面有所不同的重复属性。 before 运行良好,但在项目从 .Net Core 3.1 升级到 .net 6. 后开始出现此错误
抛出异常不是因为查询或数据,而是因为 EF 在第一次访问数据库时进行了一些模型初始化。
解决方案:我将 EF Core nuget 包从
3.1.13
升级到 6.0.14
(没有其他数据库或代码更改),并且项目运行没有错误。
有趣的是,EFCore 包版本与 .net 核心版本非常匹配,这暗示它们是相关的。
这个 MS 答案 和 这个 SO 答案 以及下面的评论描述了更多涉及 EF Core 和 .net 版本之间不匹配的场景,这些场景导致了相同的异常,并通过 EF 升级或 .net 降级解决了。
链接稍后可能会断开,但相关的片段是:
我在尝试迁移到 .net 6 时遇到了这个问题,结果发现它只需要将 EF nuget 包更新到最新版本。关于 .net 6 和较旧的 EF 版本的某些内容运行不佳。 ...将 EF Core 更新到相应的最新版本...或更新到 .NET 6.0 和 EF Core 6.
报价归功于Max Calvin和Guru Stron。
我的项目是针对
.net6
的,所以我将这些nuget包升级到版本6.0.14
:
Microsoft.EntityFrameworkCore.SqlServer
Microsoft.EntityFrameworkCore.Tools