用C ++设置windows系统时间

时间:2017-01-12 09:52:00

标签: c++ windows cmd shellexecute runas

我正在将GPS设备连接到com端口。我想把我的系统作为时间服务器。怎么做?

我尝试使用settime()SetLocalTime()SetSystemTime()等。没有人在为我工作。通过使用system("cmd")我尝试了,但我再次获得管理员权限。使用system("runas cmd")时,命令提示符将以闪烁方式打开和关闭。无法理解发生了什么。

我如何才能做到这一点?

1 个答案:

答案 0 :(得分:1)

Windows上的权限管理始终是一个棘手的部分。 MSoft认为普通凡人不应该关心它,文档通常是不完整的。

在这里,问题可能是正常进程(不以管理员身份运行)没有权限来更改系统时间。

但可能的是:

  • 检查SeSystemtimePrivilege是否被授予正在运行的进程
    • 如果只是设定时间
    • 如果没有通过ShellExectute使用与管理员相同的参数重新启动流程,则动词= "runas"

以下是代码示例(解析时间字符串并实际设置时间的部分未实现):

#include <iostream>
#include <windows.h>
#include <tchar.h>

#ifdef UNICODE
#define tcout std::wcout
#else
#define tcout std::cout
#endif

// utility used to display error messages
void errmsg() {
    DWORD err = ::GetLastError();
    LPTSTR msg;
    ::FormatMessage(FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM, NULL,
        err, LANG_NEUTRAL, (LPTSTR) &msg, 0, NULL);
    tcout << _T("Error") << std::endl;
}

int _tmain(int argc, TCHAR *argv[]) {
    // syntax check
    if (argc != 2) {
        tcout << _T("Usage ") << argv[0] << " _time_" << std::endl;
        return 1;
    }
    LPCTSTR time = argv[1];
    // get access to the current process token
    HANDLE process = ::GetCurrentProcess();
    HANDLE token;
    if (! ::OpenProcessToken(process, TOKEN_ALL_ACCESS, &token)) {
        errmsg();
        return 1;
    }
    // get priviledges from current token
    DWORD sz = 0, sz2;
    TOKEN_PRIVILEGES *privs;
    ::GetTokenInformation(token, TokenPrivileges, NULL, 0, &sz); // first only size
    if (sz == 0) {
        errmsg();
        return 1;
    }
    privs = (TOKEN_PRIVILEGES *) malloc(sz);
    if (! ::GetTokenInformation(token, TokenPrivileges, privs, sz, &sz2)) {
        errmsg();
        return 1;
    }

    // display found priviledges and look if SE_SYSTEMTIME_NAME is present
    BOOL ok = FALSE;
    for (size_t i=0; i<privs->PrivilegeCount; i++) {
        sz = 0;
        ::LookupPrivilegeName(NULL, &(privs->Privileges[i].Luid), NULL, &sz);
        LPTSTR name = (LPTSTR) malloc(sz * sizeof(TCHAR));
        if (! ::LookupPrivilegeName(NULL, &(privs->Privileges[i].Luid), name, &sz)) {
            errmsg();
            return 1;
        }
        if (0 == ::lstrcmp(name, SE_SYSTEMTIME_NAME)) {
            ok = TRUE;
        }
        // tcout << name << std::endl;
        free(name);
    }
    free(privs); // done with it
    tcout << (ok ? _T("Can") : _T("Cannot")) << _T(" change time") << std::endl;

    if (ok) {
        tcout << _T("Setting time with ") << time << std::endl;
        // actually parse the time string and call SetSystemTime
        //...
    }
    else {
        tcout << _T("Restart self as admin...") << std::endl;
        // start as cmd /K "full path" to have a chance to see eventual errors
        LPTSTR params = (LPTSTR) malloc(MAX_PATH + 7 + lstrlen(time));
        lstrcpy(params, _T(" /K \""));
        sz = ::GetModuleFileName(NULL, params + 5, MAX_PATH);
        // tcout << params + 5 << std::endl;
        ::lstrcat(params,_T("\" "));  // do not forget the trailing "
        lstrcat(params, time);
        // Let's go, the UAC control should pop to allow the privilege elevation
        if (((int) ShellExecute(NULL, _T("runas"), _T("cmd.exe"), params,
                _T("."), SW_SHOW)) < 32) {
            tcout << _T("Could not start self with elevated privileges") << std::endl;
            return 1;
        }
        free(params);
    }
    // time to clean...
    ::CloseHandle(token);
    ::CloseHandle(process);
    return 0;
}

仍然有效的事情:

  • 在管理模式下重启命令时,使用带cmd /K的控制台允许用户查看发生的情况。如果您希望更加沉默,则应使用cmd /C而使用SW_HIDE ShellExecute
  • 此程序假定新时间将作为一个简单的字符串给出,可能与您的用例相关或不相关
  • 应使用可选参数
  • 控制详细程度
  • 系统应该阻止多次尝试 runas admin (可能是第二个或可选参数)