C#常用多線程(線程同步,事件觸發(fā),信號(hào)量,互斥鎖,共享內(nèi)存,消息隊(duì)列)
C#常用多線程(線程同步,事件觸發(fā),信號(hào)量,互斥鎖,共享內(nèi)存,消息隊(duì)列)
代碼如下所示:
using System;
using System.Threading;
using System.Windows.Forms;
using UtilForm.Util;
namespace UtilForm
{
// 線程同步,事件觸發(fā),信號(hào)量,互斥鎖,共享內(nèi)存,消息隊(duì)列
public partial class frmUIThread : Form
{
SynchronizationContext context;// 子線程同步到主線程
public delegate void DoSth(object sender); // 事件觸發(fā)
public event DoSth OnMyEvent;
Mutex mutex = new Mutex(); // 互斥鎖,鎖住某一資源
ManualResetEvent manualReset = new ManualResetEvent(false); // 信號(hào)量,流程上鎖定,A執(zhí)行完再執(zhí)行B
public frmUIThread()
{
InitializeComponent();
}
private void MainForm_Load(object sender, EventArgs e)
{
context = SynchronizationContext.Current;
OnMyEvent += new DoSth(ProgressEvent);
Console.WriteLine("主線程id:" + Thread.CurrentThread.ManagedThreadId);
// 線程同步
ThreadPool.QueueUserWorkItem(state =>
{
Console.WriteLine("子線程id:" + Thread.CurrentThread.ManagedThreadId);
for (int i = 0; i < 3; ++i)
{
//textBox1.Text = i.ToString();//:“線程間操作無(wú)效: 從不是創(chuàng)建控件“textBox1”的線程訪問(wèn)它?!?
//
// 寫(xiě)法1:
//textBox1.BeginInvoke((MethodInvoker)delegate
//{
// textBox1.Text = i.ToString();
//});
//
//
// 寫(xiě)法2:
context.Send(ProgressUI, i);
if (2 == i)
{
OnMyEvent(i);// 線程間回調(diào)
}
Thread.Sleep(100);
}
});
// 模擬互斥量
ThreadPool.QueueUserWorkItem(new WaitCallback(ThdMutex));
ThreadPool.QueueUserWorkItem(new WaitCallback(ThdMutex));
// 模擬信號(hào)量在B流程中阻塞,A流程結(jié)束,繼續(xù)B流程
ThreadPool.QueueUserWorkItem(state =>
{
Console.WriteLine("模擬信號(hào)量,A開(kāi)始執(zhí)行");
Thread.Sleep(5000);
manualReset.Set();// 釋放
Console.WriteLine("模擬信號(hào)量,A執(zhí)行完成");
});
ThreadPool.QueueUserWorkItem(state =>
{
Console.WriteLine("模擬信號(hào)量,B等待執(zhí)行");
manualReset.Reset();// 重置信號(hào)量
manualReset.WaitOne();// 停止
//manualReset.WaitOne(2000);// 帶超時(shí)停止
Console.WriteLine("模擬信號(hào)量,B執(zhí)行完成");
});
// 共享內(nèi)存
ThreadPool.QueueUserWorkItem(state =>
{
MemoryCacheHelper.SetCache("10001", "123");
});
ThreadPool.QueueUserWorkItem(state =>
{
Thread.Sleep(5000);
var val = MemoryCacheHelper.GetCache("10001");
Console.WriteLine($"共享內(nèi)存:{val}");
});
}
/// <summary>
/// 更新UI
/// </summary>
/// <param name="obj"></param>
private void ProgressUI(object obj)
{
Console.WriteLine("更新UI線程::當(dāng)前線程id:" + Thread.CurrentThread.ManagedThreadId);
textBox1.Text = obj.ToString();
}
/// <summary>
/// 事件處理
/// </summary>
private void ProgressEvent(object sender)
{
Console.WriteLine($"線程事件觸發(fā):{sender.ToString()}");
}
void ThdMutex(object obj)
{
mutex.WaitOne();
Console.WriteLine($"互斥線程 {Thread.CurrentThread.ManagedThreadId} 正在執(zhí)行任務(wù)");
Thread.Sleep(1000);
Console.WriteLine($"互斥線程 {Thread.CurrentThread.ManagedThreadId} 任務(wù)執(zhí)行完畢");
mutex.ReleaseMutex();
}
}
}using System;
using System.Runtime.Caching;
namespace UtilForm.Util
{
public class MemoryCacheHelper
{
private static MemoryCache cache = MemoryCache.Default;// MemoryCache ObjectCache
/// <summary>
/// 讀取緩存
/// </summary>
/// <param name="key"></param>
/// <returns></returns>
public static object GetCache(string key)
{
var obj = cache.Get(key);
//Console.WriteLine($"-讀取緩存[{key}]:{JsonConvert.SerializeObject(obj, Formatting.Indented)}");
return obj;
}
/// <summary>
/// 寫(xiě)入緩存
/// </summary>
/// <param name="key"></param>
/// <param name="obj"></param>
/// <param name="timeout">過(guò)期時(shí)間,默認(rèn)7200秒</param>
public static void SetCache(string key, object obj, int timeout = 7200)
{
cache.Set(key, obj, DateTimeOffset.Now.AddSeconds(timeout));
//Console.WriteLine($"-寫(xiě)入緩存[{key}]:{JsonConvert.SerializeObject(obj, Formatting.Indented)}");
}
/// <summary>
/// 刪除緩存
/// </summary>
/// <param name="key"></param>
public static void RemoveCache(string key)
{
cache.Remove(key);
//Console.WriteLine($"-移除緩存[{key}]");
}
}
}
到此這篇關(guān)于C#常用多線程(線程同步,事件觸發(fā),信號(hào)量,互斥鎖,共享內(nèi)存,消息隊(duì)列)的文章就介紹到這了,更多相關(guān)C#多線程內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
一文帶你吃透C#中面向?qū)ο蟮南嚓P(guān)知識(shí)
這篇文章主要為大家詳細(xì)介紹了C#中面向?qū)ο蟮南嚓P(guān)知識(shí),文中的示例代碼講解詳細(xì),對(duì)我們學(xué)習(xí)C#有一定的幫助,需要的小伙伴可以參考一下2023-02-02
詳解C#對(duì)路徑...的訪問(wèn)被拒絕解決過(guò)程
這篇文章主要介紹了詳解C#對(duì)路徑...的訪問(wèn)被拒絕解決過(guò)程,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-12-12
詳解C#如何在不同工作簿之間復(fù)制選定單元格區(qū)域
處理Excel文檔時(shí),我們經(jīng)常需要將數(shù)據(jù)整合到一個(gè)工作表以便于我們進(jìn)行管理或數(shù)據(jù)對(duì)比。本文將演示如何通過(guò)編程方式將選定的單元格區(qū)域從一個(gè)工作簿復(fù)制到另一個(gè)工作簿2023-02-02
jQuery uploadify在谷歌和火狐瀏覽器上傳失敗的解決方案
jquery.uploadify插件是一個(gè)基于jquery來(lái)實(shí)現(xiàn)上傳的,這個(gè)插件很好用,每一次向后臺(tái)發(fā)送數(shù)據(jù)流請(qǐng)求時(shí),ie會(huì)自動(dòng)把本地cookie存儲(chǔ)捆綁在一起發(fā)送給服務(wù)器。但firefox、chrome不會(huì)這樣做,他們會(huì)認(rèn)為這樣不安全,下面介紹下jQuery uploadify上傳失敗的解決方案2015-08-08

