我使用LinqToSql作为mvc Web应用程序。如果很多人几乎同时点击了网络应用,我发现An item with the same key has already been added.
错误。此错误的堆栈如下所示:
[ArgumentException: An item with the same key has already been added.]
System.Collections.Generic.Dictionary`2.Insert(TKey key, TValue value, Boolean add) +12673712
System.Data.Linq.DataContext.GetTable(MetaTable metaTable) +286
System.Data.Linq.DataContext.GetTable() +100
CableSense.Domain.Repository.Concrete.RoleRepository.GetRolesForUser(String userName) in c:\BuildAgent\work\271278ff356daf1\CableSense.Domain\Repository\Concrete\RoleRepository.cs:84
此仅发生在我的RoleRrovider类中,该类是.net RoleProvider的自定义实现。在那里,我的ctor从Ninject获取一个存储库,如下所示:
public CustomRoleProvider()
{
_roleRepository = NinjectMVC3.Resolve<IRoleRepository>();
}
错误的方法:
public override string[] GetRolesForUser(string username)
{
return _roleRepository.GetRolesForUser(username);
}
在我的repo中只有一个返回数据的linq查询 - repo在内部实例化一个上下文,没有任何内容是静态的或共享的。
为什么会发生这种情况的任何想法?
答案 0 :(得分:1)
我不知道Ninject是否可以选择执行此操作,但每次调用resolve时都应该返回所需上下文的新isntance。
这是因为EF上下文不是线程安全的。
例如,我使用Castle.Windsor作为我选择的IoC,它有一个LifeStyle选项,将其设置为Transient而不是Singleton(这是默认设置)获得所需的行为。
答案 1 :(得分:0)
private object _lockHandle=new object();
public override string[] GetRolesForUser(string username)
{
lock(_lockHandle){
return _roleRepository.GetRolesForUser(username);
}
}