2

我从 C# 中的默认 Windows 窗体应用程序开始,我更改的只是 Progam.cs。从主要功能,我改变了

Application.Run(new Form1());

Application.Run(new MyCustomApplicationContext());

它指的是一个自定义类 (MyCustomApplicationContext:ApplicationContext),它将我的程序作为系统托盘图标而不是 Windows 窗体运行。构造函数包含以下代码:

private NotifyIcon trayIcon = new NotifyIcon();
trayIcon.ContextMenu = new ContextMenu(
    new MenuItem[] 
    {
        new MenuItem("Exit", Exit)
    });

这允许用户右键单击该图标,为他们提供一个带有“退出”选项的上下文菜单,该选项将运行一个关闭程序的功能。

在 MyCustomApplicationContext 的构造函数结束时,我在名为 Update() 的类中调用递归函数,该函数执行 ping 函数并根据 ping 延迟更改系统托盘图标。

不幸的是,我相信因为它是递归的,所以它不允许运行任何其他代码,所以右键单击上下文菜单永远不会出现。我宁愿通过事件调用 Update() 函数,例如 System.Timers.Timer Elapsed 事件。我只是不知道事件如何工作或将代码放在哪里。

4

2 回答 2

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您的 Control。否则,您将通过尝试访问不属于计时器触发的线程的对象来生成异常。

于 2013-07-30T01:45:35.287 回答
3

没错,递归调用会阻止您的应用程序主线程,从而阻止 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;
于 2013-07-30T01:33:59.953 回答