[源码下载]

重新想象 Windows 8 Store Apps (48) - 多线程之其他辅助类: SpinWait, SpinLock, Volatile, SynchronizationContext, CoreDispatcher, ThreadLocal, ThreadStaticAttribute

作者:webabcd

介绍
重新想象 Windows 8 Store Apps 之 多线程操作的其他辅助类

  • SpinWait - 自旋等待
  • SpinLock - 自旋锁
  • volatile - 必在内存
  • SynchronizationContext - 在指定的线程上同步数据
  • CoreDispatcher - 调度器,用于线程同步
  • ThreadLocal - 用于保存每个线程自己的数据
  • ThreadStaticAttribute - 所指定的静态变量对每个线程都是唯一的

示例
1、演示 SpinWait 的使用
Thread/Other/SpinWaitDemo.xaml.cs

/*
* SpinWait - 自旋等待,一个低级别的同步类型。它不会放弃任何 cpu 时间,而是让 cpu 不停的循环等待
*
* 适用场景:多核 cpu ,预期等待时间非常短(几微秒)
* 本例只是用于描述 SpinWait 的用法,而不代表适用场景
*/ using System;
using System.Threading;
using Windows.UI.Xaml.Controls;
using Windows.UI.Xaml.Navigation; namespace XamlDemo.Thread.Other
{
public sealed partial class SpinWaitDemo : Page
{
public SpinWaitDemo()
{
this.InitializeComponent();
} protected override void OnNavigatedTo(NavigationEventArgs e)
{
lblMsg.Text = DateTime.Now.ToString("mm:ss.fff"); SpinWait.SpinUntil(
() => // 以下条件成立时,结束等待
{
return false;
}
// 如果此超时时间过后指定的条件还未成立,则强制结束等待
,); lblMsg.Text += Environment.NewLine;
lblMsg.Text += DateTime.Now.ToString("mm:ss.fff"); SpinWait.SpinUntil(
() => // 以下条件成立时,结束等待
{
return DateTime.Now.Second % == ;
}); lblMsg.Text += Environment.NewLine;
lblMsg.Text += DateTime.Now.ToString("mm:ss.fff");
}
}
}

2、演示 SpinLock 的使用
Thread/Other/SpinLockDemo.xaml.cs

/*
* SpinLock - 自旋锁,一个低级别的互斥锁。它不会放弃任何 cpu 时间,而是让 cpu 不停的循环等待,直至锁变为可用为止
*
* 适用场景:多核 cpu ,预期等待时间非常短(几微秒)
* 本例只是用于描述 SpinLock 的用法,而不代表适用场景
*/ using System.Collections.Generic;
using System.Threading;
using System.Threading.Tasks;
using Windows.UI.Xaml.Controls;
using Windows.UI.Xaml.Navigation; namespace XamlDemo.Thread.Other
{
public sealed partial class SpinLockDemo : Page
{
private static int _count; public SpinLockDemo()
{
this.InitializeComponent();
} protected async override void OnNavigatedTo(NavigationEventArgs e)
{
SpinLock spinLock = new SpinLock(); List<Task> tasks = new List<Task>(); // 一共 100 个任务并行执行,每个任务均累加同一个静态变量 100000 次,以模拟并发访问静态变量的场景
for (int i = ; i < ; i++)
{
Task task = Task.Run(
() =>
{
bool lockTaken = false; try
{
// IsHeld - 锁当前是否已由任何线程占用
// IsHeldByCurrentThread - 锁是否由当前线程占用
// 要获取 IsHeldByCurrentThread 属性,则 IsThreadOwnerTrackingEnabled 必须为 true,可以在构造函数中指定,默认就是 true // 进入锁,lockTaken - 是否已获取到锁
spinLock.Enter(ref lockTaken); for (int j = ; j < ; j++)
{
_count++;
}
}
finally
{
// 释放锁
if (lockTaken)
spinLock.Exit();
}
}); tasks.Add(task);
} // 等待所有任务执行完毕
await Task.WhenAll(tasks); lblMsg.Text = "count: " + _count.ToString();
}
}
}

3、演示 volatile 的使用
Thread/Other/VolatileDemo.xaml

