Я не могу разобраться в этой ошибке, потому что, когда отладчик подключен, кажется, что она не возникает.
Коллекция была изменена; операция перечисления может не выполняться
Ниже приведен код.
Это WCF-сервер в службе Windows. Метод NotifySubscribers()
вызывается службой всякий раз, когда происходит событие данных (через случайные интервалы, но не очень часто - примерно 800 раз в день).
Когда клиент Windows Forms подписывается, идентификатор подписчика добавляется в словарь подписчиков, а когда клиент отказывается от подписки, он удаляется из словаря. Ошибка возникает, когда (или после) клиент отписывается. Похоже, что при следующем вызове метода NotifySubscribers()
цикл foreach()
завершится ошибкой с ошибкой в строке темы. Метод записывает ошибку в журнал приложения, как показано в приведенном ниже коде. Когда отладчик присоединен и клиент отписывается, код выполняется нормально.
Вы видите проблему с этим кодом? Нужно ли мне делать словарь поточно-ориентированным?
[ServiceBehavior(InstanceContextMode=InstanceContextMode.Single)]
public class SubscriptionServer : ISubscriptionServer
{
private static IDictionary<Guid, Subscriber> subscribers;
public SubscriptionServer()
{
subscribers = new Dictionary<Guid, Subscriber>();
}
public void NotifySubscribers(DataRecord sr)
{
foreach(Subscriber s in subscribers.Values)
{
try
{
s.Callback.SignalData(sr);
}
catch (Exception e)
{
DCS.WriteToApplicationLog(e.Message,
System.Diagnostics.EventLogEntryType.Error);
UnsubscribeEvent(s.ClientId);
}
}
}
public Guid SubscribeEvent(string clientDescription)
{
Subscriber subscriber = new Subscriber();
subscriber.Callback = OperationContext.Current.
GetCallbackChannel<IDCSCallback>();
subscribers.Add(subscriber.ClientId, subscriber);
return subscriber.ClientId;
}
public void UnsubscribeEvent(Guid clientId)
{
try
{
subscribers.Remove(clientId);
}
catch(Exception e)
{
System.Diagnostics.Debug.WriteLine("Unsubscribe Error " +
e.Message);
}
}
}