在 Windows 启动时运行的 c# 程序表现异常

c# Program that runs on Windows Startup behaving strangely

我有一个正在测试的程序 - 仅 运行 在一些桌面上运行 - 有时在 Windows 启动时会出现非常奇怪的行为。

程序是这样开始的:

public frmClient()
{
    _version = ApplicationDeployment.IsNetworkDeployed ? ApplicationDeployment.CurrentDeployment.CurrentVersion : _version;
    _nicText = string.Format("My Program ({0})", _version.ToString());

    Logger.LogToFile(string.Format("Startup - Build: {0}, Major: {1}, MajorRevision: {2}, Minor: {3}, MinorRevision: {4}, Revision {5}", 
        _version.Build, 
        _version.Major, 
        _version.MajorRevision, 
        _version.Minor, 
        _version.MinorRevision, 
        _version.Revision));

    InitializeComponent();
}

表单实际上开始最小化到系统托盘,在加载中它使用 _nicText 设置通知图标的工具提示,如下所示:

private void frmClient_Load(object sender, EventArgs e)
{
    try
    {
        Logger.LogToFile("frmClient.frmClient_Load");

        nicMain.Text = _nicText + " - NOT CONNECTED";
    }
    catch (Exception ex)
    {
        MessageBox.Show(ex.Message, Application.ProductName, MessageBoxButtons.OK, MessageBoxIcon.Error);
    }
}

记录器class如下:

public static class Logger
{
    public static void LogToFile(string LogMessage)

    {
        StreamWriter log;

        if (!File.Exists("logfile.txt"))
        {
            log = new StreamWriter("logfile.txt");
        }
        else
        {
            log = File.AppendText("logfile.txt");
        }

        log.WriteLine(DateTime.Now.ToString("yyyyMMddHHmmss") + " - " + LogMessage);
        log.Close();
    }
}

如果我在本地 运行 程序,我会得到 (0.0) 版本信息,否则我会得到完整版本(当前为 1.0.0.31)

但是,有时在启动时我会将用户显示 (0.0) 作为版本,在这些情况下,日志文件中没有写入任何内容。

目前我唯一的猜测是应用程序在 Windows 准备就绪之前启动(只是盲目猜测)。这会发生吗?如果是这样,我有什么办法可以延迟到 Windows 准备好吗?我还漏掉了什么吗?

导致问题的代码不在我最初问题中发布的代码中。正如用户 Hans Passant 指出的那样,问题实际上出在我的程序启动方式上。

最初,我是这样设置程序启动的:

RegistryKey regKey = Registry.CurrentUser.OpenSubKey("SOFTWARE\Microsoft\Windows\CurrentVersion\Run", true);

if (regKey.GetValue("MyKey") == null || regKey.GetValue("MyKey").ToString() != Application.ExecutablePath.ToString())
{
    regKey.SetValue("MyKey", Application.ExecutablePath.ToString());
}

阅读 Returning ClickOnce version doesn't work when launching application on startup from the Windows Registry 处的代码后,我现在将代码更改为:

string compName = "MyCompany";
string keyName = "MyKey";
string allProgramsPath = Environment.GetFolderPath(Environment.SpecialFolder.Programs);
string shortcutPath = Path.Combine(allProgramsPath, compName);
shortcutPath = Path.Combine(shortcutPath, keyName) + ".appref-ms";

RegistryKey regKey = Registry.CurrentUser.OpenSubKey("SOFTWARE\Microsoft\Windows\CurrentVersion\Run", true);

if (regKey.GetValue(keyName) == null || regKey.GetValue(keyName).ToString() != shortcutPath)
{
    regKey.SetValue(keyName, shortcutPath);
}

现在,运行 exe 不是直接在启动时,而是 运行 ClickOnce link,它似乎可以正确处理版本信息。

再次感谢 Hans Passant 的帮助!