【多线程】——定时器Timer

本文介绍 Java 中 Timer 类的功能和使用方法,包括一次性任务、周期性任务等不同类型的调度方式,并提供实例代码。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >


在JDK库中Timer类主要负责计划任务的功能,在指定的时间开始执行某一个任务。

主要作用:设置计划任务。

执行计划任务的代码要放入TimerTask的子类中,因为TimerTask是一个抽象类。



方法schedule(TimerTask task,Date time)

在指定的日期执行一次某一任务。


1、执行任务的时间晚于当前时间:在未来执行的效果

package test;

import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.Timer;
import java.util.TimerTask;

public class Run1TimerIsDaemon {

	private static Timer timer = new Timer(true);

	static public class MyTask extends TimerTask {
		@Override
		public void run() {
			System.out.println("运行了!时间为:" + new Date());
		}
	}

	public static void main(String[] args) {
		try {
			MyTask task = new MyTask();
			SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
			String dateString = "2014-10-12 12:05:00";
			Date dateRef = sdf.parse(dateString);
			System.out.println("字符串时间:" + dateRef.toLocaleString() + " 当前时间:"
					+ new Date().toLocaleString());
			timer.schedule(task, dateRef);
		} catch (ParseException e) {
			e.printStackTrace();
		}
	}
}

运行结果:



2、计划时间早于当前时间,立即执行task任务,提前执行的效果  

public static void main(String[] args) {
		try {
			MyTask task = new MyTask();
			SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
			String dateString = "2014-10-12 09:08:00";
                        Timer timer = new Timer();
			Date dateRef = sdf.parse(dateString);
			System.out.println("字符串时间:" + dateRef.toLocaleString() + " 当前时间:"
					+ new Date().toLocaleString());
			timer.schedule(task, dateRef);
		} catch (ParseException e) {
			e.printStackTrace();
		}
	}

运行结果:



3、多个TimerTask任务及延时,Timer中可以有多个TimerTask任务,以队列的方式一个一个被顺序执行,执行的时间有可能和与其的时间不一致,所以,前面的任务可能消耗的时间较长,而后面的任务运行的时间也会被延迟。


    

方法schedule(TimerTask task,Date firstTime,long period)

在指定的日期之后,按指定的间隔周期性地无限循环的执行某一任务。

    

1、计划时间晚于当前时间,项目在未来执行

package test;

import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.Timer;
import java.util.TimerTask;

public class Run {
	static public class MyTask extends TimerTask {
		@Override
		public void run() {
			System.out.println("运行了!时间为:" + new Date());
		}
	}

	public static void main(String[] args) {
		try {
			MyTask task = new MyTask();
			SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
			String dateString = "2014-10-12 09:12:00";
			Timer timer = new Timer();
			Date dateRef = sdf.parse(dateString);
			System.out.println("字符串时间:" + dateRef.toLocaleString() + " 当前时间:"
					+ new Date().toLocaleString());
			timer.schedule(task, dateRef, 4000);
		} catch (ParseException e) {
			e.printStackTrace();
		}
	}
}

运行结果:


2、如果计划时间早于当前时间,则立即执行task任务

3、TimerTask类中的cancel()方法,将自身从任务队列中清除

4、Timer类中的cancel()方法,将任务队列中的全部任务清空



方法schedule(TimerTask task,long delay)

作用:以执行schedule(TimerTask task,long delay)方法当前的时间为参考时间,在此时间基础上延迟指定的毫秒数后执行一次TimerTask任务。

package test;

import java.util.Date;
import java.util.Timer;
import java.util.TimerTask;

public class Run {
	static public class MyTask extends TimerTask {
		@Override
		public void run() {
			System.out.println("运行了!时间为:" + new Date());
		}
	}
	public static void main(String[] args) {
		MyTask task = new MyTask();
		Timer timer = new Timer();
		System.out.println("当前时间:" + new Date().toLocaleString());
		timer.schedule(task, 7000);
	}
}

运行结果:


    

方法schedule(TimerTask task,long delay,long period)

作用:以执行schedule(TimerTask task,long delay,long period)方法当前的时间为参考时间,在此时间基础上延迟指定的毫秒数,再以某一时间间隔无限次数执行某一任务。

package test;

import java.util.Date;
import java.util.Timer;
import java.util.TimerTask;

public class Run {
	static public class MyTask extends TimerTask {
		@Override
		public void run() {
			System.out.println("运行了!时间为:" + new Date());
		}
	}

	public static void main(String[] args) {
			MyTask task = new MyTask();
			Timer timer = new Timer();
			System.out.println("当前时间:"+new Date().toLocaleString());
			timer.schedule(task, 3000,5000);
	}
}

运行效果:


方法中带有period参数的,都是无限循环执行TimerTask中的任务。




### C# 上位机多线程定时器读取数据最佳实践 #### 多线程方式的优点 在C#上位机程序中,当涉及到实时读取及显示功能时,多线程是一种常用的方法。这种方式能够有效地处理长时间运行的任务而不阻塞主线程,确保用户界面保持响应状态[^1]。 对于需要频繁访问外部设备或网络服务的应用场景来说,使用独立的工作线程可以提高系统的稳定性和效率。如果遇到连接失败等问题,可以通过增加重试机制来增强可靠性。此外,在面对复杂业务逻辑的情况下,合理设计多个子任务并发执行也有助于提升整体性能。 ```csharp private Thread _readThread; public void StartReading() { _readThread = new Thread(() => { while (true) { try { var data = ReadDataFromSource(); Invoke(new Action(() => UpdateUIData(data))); } catch (Exception ex) { HandleError(ex); ReconnectIfNecessary(); } Thread.Sleep(intervalMilliseconds); // 控制循环频率 } }); _readThread.Start(); } ``` #### 定时器方式的特点 另一方面,利用`System.Timers.Timer`或其他类型的计时器组件也可以完成周期性的数据采集工作。相比起手动管理线程生命周期而言,这种方法更加简洁直观,并且减少了潜在的风险因素,比如内存泄漏或者资源竞争等现象的发生概率[^2]。 特别是针对那些仅需定期查询固定间隔时间内的最新记录的情形,设定好触发条件之后几乎不需要额外维护成本就能满足需求。不过需要注意的是,由于每次回调都是在一个新的线程池成员里发生的,因此同样要考虑到跨线程调用UI控件的安全性问题。 ```csharp using System.Timers; Timer timer = null; public void SetupPeriodicReads(int intervalMs) { timer = new Timer(intervalMs); timer.Elapsed += OnTimedEvent; timer.AutoReset = true; timer.Enabled = true; } private void OnTimedEvent(Object source, ElapsedEventArgs e) { var data = ReadDataFromSource(); this.Invoke((MethodInvoker)delegate { UpdateUIData(data); }); } ``` #### 如何选择合适的技术方案? 为了决定究竟应该选用哪种策略作为最终解决方案,则取决于具体应用场景的要求: - 对于简单重复的任务或是希望减少发难度的情况之下,基于现有框架提供的工具类(如各种形式的计时器)往往更为便捷高效。 无论采取何种手段,都需要充分考虑如何妥善处理可能出现的各种异常状况,保障整个通信过程中的稳定性与安全性。另外,鉴于两者之间存在一定的相似之处——即都会涉及不同步的操作环境之间的交互,所以务必遵循良好的编程习惯,例如始终记得释放不再使用的资源、避免不必要的锁争用等等[^3]。
评论 23
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值