網(wǎng)站首頁 編程語言 正文
Timer實現(xiàn)Tick使用精度
我們想在C#中實現(xiàn)一秒鐘執(zhí)行n次的一個事件,然后其他方法可以監(jiān)聽這個事件,最終實現(xiàn)每一幀隨著Tick改變,我們的倒計時開始計數(shù).
在使用Timer過程中發(fā)現(xiàn)Timer的精度有問題,最高頻率是1秒調(diào)用62次,不滿足我們的使用需求,100幀每秒,于是我們采用了別的方式實現(xiàn)這樣的功能
實現(xiàn)效果
實現(xiàn)誤區(qū)
我們最早實現(xiàn)方法是直接開啟一個新線程,在線程內(nèi)部開啟一個Timer,設(shè)定Timer的延遲,但是最終發(fā)現(xiàn)它的執(zhí)行精度最高只能達(dá)到62幀每秒,我就算把Timer的間隔時間設(shè)置為1ms也是只能執(zhí)行62幀每秒,原因我們初步推測是因為Timer的精度不足.
解決思路
于是我們前往了C#源碼查看,發(fā)現(xiàn)源碼是通過Stopwatch實現(xiàn)的Timer類,Stopwatch類主要是一個倒計時秒表,既然知道是什么東西了那么就好實現(xiàn)了,我們開啟一個新線程,保證線程一直執(zhí)行就加上無限循環(huán)while(true). 在外部 我們開啟Stopwatch,然后我們在while中判斷秒表是否達(dá)到我們的需求,如果達(dá)到了那么就調(diào)用一個事件,然后在外部監(jiān)聽這個事件,就可以實現(xiàn)了
代碼片段
class Program
{
/// <summary>
/// 10ms trigger ont time
/// </summary>
private const int tickTime = 10;
private static Action<long> Tick;
static void Main(string[] args)
{
Tick += (x) => { Console.WriteLine("Time:" + x); };
Thread t = new Thread(() =>
{
Stopwatch s = new Stopwatch();
s.Start();
long temp = 0;
while (true)
{
if (s.ElapsedMilliseconds >= temp + 10)
{
temp = s.ElapsedMilliseconds;
Tick?.Invoke(temp);
}
}
});
t.IsBackground = true;
t.Start();
while (true) { }
}
}
效率
測試上述代碼后發(fā)現(xiàn),設(shè)定為10ms執(zhí)行一次的,按照理論上執(zhí)行次數(shù)是1秒鐘100幀,但實際效果是90幀,于是我們得出結(jié)論,使用這種方式在10ms的時候,效率是90%,然后我們測試了1ms執(zhí)行一次的效果,效率為50%,也就是1秒鐘執(zhí)行了500次.
三種Timer組件的區(qū)別
timer計時器,每隔間隔的時間就會觸發(fā)事件。??
1. System.Windows.Forms.Timer ?
--應(yīng)用于Windows應(yīng)用程序,基于UI,獨占一個線程。
--屬性 interval:時間間隔 ?ms
--事件 Tick事件,如果在此事件中執(zhí)行的任務(wù)過多,會發(fā)生阻塞。
--應(yīng)用 主要應(yīng)用修改UI元素(窗體的窗體屬性)
--注意事項 如果單次執(zhí)行時間超過設(shè)置的間隔時間,會影響下次觸發(fā),精度較差。
2. System.Timers.Timer 基于服務(wù)
--輕量級的計時器,每隔間隔時間,觸發(fā)Elapsed事件,可加載成控件使用,也可以利用代碼使用(System.Timers.Timer timer2 = new System.Timers.Timer()).
--應(yīng)用:服務(wù)器,獲取數(shù)據(jù)。
--局限:不可以修改UI元素,但可以通過UI元素this.invoke(action)調(diào)用委托修改UI元素。
--屬性:timer2.interval =1000;timer2.AutoReset = false;//只會印發(fā)一次就停止了。
--事件: timer2.Elapsed += Timer_Elapsed;
--啟動:timer2.start();
--停止:timer2.stop();
--優(yōu)點:如果事件里單次執(zhí)行了耗時的操作,不會使UI失去響應(yīng),不會影響下一次觸發(fā)。
3. System.Threading.Timer 基于線程
--輕量級的計時器,每隔間隔時間,回調(diào)方法執(zhí)行操作,可加載成控件使用,也可以利用代碼使用。
--回調(diào)方法原型:public Timer(TimerCallback callback,object state,int dueTime,int period);
參數(shù)1(state):要使用信息的對象或者設(shè)為null;
參數(shù)2(dueTime):延遲啟動的時間,單位ms;
參數(shù)3(period): 時間間隔,ms,period 時間間隔 設(shè)置為0或者-1,只會執(zhí)行一次;Change方法可以讓計時器重新啟動。 ? ? ??
--demo
System.Threading.Timer timer3 = new System.Threading.Timer(new System.Threading.TimerCallback(o=>{
? ? ?count2+=2;
? ? ?Action<int> act = ShowCount;//定義委托
? ? ?this.Invoke(act,count2);
? ??
? ? ?}),null,0,1000);
? ? ? ? ?
? ? private void ShowCount(int count)
? ? {
? ? ? ? txtCount.Text = count.ToString();
? ? }
?--對線程池線程執(zhí)行方法的機制,也就是基于多線程的,精度比較高。
--優(yōu)點:如果事件里單次執(zhí)行了耗時的操作,不會使UI失去響應(yīng),不會影響下一次觸發(fā)。
--方法:timer3.Chang(2000,2000);//改變延遲啟動時間和時間間隔。
--停止:timer3.Dispose();
--局限:不可以修改UI元素,但可以通過UI元素this.invoke(action)調(diào)用委托修改UI元素。?
原文鏈接:https://blog.csdn.net/qq_29487981/article/details/105853688
相關(guān)推薦
- 2024-04-03 clickhouse報Ports are not available
- 2022-06-16 C#使用符號表實現(xiàn)查找算法_C#教程
- 2024-07-18 redisson分布式鎖中waittime的設(shè)置
- 2022-09-05 RocketMQ 4.9.4使用(五)關(guān)閉info日志打印
- 2022-05-15 c++11?實現(xiàn)枚舉值到枚舉名的轉(zhuǎn)換問題_C 語言
- 2023-07-27 使用Echarts圖表時,頁面切換后并且改變頁面窗口大小,再切回原來頁面Echarts圖表顯示有問題
- 2023-10-15 el-input有時候添加不了有時候刪不了
- 2022-12-22 Object?arrays?cannot?be?loaded?when?allow_pickle=F
- 最近更新
-
- window11 系統(tǒng)安裝 yarn
- 超詳細(xì)win安裝深度學(xué)習(xí)環(huán)境2025年最新版(
- Linux 中運行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎(chǔ)操作-- 運算符,流程控制 Flo
- 1. Int 和Integer 的區(qū)別,Jav
- spring @retryable不生效的一種
- Spring Security之認(rèn)證信息的處理
- Spring Security之認(rèn)證過濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權(quán)
- redisson分布式鎖中waittime的設(shè)
- maven:解決release錯誤:Artif
- restTemplate使用總結(jié)
- Spring Security之安全異常處理
- MybatisPlus優(yōu)雅實現(xiàn)加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務(wù)發(fā)現(xiàn)-Nac
- Spring Security之基于HttpR
- Redis 底層數(shù)據(jù)結(jié)構(gòu)-簡單動態(tài)字符串(SD
- arthas操作spring被代理目標(biāo)對象命令
- Spring中的單例模式應(yīng)用詳解
- 聊聊消息隊列,發(fā)送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠(yuǎn)程分支