8

Есть ли способ узнать, сколько времени осталось до запланированного завершения работы в Windows 7?

Под запланированным завершением работы я подразумеваю завершение работы (или перезапуск), запланированное с помощью приложения shutdown.exe из командной строки, например:

shutdown /t 600 /s

Появится окно с сообщением о том, что до выключения осталось 10 минут, а также фактическое время выключения системы.

В любое время, превышающее 600 секунд, вместо этого используется всплывающее сообщение.

Я смутно припоминаю, что XP предоставляет монитор прогресса с графическим интерфейсом и обратным отсчетом.

Как они работали? Оставшееся время отключения или запланированное время должно где-то храниться, как я могу получить к нему доступ?

Я предпочитаю какой-то метод через командную строку (т.е. не-GUI). Что еще более важно, это выполнимо в Windows, то есть без каких-либо сторонних программ?

Если это невозможно осуществить с помощью стандартных утилит Windows (и я граничу с территорией переполнения стека), существует ли Windows API (или .NET, который может даже использоваться в PowerShell!) функционировать? Любой из них будет предпочтительнее какой-либо случайной сторонней программы.

1 ответ1

5

Вопрос был интересным, поэтому я написал пакет, который может сделать работу за вас.

По сути, это таймер обратного отсчета, я не включил команду shutdown но оставил это как домашнее задание. Расчеты не точные, это не заботится о високосных годах, длине месяца, ... это главным образом для удобочитаемости, поскольку я хотел, чтобы было легко учиться.

В настоящее время два файла работают вместе: один запускает таймер, а другой, при вызове, сообщает, сколько времени прошло с момента запуска таймера. Эти два файла можно довольно легко объединить в один сценарий cmd, таким образом, вы можете сделать некоторые строки многократно используемыми (например, определенные методы).

Хорошо, давайте посмотрим на сценарии,

Вот StartTimer.cmd:

@ECHO OFF
FOR /F "skip=1 tokens=1-6" %%a IN ('wmic Path Win32_Localtime GET year^,month^,day^,hour^,minute^,second /format:TABLE ^| findstr /r "."') DO (
  SET DateD=%%a
  SET TimeH=%%b
  SET TimeM=%%c
  SET DateM=%%d
  SET TimeS=%%e
  SET DateY=%%f
)
SET /A EPOCH=((%DateY%-1970)*365*24*60*60)+(%DateM%*30*24*60*60)+(%DateD%*24*60*60)+(%TimeH%*60*60)+(%TimeM%*60)+%TimeS%
ECHO %EPOCH% > Timer.start.state

В строке, начинающейся с FOR /F "skip=1 ...:
1. Спросите wmic для текущего времени и даты.
2. Пропустить первую строку вывода, которая является заголовками, по skip=1
3. Считать значения с помощью tokens=1-6 означающих %%a-%%f разделенных пробелом or TAB.
4. Сохранить значения в переменных %DateY% , ... %TimeH% , %TimeM% , ...

После этого при запуске строки SET /A EPOCH=... мы используем выражение, которое вычисляет секунды, прошедшие с 1.1.1970 (это просто приблизительное, а не реальное время Unix.Смотрите Google, если вам нужно точно).

Затем в последней строке ECHO %EPOCH% > ... записывает рассчитанное время в файл с именем « Timer.start.state ».

А вот GetTimerValue.cmd:

@ECHO OFF

GOTO CHECKFILE

:TIMEOUT
FOR /F "skip=1 tokens=1-6" %%a IN ('wmic Path Win32_Localtime GET year^,month^,day^,hour^,minute^,second /format:TABLE ^| findstr /r "."') DO (
    SET DateD=%%a
    SET TimeH=%%b
    SET TimeM=%%c
    SET DateM=%%d
    SET TimeS=%%e
    SET DateY=%%f
)
SET /P Timer=<"Timer.start.state"
SET /A EPOCH=((%DateY%-1970)*365*24*60*60)+(%DateM%*30*24*60*60)+(%DateD%*24*60*60)+(%TimeH%*60*60)+(%TimeM%*60)+%TimeS%
SET /A Elapsed=%EPOCH%-%Timer%
ECHO "Seconds since timer started: %Elapsed%"
GOTO EOF

:CHECKFILE
IF EXIST "Timer.start.state" (
  GOTO TIMEOUT
) ELSE (
  ECHO "Timer not started. Start timer by running Timer.cmd"
)

:EOF

Здесь поток кода не является линейным сверху вниз. Сначала мы GOTO CHECKFILE чтобы команды выполнения действительно начинались со строки, которая говорит :CHECKFILE . Многие вещи здесь такие же, как и в StartTimer.cmd поэтому я объясняю только новые или измененные строки.

