1 Windows程序开机自启动的原理

windows系统在开机的时候会主动从注册表处HKEY_CURRENT_USER\Software\Microsoft\Windows\CurrentVersion\Run读取该路径下所有的键值对并启动相关软件,所以我们如果要设置某个程序开机自启动就只需要在注册表HKEY_CURRENT_USER\Software\Microsoft\Windows\CurrentVersion\Run下按规则写入键值对即可,键即为程序名称,值则为该程序在本机上的存放的全路径。

Duilib – 程序开机自启动-StubbornHuang Blog

2 duilib程序开机自启动实现

在duilib程序的winmain函数中初始化com库完成之后调用WindowsBootAutoStart函数:

int APIENTRY _tWinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPTSTR lpCmdLine, int nCmdShow)
{
    // 第一步: 实例句柄与渲染类关联
    CPaintManagerUI::SetInstance(hInstance);
    //CPaintManagerUI::SetResourcePath(CPaintManagerUI::GetInstancePath());
    CWndShadow::Initialize(hInstance);//窗体阴影初始化

    MyRegisterWnd(_T("DuilibWnd"), DuilibWndDisplayProc);

    // 第二步:初始化COM库, 为加载COM库提供支持
    HRESULT Hr = ::CoInitialize(NULL);
    if (FAILED(Hr))
    {
        return 0;
    }

    // 设置程序开机自启
    WindowsBootAutoStart();

}

WindowsBootAutoStart函数实现如下:

void WindowsBootAutoStart()
{
    HKEY hKey;
    LPCTSTR strRegPath = "SOFTWARE\\Microsoft\\Windows\\CurrentVersion\\Run";

    //1 找到系统的启动项  
    if (RegOpenKeyEx(HKEY_CURRENT_USER, strRegPath, 0, KEY_ALL_ACCESS, &hKey) == ERROR_SUCCESS) ///打开启动项
    {
        //2 得到本程序自身的全路径
        TCHAR strExeFullDir[MAX_PATH];
        GetModuleFileName(NULL, strExeFullDir, MAX_PATH);

        //3 判断注册表项是否已经存在
        TCHAR strDir[MAX_PATH] = {};
        DWORD nLength = MAX_PATH;
        long result = RegGetValue(hKey, nullptr, "MyClient", RRF_RT_REG_SZ, 0, strDir, &nLength);

        //4 已经存在
        if (result != ERROR_SUCCESS || _tcscmp(strExeFullDir, strDir) != 0)
        {
            //5 添加一个子Key,并设置值,"MyClient"是应用程序名字(不加后缀.exe) 
            RegSetValueEx(hKey, "MyClient", 0, REG_SZ, (LPBYTE)strExeFullDir, (lstrlen(strExeFullDir) + 1) * sizeof(TCHAR));

            //6 关闭注册表
            RegCloseKey(hKey);
        }
    }
}

在每次打开程序都会检测HKEY_CURRENT_USER\Software\Microsoft\Windows\CurrentVersion\Run路径下是否有相应的键值对,如果没有则会在该路径下添加键值对,在windows系统重启之后则会自动重启该应用程序。