有一個場景:一個搶購的項目,如果有5件商品。誰先搶到誰能夠買,可是如果此時此刻(這里的此時此刻如果是同樣的時間),有100人去搶這個商品,如果使用平時的方法會出現什么情況呢?你懂的。這里所說是就是有關並發的問題。
平時我們去超市購物去結賬的時候就是排隊,這里我們先讓搶購人排好隊,按時間,誰先點擊的搶購button誰就排在前面,這樣就形成了一個隊列,然后我們再對這個隊列處理。這樣就不會出現並發的問題了。
(至少能夠處理這樣簡單的並發,這里不討論太復雜的並發)
案例:
要求:有一個公布文章的接口。每公布一篇文章,調用一下接口。
(這里不用批量公布,為了解說這個)
建立一個這種處理程序類,BusinessInfoHelper.cs
namespace MyNameSpace { //隊列暫時類 public class QueueInfo { public string medias { get; set; } public string proids { get; set; } public string host { get; set; } public string userid { get; set; } public string feedid { get; set; } } public class BusinessInfoHelper { #region 解決公布時含有優質媒體時。前台頁面卡住的現象 //原理:利用生產者消費者模式進行入列出列操作 public readonly static BusinessInfoHelper Instance = new BusinessInfoHelper(); private BusinessInfoHelper() { } private Queue<QueueInfo> ListQueue = new Queue<QueueInfo>(); public void AddQueue(string medias, string proids, string host, string userid, string feedid) //入列 { QueueInfo queueinfo = new QueueInfo(); queueinfo.medias = medias; queueinfo.proids = proids; queueinfo.host = host; queueinfo.userid = userid; queueinfo.feedid = feedid; ListQueue.Enqueue(queueinfo); } public void Start()//啟動 { Thread thread = new Thread(threadStart); thread.IsBackground = true; thread.Start(); } private void threadStart() { while (true) { if (ListQueue.Count > 0) { try { ScanQueue(); } catch (Exception ex) { LO_LogInfo.WLlog(ex.ToString()); } } else { //沒有任務,歇息3秒鍾 Thread.Sleep(3000); } } } //要運行的方法 private void ScanQueue() { while (ListQueue.Count > 0) { try { //從隊列中取出 QueueInfo queueinfo = ListQueue.Dequeue(); //取出的queueinfo就能夠用了,里面有你要的東西 //下面就是處理程序了 //。。。。
。
。 } catch (Exception ex) { throw; } } } #endregion } }
以上頁面寫好后。在程序開始執行時就得啟動這個線程去不停的處理任務,那么我們在Global的Application_Start里能夠這樣寫:
//啟動公布優質媒體程序 MyNameSpace.BusinessInfoHelper.Instance.Start();
有一個問題出來了。假設我處理完隊列中的一條記錄后。想返回這條記錄的ID,這個程序好像不能完畢。我就使用了還有一個方法 Lock方法 ,把方法鎖定,詳細的例如以下,
在頁面中定義全局的鎖:
private static object lockObject= new Object();
在方法 中這樣調用:
lock(lockObject) { //........ }
假設不使用另外一種方法的全局鎖,不知各位大俠有沒有好的解決的方法,假設有,能夠跟貼,很感謝。