我开始使用C#中的默认Windows窗体应用程序,我改变的只是Progam.cs。从主要功能,我改变了
Application.Run(new Form1());
到
Application.Run(new MyCustomApplicationContext());
这是指将我的程序作为系统托盘图标而不是Windows窗体运行的自定义类(MyCustomApplicationContext:ApplicationContext)。构造函数包含以下代码:
private NotifyIcon trayIcon = new NotifyIcon();
trayIcon.ContextMenu = new ContextMenu(
new MenuItem[]
{
new MenuItem("Exit", Exit)
});
这允许用户右键单击该图标,为他们提供一个带有“退出”选项的上下文菜单,该选项将运行一个关闭程序的功能。
在MyCustomApplicationContext的构造函数的末尾,我在名为Update()的类中调用一个递归函数,该函数执行ping函数并根据ping延迟更改系统托盘图标。
不幸的是我相信因为它是递归的,所以不允许运行任何其他代码,因此右键单击上下文菜单不会显示。我宁愿通过一个事件调用Update()函数,比如System.Timers.Timer Elapsed事件。我只是不知道事件是如何工作的或代码放在何处。
答案 0 :(得分:5)
正如dmay所说,你可以使用Timer类来调用你的更新函数;但是,通过阅读您的描述,听起来您还需要考虑线程。
...
aTimer = new System.Timers.Timer(10000);
// Hook up the Elapsed event for the timer.
aTimer.Elapsed += UpdateTimer;
aTimer.Interval = 2000;
aTimer.Enabled = true;
...
public delegate void delUpdate(); // This is your delegate. Put it in your MyCustomApplicationContext class.
// This method will invoke your delegate method.
public void UpdateTimer(object sender, ElapsedEventArgs e)
{
this.Invoke((delUpdate)Update);
}
使用Invoke方法的原因是计时器将从另一个线程运行,如果您想调用更新用户界面的方法,则需要Invoke您的控件。否则,您将通过尝试访问不属于计时器正在触发的线程的对象来生成异常。
答案 1 :(得分:3)
你是对的,递归调用阻止你的app主线程停止GUI中的任何交互。如果您有Windows应用程序,则将其标记为“(未响应)”。
您可以使用Timer类
http://msdn.microsoft.com/en-us/library/system.timers.timer.aspx
var timer = new Timer(tick_milliseconds);
timer.Elapsed += DoOnTimerClick;
timer.Enabled = true;