[CLK Framework] CLK.Threading.PortableTimer - 跨平台的Timer类别
[CLK Framework] CLK.Threading.PortableTimer - 跨平台的Timer类别
问题情景
开发应用程式的时候,免不了需要加入一些定时执行的设计,例如说:定时更新画面资料、定时检查资料库内容、定时检查通讯是否断线...等等。而.NET Framework也很贴心的提供三种不同的Timer类别,用来帮助开发人员设计出定时执行的功能模组。
.NET Framework提供的三种Timer类别,可以参考Bill叔的部落格:
但是当功能模组的开发,是以跨平台执行为目标来做设计的时候,因为不是每个平台都支援上列三种Timer,所以连带的在跨平台的专案中,也就不支援参考使用.NET Framework所提供的Timer类别。像是下图中所建立的Portable Class Library专案,就无法参考使用到System.Threading.Timer类别。
遇到这样跨平台的功能模组开发,该如何提供跨平台的定时执行功能呢?

解决方案
处理跨平台的定时执行功能,其实解决方案很简单,只要建立一个跨平台的Timer类别,用来提供定时执行的功能,就可以满足这个设计需求。
模组设计
建立Timer类别最简单的设计,就是开启一条独立的执行绪,透过这个执行绪定时去执行Callback函式,这就完成了Timer类别的功能设计。但是因为.NET Framework中所提供的System.Threading.Thread并不支援跨平台使用。所以执行绪的建立工作,必须改为可以跨平台使用的System.Threading.Tasks.Task来建立执行绪,这样才能符合跨平台的开发需求。
使用跨平台的System.Threading.Tasks.Task类别来建立的执行绪,并且使用这个执行绪来定时执行Callback函式,这就完成了跨平台Timer类别的功能设计。
模组下载
程式码下载:由此进入GitHub后,点选右下角的「Download ZIP」按钮下载。
(开启程式码的时候,建议使用Visual Studio所提供的「大纲->折叠至定义」功能来折叠程式码,以能得到较适合阅读的排版样式。)
物件程式
using CLK.Diagnostics;
using System;
using System.Threading;
using System.Threading.Tasks;
namespace CLK.Threading
{
public sealed class PortableTimer : IDisposable
{
// Fields
private readonly ManualResetEvent _executeThreadEvent = new ManualResetEvent(false);
private readonly Action _callback = null;
private readonly int _interval = 0;
// Constructors
public PortableTimer(Action callback, int interval)
{
#region Contracts
if (callback == null) throw new ArgumentNullException();
#endregion
// Require
if (interval <= 0) throw new ArgumentException();
// Arguments
_callback = callback;
_interval = interval;
// Begin
Task.Factory.StartNew(this.Execute);
}
public void Dispose()
{
// End
_executeThreadEvent.Set();
}
// Methods
private void Execute()
{
while (true)
{
// Wait
if (_executeThreadEvent.WaitOne(_interval) == true)
{
return;
}
// Execute
try
{
// Callback
_callback();
}
catch (Exception ex)
{
// Fail
DebugContext.Current.Fail(string.Format("Action:{0}, State:{1}, Message:{2}", "Callback", "Exception", ex.Message));
}
}
}
}
}
使用范例
CLK.Threading.Samples.No001 - 在Windows Store App中使用PortableTimer
使用范例
using System;
using System.Threading;
using Windows.UI.Xaml;
using Windows.UI.Xaml.Controls; namespace CLK.Threading.Samples.No001
{
public sealed partial class MainPage : Page
{
// Fields
private readonly object _syncRoot = new object(); private readonly SynchronizationContext _syncContext = null; private PortableTimer _operateTimer = null; // Constructors
public MainPage()
{
// Base
this.InitializeComponent(); // SyncContext
_syncContext = SynchronizationContext.Current;
} // Handlers
private void MainPage_Loaded(object sender, RoutedEventArgs e)
{
lock (_syncRoot)
{
// Require
if (_operateTimer != null) return; // Begin
_operateTimer = new PortableTimer(this.Timer_Ticked, 500);
}
} private void MainPage_Unloaded(object sender, RoutedEventArgs e)
{
lock (_syncRoot)
{
// Require
if (_operateTimer == null) return; // End
_operateTimer.Dispose();
_operateTimer = null;
}
} private void Timer_Ticked()
{
System.Threading.SendOrPostCallback methodDelegate = delegate(object state)
{
// Display
this.TextBlock001.Text = DateTime.Now.ToString();
};
_syncContext.Post(methodDelegate, null);
}
}
}
执行结果

