2012-05-11 2 views
1

Я использую LinqToSql для веб-приложения mvc. Если многие пользователи попадают в веб-приложение примерно в одно и то же время, я вижу ошибку An item with the same key has already been added.. Стек для этой ошибки выглядит следующим образом:LinqToSql - элемент с тем же ключом уже добавлен

[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. Там, мой т е р получает хранилище от Ninject так:

public CustomRoleProvider() 
    { 
     _roleRepository = NinjectMVC3.Resolve<IRoleRepository>(); 
    } 

Метод, ошибки:

public override string[] GetRolesForUser(string username) 
    { 
     return _roleRepository.GetRolesForUser(username); 
    } 

В моем репо не что иное, как Linq запрос, который возвращает данные - репо конкретизирует контекст внутри, ничто не является статичным или общим.

Любые идеи, почему это может произойти?

ответ

1

Я не знаю, имеет ли Ninject возможность сделать это, но он должен возвращать новый isntance требуемого контекста каждый раз, когда вы вызываете решение.

Это связано с тем, что контексты EF не являются потокобезопасными.

Например, я использую Castle.Windsor как мой IoC по выбору, и у него есть опция LifeStyle, устанавливающая его Transient вместо Singleton (по умолчанию), получает желаемое поведение.

+0

+1 Контекст должен представлять собой единицу работы, поэтому для него не имеет смысла делиться между потоками – AHM

0
private object _lockHandle=new object(); 

public override string[] GetRolesForUser(string username) 
    { 
lock(_lockHandle){ 
     return _roleRepository.GetRolesForUser(username); 
} 
    } 
Смежные вопросы