Я пытаюсь создать очень простую программу, которая будет запускать Notepad через две минуты (все они упрощены, чтобы задать более четкий вопрос). Я попытался объединить некоторые из MSDN's examples, и, к моему удивлению, все скомпилированное успешно, при запуске программы в C:\Windows\Tasks
создана задача .job
. Я не вижу новую задачу в GUI-программе «Планировщик заданий», и я предполагаю, что это связано с тем, что «Планировщик заданий» показывает только задачи Task Scheduler 2.0, но я не уверен. Проблема в том, что задача не выполняется. Я прикрепляю код, он немного длинный, но хорошо документированный.Попытка создать запланированную задачу для запуска один раз с использованием C++, Task Scheduler 1.0 на win7
Может кто-нибудь помочь понять это? Если у задачи возникла ошибка при запуске, где она зарегистрирована?
Примечание 1: Я знаю, что время, которое я рассчитываю, не является идеальным (и даст ошибочные результаты в моменты, когда минуты >57
). Как я уже говорил, приведенный ниже код является упрощенной версией.
Примечание 2: Я запускаю его на win7, и я предполагаю, что он совместим с Task Scheduler 1.0 API. Этот код должен работать как на XP, так и на win7-машинах (и, надеюсь, на выигрыше 8 в будущем)
Примечание 3: Для тех, кто достаточно храбр, чтобы попробовать, если вы получите ошибку 0x80070050
, это означает, что задача с таким именем уже существует, удалите файл .job
или измените его имя.
#include <windows.h>
#include <initguid.h>
#include <ole2.h>
#include <mstask.h>
#include <msterr.h>
#include <objidl.h>
#include <wchar.h>
#include <stdio.h>
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;
}
EDIT:
Я добавил следующее:
/////////////////////////////////////////////////////////////////
//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 не отображается, поскольку учетная запись SYSTEM не является интерактивной с рабочим столом или чем-то еще). Если я изменю его на hr = pITask->SetAccountInformation(L"MyUserName", NULL);
, тогда появится Блокнот. Задача решена :).