c# – ConcurrentDictionary和ConcurrentQueue的这种组合是线程安全的吗?

我在以下代码中使用.NET 4中的ConcurrentDictionary和ConcurrentQueue类.

这段代码是线程安全的吗?如果没有,我怎样才能使它成为线程安全的?

public class Page
{
    public string Name {get; set; }
}

public class PageQueue
{
    private ConcurrentDictionary<int,ConcurrentQueue<Page>> pages =
        new ConcurrentDictionary<int,ConcurrentQueue<Page>>();

    public void Add(int id,Page page)
    {
        if (!this.pages.ContainsKey(id))
            this.pages[id] = new ConcurrentQueue<Page>();

        this.pages[id].Enqueue(page);
    }

    public Page GetAndRemove(int id)
    {
        Page lp = null;

        if(this.pages.ContainsKey(id))
            this.pages[id].TryDequeue(out lp);

        return lp;
    }
}

演示:

public class Demo
 {
     public void RunAll()
     {
         for (int i = 0; i < 10; i++)
             Task.Factory.StartNew(() => Run());
     }

     public void Run()
     {
         PageQueue pq = new PageQueue();
         pq.Add(1,new Page());

         pq.GetAndRemove(1);
     }
 }

解决方法

正如@Femaref正确指出的那样,代码中存在一些缺陷.我建议你利用 ConcurrentDictionary<K,V>提供的许多方法使代码线程安全而不需要锁定语句:
public class PageQueue
{
    private ConcurrentDictionary<int,ConcurrentQueue<Page>>();

    public void Enqueue(int id,Page page)
    {
        var queue = this.pages.GetOrAdd(id,_ => new ConcurrentQueue<Page>());

        queue.Enqueue(page);
    }

    public bool TryDequeue(int id,out Page page)
    {
        ConcurrentQueue<Page> queue;

        if (this.pages.TryGetValue(id,out queue))
        {
            return queue.TryDequeue(out page);
        }

        page = null;
        return false;
    }
}

相关文章

项目中经常遇到CSV文件的读写需求,其中的难点主要是CSV文件...
简介 本文的初衷是希望帮助那些有其它平台视觉算法开发经验的...
这篇文章主要简单记录一下C#项目的dll文件管理方法,以便后期...
在C#中的使用JSON序列化及反序列化时,推荐使用Json.NET——...
事件总线是对发布-订阅模式的一种实现,是一种集中式事件处理...
通用翻译API的HTTPS 地址为https://fanyi-api.baidu.com/api...