<Page
x:Class="XamlDemo.Thread.Other.VolatileDemo"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:local="using:XamlDemo.Thread.Other"
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
mc:Ignorable="d"> <Grid Background="Transparent">
<StackPanel Margin="120 0 0 0"> <TextBlock FontSize="14.667" LineHeight="20">
<Run>如果编译器认为某字段无外部修改,则为了优化会将其放入寄存器</Run>
<LineBreak />
<Run>标记为 volatile 的字段,则必然会被放进内存</Run>
<LineBreak />
<Run>编写 Windows Store Apps 后台任务类的时候,如果某个字段会被后台任务的调用者修改的话,就要将其标记为 volatile,因为这种情况下编译器会认为此字段无外部修改</Run>
</TextBlock> </StackPanel>
</Grid>
</Page>

4、演示 SynchronizationContext 的使用
Thread/Other/SynchronizationContextDemo.xaml.cs

/*
* SynchronizationContext - 在指定的线程上同步数据
* Current - 获取当前线程的 SynchronizationContext 对象
* Post(SendOrPostCallback d, object state) - 同步数据到此 SynchronizationContext 所关联的线程上
*/ using System;
using Windows.System.Threading;
using Windows.UI.Xaml.Controls;
using Windows.UI.Xaml.Navigation; namespace XamlDemo.Thread.Other
{
public sealed partial class SynchronizationContextDemo : Page
{
System.Threading.SynchronizationContext _syncContext; public SynchronizationContextDemo()
{
this.InitializeComponent(); // 获取当前线程,即 UI 线程
_syncContext = System.Threading.SynchronizationContext.Current; ThreadPoolTimer.CreatePeriodicTimer(
(timer) =>
{
// 在指定的线程(UI 线程)上同步数据
_syncContext.Post(
(ctx) =>
{
lblMsg.Text = DateTime.Now.ToString("mm:ss.fff");
},
null);
},
TimeSpan.FromMilliseconds());
}
}
}

5、演示 CoreDispatcher 的使用
Thread/Other/CoreDispatcherDemo.xaml.cs

/*
* CoreDispatcher - 调度器,用于线程同步
*/ using System;
using Windows.System.Threading;
using Windows.UI.Core;
using Windows.UI.Xaml;
using Windows.UI.Xaml.Controls; namespace XamlDemo.Thread.Other
{
public sealed partial class CoreDispatcherDemo : Page
{
public CoreDispatcherDemo()
{
this.InitializeComponent(); // 获取 UI 线程的 CoreDispatcher
CoreDispatcher dispatcher = Window.Current.Dispatcher; ThreadPoolTimer.CreatePeriodicTimer(
(timer) =>
{
// 通过 CoreDispatcher 同步数据
// var ignored = this.Dispatcher.RunAsync(CoreDispatcherPriority.Normal,
var ignored = dispatcher.RunAsync(CoreDispatcherPriority.Normal,
() =>
{
lblMsg.Text = DateTime.Now.ToString("mm:ss.fff");
});
},
TimeSpan.FromMilliseconds());
}
}
}

6、演示 ThreadLocal 的使用
Thread/Other/ThreadLocalDemo.xaml.cs

/*
* ThreadLocal<T> - 用于保存每个线程自己的数据,T - 需要保存的数据的数据类型
* ThreadLocal(Func<T> valueFactory, bool trackAllValues) - 构造函数
* valueFactory - 指定当前线程个性化数据的初始值
* trackAllValues - 是否需要获取所有线程的个性化数据
* T Value - 当前线程的个性化数据
* IList<T> Values - 获取所有线程的个性化数据(trackAllValues == true 才能获取)
*
*
* 注:ThreadStaticAttribute 与 ThreadLocal<T> 的作用差不多
*/ using System;
using System.Threading;
using System.Threading.Tasks;
using Windows.UI.Xaml.Controls;
using Windows.UI.Xaml.Navigation; namespace XamlDemo.Thread.Other
{
public sealed partial class ThreadLocalDemo : Page
{
System.Threading.SynchronizationContext _syncContext; public ThreadLocalDemo()
{
this.InitializeComponent(); // 获取当前线程,即 UI 线程
_syncContext = System.Threading.SynchronizationContext.Current;
} protected override void OnNavigatedTo(NavigationEventArgs e)
{
ThreadLocal<string> localThread = new ThreadLocal<string>(
() => "ui thread webabcd", // ui 线程的个性化数据
false); Task.Run(() =>
{
// 此任务的个性化数据
localThread.Value = "thread 1 webabcd"; _syncContext.Post((ctx) =>
{
lblMsg.Text += Environment.NewLine;
lblMsg.Text += ctx.ToString();
},
localThread.Value);
}); Task.Run(() =>
{
// 此任务的个性化数据
localThread.Value = "thread 2 webabcd"; _syncContext.Post((ctx) =>
{
lblMsg.Text += Environment.NewLine;
lblMsg.Text += ctx.ToString();
},
localThread.Value);
}); lblMsg.Text += localThread.Value;
}
}
}