CLK.Threading.Samples.No002 - 在Windows Phone App中使用PortableTimer
使用范例
using System;
using System.Windows;
using Microsoft.Phone.Controls;
using System.Threading; namespace CLK.Threading.Samples.No002
{
public partial class MainPage : PhoneApplicationPage
{
// Fields
private readonly object _syncRoot = new object(); private readonly SynchronizationContext _syncContext = null; private PortableTimer _operateTimer = null; // Constructors
public MainPage()
{
// Base
this.InitializeComponent(); // SyncContext
_syncContext = SynchronizationContext.Current;
} // Handlers
private void PhoneApplicationPage_Loaded(object sender, RoutedEventArgs e)
{
lock (_syncRoot)
{
// Require
if (_operateTimer != null) return; // Begin
_operateTimer = new PortableTimer(this.Timer_Ticked, 500);
}
} private void PhoneApplicationPage_Unloaded(object sender, RoutedEventArgs e)
{
lock (_syncRoot)
{
// Require
if (_operateTimer == null) return; // End
_operateTimer.Dispose();
_operateTimer = null;
}
} private void Timer_Ticked()
{
System.Threading.SendOrPostCallback methodDelegate = delegate(object state)
{
// Display
this.TextBlock001.Text = DateTime.Now.ToString();
};
_syncContext.Post(methodDelegate, null);
}
}
}
执行结果

CLK.Threading.Samples.No003 - 在WPF中使用PortableTimer
使用范例
using System;
using System.Threading;
using System.Windows; namespace CLK.Threading.Samples.No003
{
public partial class MainWindow : Window
{
// Fields
private readonly object _syncRoot = new object(); private readonly SynchronizationContext _syncContext = null; private PortableTimer _operateTimer = null; // Constructors
public MainWindow()
{
// Base
this.InitializeComponent(); // SyncContext
_syncContext = SynchronizationContext.Current;
} // Handlers
private void Window_Loaded(object sender, RoutedEventArgs e)
{
lock (_syncRoot)
{
// Require
if (_operateTimer != null) return; // Begin
_operateTimer = new PortableTimer(this.Timer_Ticked, 500);
}
} private void Window_Unloaded(object sender, RoutedEventArgs e)
{
lock (_syncRoot)
{
// Require
if (_operateTimer == null) return; // End
_operateTimer.Dispose();
_operateTimer = null;
}
} private void Timer_Ticked()
{
System.Threading.SendOrPostCallback methodDelegate = delegate(object state)
{
// Display
this.TextBlock001.Text = DateTime.Now.ToString();
};
_syncContext.Post(methodDelegate, null);
}
}
}
执行结果

