Я пытаюсь создать очень простую программу, которая запускает Блокнот через две минуты (все они упрощены, чтобы сделать вопрос более ясным). Я попытался объединить некоторые из примеров MSDN и, к моему удивлению, все успешно скомпилировано, при запуске программы в C:\Windows\Tasks
создается задача .job
. ]. Я не вижу новую задачу в программе с графическим интерфейсом «Планировщик заданий», и я предполагаю, что это связано с тем, что «Планировщик заданий» показывает только задачи Планировщика заданий 2.0, но я не уверен. Проблема в том, что задание не выполняется. Я прилагаю код, он немного длинный, но хорошо документирован.
Кто-нибудь может помочь разобраться? Если задача получила ошибку при запуске, где она регистрируется?
Примечание 1: я знаю, что способ, которым я вычисляю время начала, не идеален (и будет давать ошибочные результаты, когда минуты >57
). Как я уже говорил, приведенный ниже код является упрощенной версией.
Примечание 2. Я запускаю его на Win7 и предполагаю, что он совместим с API Task Scheduler 1.0. Этот код должен работать как на XP, так и на машинах с Win7 (и, надеюсь, в будущем на Win 8)
Примечание 3: Для всех, кто достаточно смел, чтобы попробовать его, если вы получаете сообщение об ошибке 0x80070050
, это означает, что задача с таким именем уже существует, удалите файл .job
или измените имя.
#include
#include
#include
#include
#include
#include
#include
#include
int main(int argc, char **argv)
{
HRESULT hr = S_OK;
ITaskScheduler *pITS;
/////////////////////////////////////////////////////////////////
// Call CoInitialize to initialize the COM library and then
// call CoCreateInstance to get the Task Scheduler object.
/////////////////////////////////////////////////////////////////
hr = CoInitialize(NULL);
if (SUCCEEDED(hr))
{
hr = CoCreateInstance(CLSID_CTaskScheduler,
NULL,
CLSCTX_INPROC_SERVER,
IID_ITaskScheduler,
(void **) &pITS);
if (FAILED(hr))
{
CoUninitialize();
return 1;
}
}
else
{
return 1;
}
/////////////////////////////////////////////////////////////////
// Call ITaskScheduler::NewWorkItem to create new task.
/////////////////////////////////////////////////////////////////
LPCWSTR pwszTaskName;
ITask *pITask;
IPersistFile *pIPersistFile;
pwszTaskName = L"Test Task";
hr = pITS->NewWorkItem(pwszTaskName, // Name of task
CLSID_CTask, // Class identifier
IID_ITask, // Interface identifier
(IUnknown**)&pITask); // Address of task
// interface
pITS->Release(); // Release object
if (FAILED(hr))
{
CoUninitialize();
fprintf(stderr, "Failed calling NewWorkItem, error = 0x%x\n",hr);
return 1;
}
/////////////////////////////////////////////////////////////////
//Set Comment, Name, Working dir, Params
/////////////////////////////////////////////////////////////////
pITask->SetComment(L"This is a comment");
pITask->SetApplicationName(L"C:\\Windows\\System32\\notepad.exe");
pITask->SetWorkingDirectory(L"C:\\Windows\\System32");
pITask->SetParameters(L"");
///////////////////////////////////////////////////////////////////
// Call ITask::CreateTrigger to create new trigger.
///////////////////////////////////////////////////////////////////
ITaskTrigger *pITaskTrigger;
WORD piNewTrigger;
hr = pITask->CreateTrigger(&piNewTrigger,
&pITaskTrigger);
if (FAILED(hr))
{
wprintf(L"Failed calling ITask::CreatTrigger: ");
wprintf(L"error = 0x%x\n",hr);
pITask->Release();
CoUninitialize();
return 1;
}
//////////////////////////////////////////////////////
// Define TASK_TRIGGER structure. Note that wBeginDay,
// wBeginMonth, and wBeginYear must be set to a valid
// day, month, and year respectively.
//////////////////////////////////////////////////////
TASK_TRIGGER pTrigger;
ZeroMemory(&pTrigger, sizeof (TASK_TRIGGER));
LPSYSTEMTIME lpSystemTime;
GetLocalTime(lpSystemTime);
// Add code to set trigger structure?
pTrigger.wBeginDay = lpSystemTime->wDay; // Required
pTrigger.wBeginMonth = lpSystemTime->wMonth; // Required
pTrigger.wBeginYear =lpSystemTime->wYear; // Required
pTrigger.cbTriggerSize = sizeof (TASK_TRIGGER);
pTrigger.wStartHour = lpSystemTime->wHour;
pTrigger.wStartMinute = lpSystemTime->wMinute + 2;
pTrigger.TriggerType = TASK_TIME_TRIGGER_DAILY;
pTrigger.Type.Daily.DaysInterval = 1;
///////////////////////////////////////////////////////////////////
// Call ITaskTrigger::SetTrigger to set trigger criteria.
///////////////////////////////////////////////////////////////////
hr = pITaskTrigger->SetTrigger (&pTrigger);
if (FAILED(hr))
{
wprintf(L"Failed calling ITaskTrigger::SetTrigger: ");
wprintf(L"error = 0x%x\n",hr);
pITask->Release();
pITaskTrigger->Release();
CoUninitialize();
return 1;
}
/////////////////////////////////////////////////////////////////
// Call IUnknown::QueryInterface to get a pointer to
// IPersistFile and IPersistFile::Save to save
// the new task to disk.
/////////////////////////////////////////////////////////////////
hr = pITask->QueryInterface(IID_IPersistFile,
(void **)&pIPersistFile);
pITask->Release();
if (FAILED(hr))
{
CoUninitialize();
fprintf(stderr, "Failed calling QueryInterface, error = 0x%x\n",hr);
return 1;
}
hr = pIPersistFile->Save(NULL,
TRUE);
pIPersistFile->Release();
if (FAILED(hr))
{
CoUninitialize();
fprintf(stderr, "Failed calling Save, error = 0x%x\n",hr);
return 1;
}
CoUninitialize();
printf("Created task.\n");
return 0;
}
Я добавил следующее:
/////////////////////////////////////////////////////////////////
//Set Flags
/////////////////////////////////////////////////////////////////
pITask->SetFlags(TASK_FLAG_RUN_ONLY_IF_LOGGED_ON);
и:
///////////////////////////////////////////////////////////////////
// Call ITask::SetAccountInformation to specify the account name
// and the account password for Test Task.
///////////////////////////////////////////////////////////////////
hr = pITask->SetAccountInformation(L"",
NULL);
if (FAILED(hr))
{
wprintf(L"Failed calling ITask::SetAccountInformation: ");
wprintf(L"error = 0x%x\n",hr);
pITask->Release();
CoUninitialize();
return 1;
}
Теперь задача отображается в планировщике задач и выполняется под СИСТЕМНОЙ учетной записью (Notepad.exe не отображается, поскольку СИСТЕМНАЯ учетная запись не интерактивный с рабочим столом или что-то в этом роде). Если я изменю его на hr = pITask->SetAccountInformation(L"MyUserName", NULL);
, то будет отображаться Блокнот. Проблема решена :).