7、演示 ThreadStaticAttribute 的使用
Thread/Other/ThreadStaticAttributeDemo.xaml.cs

/*
* ThreadStaticAttribute - 所指定的静态变量对每个线程都是唯一的
*
*
* 注:ThreadStaticAttribute 与 ThreadLocal<T> 的作用差不多
*/ using System;
using System.Threading.Tasks;
using Windows.UI.Xaml.Controls;
using Windows.UI.Xaml.Navigation; namespace XamlDemo.Thread.Other
{
public sealed partial class ThreadStaticAttributeDemo : Page
{
// 此静态变量对每个线程都是唯一的
[ThreadStatic]
private static int _testValue = ; System.Threading.SynchronizationContext _syncContext; public ThreadStaticAttributeDemo()
{
this.InitializeComponent(); // 获取当前线程,即 UI 线程
_syncContext = System.Threading.SynchronizationContext.Current;
} protected override void OnNavigatedTo(NavigationEventArgs e)
{
Task.Run(() =>
{
_testValue = ; _syncContext.Post((testValue) =>
{
lblMsg.Text += Environment.NewLine;
// 此 Task 上的 _testValue 的值
lblMsg.Text += "thread 1 testValue: " + testValue.ToString();
},
_testValue);
}); Task.Run(() =>
{
_testValue = ; _syncContext.Post((testValue) =>
{
lblMsg.Text += Environment.NewLine;
// 此 Task 上的 _testValue 的值
lblMsg.Text += "thread 2 testValue: " + testValue.ToString();
},
_testValue);
}); // ui 线程上的 _testValue 的值
lblMsg.Text = "ui thread testValue: " + _testValue.ToString();
}
}
}

OK
[源码下载]

重新想象 Windows 8 Store Apps (48) - 多线程之其他辅助类: SpinWait, SpinLock, Volatile, SynchronizationContext, CoreDispatcher, ThreadLocal, ThreadStaticAttribute的更多相关文章

  1. 重新想象 Windows 8 Store Apps (42) - 多线程之线程池: 延迟执行, 周期执行, 在线程池中找一个线程去执行指定的方法

    [源码下载] 重新想象 Windows 8 Store Apps (42) - 多线程之线程池: 延迟执行, 周期执行, 在线程池中找一个线程去执行指定的方法 作者:webabcd 介绍重新想象 Wi ...

  2. 重新想象 Windows 8 Store Apps (43) - 多线程之任务: Task 基础, 多任务并行执行, 并行运算(Parallel)

    [源码下载] 重新想象 Windows 8 Store Apps (43) - 多线程之任务: Task 基础, 多任务并行执行, 并行运算(Parallel) 作者:webabcd 介绍重新想象 W ...

  3. 重新想象 Windows 8 Store Apps (44) - 多线程之异步编程: 经典和最新的异步编程模型, IAsyncInfo 与 Task 相互转换

    [源码下载] 重新想象 Windows 8 Store Apps (44) - 多线程之异步编程: 经典和最新的异步编程模型, IAsyncInfo 与 Task 相互转换 作者:webabcd 介绍 ...

  4. 重新想象 Windows 8 Store Apps (45) - 多线程之异步编程: IAsyncAction, IAsyncOperation, IAsyncActionWithProgress, IAsyncOperationWithProgress

    [源码下载] 重新想象 Windows 8 Store Apps (45) - 多线程之异步编程: IAsyncAction, IAsyncOperation, IAsyncActionWithPro ...

  5. 重新想象 Windows 8 Store Apps (46) - 多线程之线程同步: Lock, Monitor, Interlocked, Mutex, ReaderWriterLock

    [源码下载] 重新想象 Windows 8 Store Apps (46) - 多线程之线程同步: Lock, Monitor, Interlocked, Mutex, ReaderWriterLoc ...

  6. 重新想象 Windows 8 Store Apps (47) - 多线程之线程同步: Semaphore, CountdownEvent, Barrier, ManualResetEvent, AutoResetEvent

    [源码下载] 重新想象 Windows 8 Store Apps (47) - 多线程之线程同步: Semaphore, CountdownEvent, Barrier, ManualResetEve ...

  7. 重新想象 Windows 8 Store Apps 系列文章索引

    [源码下载][重新想象 Windows 8.1 Store Apps 系列文章] 重新想象 Windows 8 Store Apps 系列文章索引 作者:webabcd 1.重新想象 Windows ...

  8. 重新想象 Windows 8 Store Apps (56) - 系统 UI: Scale, Snap, Orientation, High Contrast 等

    [源码下载] 重新想象 Windows 8 Store Apps (56) - 系统 UI: Scale, Snap, Orientation, High Contrast 等 作者:webabcd ...

  9. 重新想象 Windows 8 Store Apps (30) - 信息: 获取包信息, 系统信息, 硬件信息, PnP信息, 常用设备信息

    原文:重新想象 Windows 8 Store Apps (30) - 信息: 获取包信息, 系统信息, 硬件信息, PnP信息, 常用设备信息 [源码下载] 重新想象 Windows 8 Store ...