[CLK Framework] CLK.Threading.PortableTimer - 跨平台的Timer类别的更多相关文章
- [CLK Framework] CLK.Settings - 跨平台的参数存取模块
[CLK Framework] CLK.Settings - 跨平台的参数存取模块 问题情景 开发功能模块的时候,常常免不了有一些参数(例如ConnectionString),需要存放在Config檔 ...
- [Architecture Design] CLK Architecture
CLK.Prototype.Architecture 最近找数据,看到了博客园在不久之前,办了一个架构分享的活动:.Net项目分层与文件夹结构大全.看完之后觉得获益良多,接着也忍不住手痒,开始整理属于 ...
- Forms.Timer、Timers.Timer、Threading.Timer的研究
.NET Framework里面提供了三种Timer System.Windows.Forms.Timer System.Timers.Timer System.Threading.Timer 一.S ...
- Microsoft Win32 to Microsoft .NET Framework API Map
Microsoft Win32 to Microsoft .NET Framework API Map .NET Development (General) Technical Articles ...
- .net Framework Class Library(FCL)
from:http://msdn.microsoft.com/en-us/library/ms229335.aspx 我们平时在VS.net里引用的那些类库就是从这里来的 The .NET Frame ...
- System.Windows.Forms.Timer与System.Timers.Timer的区别(zz)
.NET Framework里面提供了三种Timer: System.Windows.Forms.Timer System.Timers.Timer System.Threading.Timer VS ...
- 拥抱.NET Core,如何开发一个跨平台类库 (1)
在此前的文章中详细介绍了使用.NET Core的基本知识,如果还没有看,可以先去了解“拥抱.NET Core,学习.NET Core的基础知识补遗”,以便接下来的阅读. 在本文将介绍如何配置类库项目支 ...
- [C#].NET中几种Timer的使用
这篇博客将梳理一下.NET中4个Timer类,及其用法. 1. System.Threading.Timer public Timer(TimerCallback callback, object s ...
- .NET Core与.NET Framework、Mono之间的关系
随着微软的.NET开源的推进,现在在.NET的实现上有了三个.NET Framework,Mono和.NET Core.经常被问起Mono的稳定性怎么样,后续Mono的前景如何,要回答这个问题就需要搞 ...
随机推荐
- java匿名类
一般情况下,我们需要声明一个类去继承一个接口,然后再new这个类,赋值给接口.但有时后这个类只会被调用一次,为了调用方便,那么就可以用匿名类来简化这个步骤. interface IKey{ void ...
- JavaScript - 2个等号与3个等号的区别
简言之,== equality 等同,=== identity 恒等. ==, 两边值类型不同的时候,要先进行类型转换,再比较. ===,不做类型转换,类型不同的一定不等. 或: = 赋值运算符 == ...
- [原]零基础学习视频解码之android篇系列文章
截止今天,<零基础学习视频解码系列文章>.<零基础学习在Android进行SDL开发系列文章>以及<零基础学习视频解码之android篇>系列文章基本算是告一段落了 ...
- MySQL Cluster 集群
本文转载 http://www.cnblogs.com/gomysql/p/3664783.html MySQL Cluster是一个基于NDB Cluster存储引擎的完整的分布式数据库系统.不仅仅 ...
- nodejs+express中设置登录拦截器
在nodejs+express中,采用nodejs后端路由控制用户登录后,为了加强前端的安全性控制,阻止用户通过在浏览器地址栏中输入地址访问后台接口,在app.js中需要加入拦截器进行拦截: /*** ...
- ServiceStack.Redis 中关系操作的局限与bug
redis是文档型的,nosql中难处理的是关系. 比如人可以发博客,博客可以有分类.按照传统sql中,用户表和分类表都是主表,博客表是从表,有用户的外键和分类的外键 如果使用文档型的思考方式. 为用 ...
- 两分钟了解REACTIVEX
可能在之前,你就已经看过这篇响应式编程的入门.什么?太长?好吧,这都没关系,Rx并不难,你甚至可以自己实现一个这样的框架. 知道数组吧?你当然知道,这就是: [ 14, 9, 5, 2, 10, 13 ...
- 《构建之法》之第8、9、10章读后感 ,以及sprint总结
第8章: 主要介绍了软件需求的类型.利益相关者,获取用户需求分析的常用方法与步骤.竞争性需求分析的框架NABCD,四象限方法以及项目计划和估计的技术. 1.软件需求:人们为了解决现实社会和生活中的各种 ...
- 组合数学 - 置换群的幂运算 --- poj CARDS (洗牌机)
CARDS Time Limit: 1000MS Memory Limit: 10000K Total Submissions: 1448 Accepted: 773 Description ...
- KMP算法 - 求最小覆盖子串
KMP与最小覆盖子串 最小覆盖子串:对于某个字符串s,它的最小覆盖子串指的是长度最小的子串p,p满足通过自身的多次连接得到q,最后能够使s成为q的子串. 比如: 对于s="abcab&quo ...