Сначала мы проверяем, существует ли файл состояния Timer.start.state , если нет, то завершаем работу с сообщением. Если таймер запущен и файлы существуют, мы продолжаем и делаем некоторую математику:
1. Перейти к :TIMEOUT где мы выводим прошедшие секунды. Вы можете выбрать лучший ярлык для этого ...
2. SET /P %Timer% ... читает время запуска из файла и присваивает значение переменной %Timer% .
3. SET /A Elapsed ... вычисляет секунды между временем запуска и сейчас.
4. ECHO "Seconds ... выводит рассчитанное значение (секунды с момента запуска таймера.

Как получить оставшееся время:

Вы можете вычесть %Elapsed% из %MaxTime%:

@ECHO OFF
GetTimerValue
SET /A TimeLeft=%MaxTime%-%Elapsed%
ECHO "You have %TimeLeft% seconds remaining"

Хорошо, объяснил достаточно. Вот полный скрипт Timer.cmd:

На данный момент забыл, все уже сказано. Этот пакетный файл здесь работает сам по себе, поэтому нет отдельного StartTimer.cmd или GetTimerValue.cmd а только этот Timer.cmd . В основном это все то же самое, но поток кода отличается, и использование также совершенно другое. Это, однако, не сохраняет состояние в файл, поэтому вы больше не можете получить значение таймера после выхода из системы / перезагрузки, если вам нужен этот пример использования выше для сохранения / чтения состояния в / из файла.

Использование / справка для Timer.cmd при запуске без аргументов. Я не полностью проверил это, и могут быть некоторые опечатки и т.д., Аргументы не проверены на защиту от ошибок.

@ECHO OFF

GOTO TimerUser%1

:: ===================================
:: Operation is start, (re)start timer
:: ===================================
:TimerUserstart
ECHO %2 | FINDSTR /R "^[1-9]"
IF %ERRORLEVEL%==0 (
  SET Timer.max=%2
  GOTO TimerStart
) ELSE (
  ECHO.
  ECHO   !! ERROR !!
  ECHO   You must specify maximum value.
  ECHO   ===============================
  GOTO TimerUser
)

:: =============================================
:: Operation is get, get values if timer running
:: =============================================
:TimerUserget
IF DEFINED Timer.start.state (
  GOTO TIMEOUT
) ELSE (
  ECHO.
  ECHO   !! ERROR !!
  ECHO   Timer is not started, start timer first.
  ECHO   ========================================
  GOTO TimerUser
)

:: ============
:: Usage / Help
:: ============
:TimerUser
  ECHO.
  ECHO Timer.cmd Usage: Timer.cmd operation [options]
  ECHO.
  ECHO - Operations
  ECHO   start    Start new timer, n must be specified.
  ECHO   get      Get current value of started timer.
  ECHO.
  ECHO - Options:
  ECHO   n        Max value in seconds, used to calculate remaining time.
  ECHO.
GOTO EOF

:: =============================
:: Update %Timer.epoch% variable
:: =============================
:TimerUpdateEpoch
FOR /F "skip=1 tokens=1-6" %%a IN ('wmic Path Win32_Localtime GET year^,month^,day^,hour^,minute^,second /format:TABLE ^| findstr /r "."') DO (
    SET DateD=%%a
    SET TimeH=%%b
    SET TimeM=%%c
    SET DateM=%%d
    SET TimeS=%%e
    SET DateY=%%f
)
SET /A Timer.epoch=((%DateY%-1970)*365*24*60*60)+(%DateM%*30*24*60*60)+(%DateD%*24*60*60)+(%TimeH%*60*60)+(%TimeM%*60)+%TimeS%
GOTO %Timer.Callback%

:: ===========================================
:: Start new timer destoying/resetting old one
:: ===========================================
:TimerStart
SET Timer.Callback=TimerStartC1
GOTO TimerUpdateEpoch
:TimerStartC1
SET Timer.start.state=%Timer.epoch%
GOTO EOF

:: =============================
:: Echo out current timer values
:: =============================
:TIMEOUT
SET Timer.Callback=TimerEchoJ1
GOTO TimerUpdateEpoch
:TimerEchoJ1
SET /A Timer.elapsed=%Timer.epoch%-%Timer.start.state%
SET /A Timer.remaining=%Timer.max%-%Timer.elapsed%
ECHO "Seconds since timer started: %Timer.elapsed% Time remaining %Timer.remaining%"
GOTO EOF

:EOF

Интеграция с shutdown или любой другой командой:

Файл poweroff.cmd:

@ECHO OFF
Timer start %1
shutdown -t %1 -s

Использование: poweroff <wait_seconds> для запуска выключения по Timer get и таймер для получения истекшего / оставшегося времени.

Всё ещё ищете ответ? Посмотрите другие вопросы с метками .