随机推荐

  1. HTML5+javascript实现图片加载进度动画效果

    在网上找资料的时候,看到网上有图片加载进度的效果,手痒就自己也写了一个. 图片加载完后,隐藏loading效果. 想看加载效果,请ctrel+F5强制刷新或者清理缓存. 效果预览:   0%   // ...

  2. Hybris电商方案介绍(企业全渠道) B2B B2C O2O建设

    1). 什么是Hybris: hybris software成立于1997年,2013年与SAP整合,成为SAP旗下的一份子,提供全渠道客户互动与商务解决方案,该解决方案能够为各机构提供客户的实时背景 ...

  3. 我也要学iOS逆向工程--函数

    大家好,这篇我开始学习函数了.先学 C 函数,然后再 OC 的吧.OC 应该复杂点的吧. 然后看看汇编情况哦! 学习函数呢,肯定要弄清楚几个事情. 1.跳转地址. 2.返回地址 3.参数 4.函数获取 ...

  4. 网友对twisted deferr的理解

    事實上Deferred的確就像是一連串的動作,用callback的形式被串在一起,我們用deferred或許可以這樣寫 d.addCallback(洗菜)d.addCallback(切菜)d.addC ...

  5. C++中文件按行读取和逐词读取 backup

    http://blog.csdn.net/zhangchao3322218/article/details/7930857 #include  <iostream>#include  &l ...

  6. ubuntu下查看环境变量

    在Windows下,查看环境变量的命令是:set,这个命令会输出系统当前的环境变量.   Linux下准确的说是REDHAT下应该如何查看呢,命令是:   export   如果你想查看某一个名称的环 ...

  7. [转]Android开发最佳实践

    ——欢迎转载,请注明出处 http://blog.csdn.net/asce1885 ,未经本人同意请勿用于商业用途,谢谢—— 原文链接:https://github.com/futurice/and ...

  8. C# 退出程序方法

    1.this.Close();   只是关闭当前窗口,若不是主窗体的话,是无法退出程序的,另外若有托管线程(非主线程),也无法干净地退出: 2.Application.Exit();  强制所有消息中 ...

  9. 解决访问StackOverFlow太慢的问题

    Stackoverflow加载时访问了被屏蔽的站点ajax.googleapis.com,导致加载缓慢,把这个站点加到Hosts里,指向127.0.0.1即可

  10. cocos2d-x开发: 整合apache http,用于自己检索多项目svn文件

    本来我的项目都是放在自己的虚拟机svn仓库中,随着仓库越来越多,有的时候需要去查看项目文件.check out到本地之后,挨个查看也是可以的,可是check out也是需要时间的,就想起了apache ...