Kevin-moon

学习在于分享
随笔 - 34, 文章 - 3, 评论 - 485, 阅读 - 21万
  博客园  :: 首页  :: 新随笔  :: 联系 :: 订阅 订阅  :: 管理
< 2025年3月 >
23 24 25 26 27 28 1
2 3 4 5 6 7 8
9 10 11 12 13 14 15
16 17 18 19 20 21 22
23 24 25 26 27 28 29
30 31 1 2 3 4 5

奇妙的SynchronizationContext

Posted on   Kevin-moon  阅读(16354)  评论(13编辑  收藏  举报
     上一篇中已经讲了SynchronizationContext 的一些内容,现在让我们更加深入地去了解它!
     继上篇中的问题"在UI线程上对SynchronizationContext的使用,可以适用于其他线程呢?"
     OK,我们把它放置在非UI线程上,这是你用SynchronizationContext.Current的属性来获取,你会发现你得到的是null,这时候,你可能会说,既然它不存在,那么我自己创建一个SynchronizationContext对象,这样就没问题了吧!?可是,最后它并不会像UI线程中那样去工作。
    让我们看下面的例子:
复制代码
class Program
{
    
private static SynchronizationContext mT1 = null;

    
static void Main(string[] args)
    {
        
// log the thread id
        int id = Thread.CurrentThread.ManagedThreadId;
        Console.WriteLine(
"Main thread is " + id);

        
// create a sync context for this thread
        var context = new SynchronizationContext();
        
// set this context for this thread.
        SynchronizationContext.SetSynchronizationContext(context);

        
// create a thread, and pass it the main sync context.
        Thread t1 = new Thread(new ParameterizedThreadStart(Run1));
        t1.Start(SynchronizationContext.Current);
        Console.ReadLine();
    }

    
static private void Run1(object state)
    {
        
int id = Thread.CurrentThread.ManagedThreadId;
        Console.WriteLine(
"Run1 Thread ID: " + id);

        
// grab  the sync context that main has set
        var context = state as SynchronizationContext;

        
// call the sync context of main, expecting
        
// the following code to run on the main thread
        
// but it will not.
        context.Send(DoWork, null);

        
while (true)
            Thread.Sleep(
10000000);
    }

    
static void DoWork(object state)
    {
        
int id = Thread.CurrentThread.ManagedThreadId;
        Console.WriteLine(
"DoWork Thread ID:" + id);
    }
}
复制代码
输出的结果:
Main thread is 10
Run1 Thread ID: 
11
DoWork Thread ID:
11
     注意上面的输出结果,DoWork和Run1是运行在同一线程中的,SynchronizationContext并没有把DoWork带入到主线程中执行,为什么呢?!
我们可以先看SynchronizationContext的原码(SynchronizationContext原代码):
Code
注意Send和Post的部分:
复制代码
public virtual void Send(SendOrPostCallback d, Object state)
{
    d(state);
}

public virtual void Post(SendOrPostCallback d, Object state)
{
    ThreadPool.QueueUserWorkItem(
new WaitCallback(d), state);
}
复制代码
     Send就是简单在当前的线程上面去调用委托来实现,而Post是通过线程池来实现。
     这时候你也许会奇怪,为什么UI线程上,SynchronizationContext就发挥了不同的作用呢!其实在UI线程中使用的并不是SynchronizationContext这个类,而是WindowsFormsSynchronizationContext这个东东。

     它重写了Send和Post方法。至于它是如何重写实现的,这个我也不是很了解,也没有找到相关的文章,只是知道通过"消息泵"来实现的,但是细节就不清楚了,如果大家知道的话,可以告诉下我,我很想了解下!呵呵
     最后,我画了一副图,让我们更加清楚地了解SynchronizationContext在UI线程和一般线程之间的不同,
     

     上一篇:线程之间的通讯---SynchronizationContext
编辑推荐:
· 如何编写易于单元测试的代码
· 10年+ .NET Coder 心语,封装的思维:从隐藏、稳定开始理解其本质意义
· .NET Core 中如何实现缓存的预热?
· 从 HTTP 原因短语缺失研究 HTTP/2 和 HTTP/3 的设计差异
· AI与.NET技术实操系列:向量存储与相似性搜索在 .NET 中的实现
阅读排行:
· 周边上新:园子的第一款马克杯温暖上架
· Open-Sora 2.0 重磅开源!
· 分享 3 个 .NET 开源的文件压缩处理库,助力快速实现文件压缩解压功能!
· Ollama——大语言模型本地部署的极速利器
· [AI/GPT/综述] AI Agent的设计模式综述
点击右上角即可分享
微信分享提示