Кафедра: Автоматика и Информационные Технологии
Структуры и функции времени
Дается обзор и описание встроенных структур данных, связанных с датами и временем. Дается описание основных функций времени. Приводятся практические задания для лучшего усвоения материала, а также указания по подготовке и выполнению лабораторных заданий.
Предположим, что мы задались целью выяснить наше географическое местоположение с помощью оказавшегося под рукой персонального компьютера. Разумеется, мы не сможем узнать точных координат, но определить, в пределах какого часового пояса находится компьютер, довольно легко. Для этого достаточно знать разницу между нашим поясным временем и Гринвичским средним временем. Стандартная библиотека Си располагает функцией time(), которая сообщает, сколько секунд прошло с того момента, когда на Гринвичском меридиане было 00:00:00 1 января 1970 года (этот момент - некий аналог Рождества Христова - для ясности называется "эпохой").
#include <time.h>
long time(long *timeptr);
Указанное число секунд записывается также по адресу переменной timeptr (если timeptr = NULL, эта запись не производится). Обычный год содержит 31 536000 секунд, плюс 86400 секунд (один день) для високосного года, час содержит 3600 секунд. Не составит труда написать программу, вычисляющую все необходимые остатки от деления и добавляющую к ним разницу по сравнению с Гринвичем. Каков же будет ее результат? Ошибка в определении поясного времени будет, вероятно, примерно соответствовать земному полушарию. Наш компьютер "считает", что он находится в Редмонде, штат Вашингтон, США.
Причина этого несоответствия – MS DOS, так как в Редмонде находится фирма Microsoft, создавшая эту операционную систему. Среди зарезервированных имен переменных окружения, которые можно задать командой SET, фигурирует переменная с коротким именем TZ (Time: Zone), по значению которой функция tiтe() и устанавливает разницу во времени с GMT (Greenwich mean time). Если TZ не задана, ей присваивается значение PST8PDT (Pacific time zone), которое и приводит к ошибке.
Поясним характер и порядок ее использования. Установка переменной TZ определяет значения глобальных переменных daylight, timezoпe и tzname, которые используются несколькими функциями, работающими со временем и датой.:
#include <time.h>
int daylight; /* флаг летнего времени */
long timezone; /* отличие в секундах от GMT */
char *tzname[2] /* обозначение зоны времени */
Значение переменной окружения TZ представляет собой трехбуквенное обозначение зоны времени, за которым может следовать число со знаком, определяющим разницу между Гринвичским и локальным временем. Число это должно быть положительно к востоку от Гринвича и отрицательно - к западу. Кроме того, за данным числом может находиться обозначение зоны летнего времени DST (daylight-saving-time). Например, следующая команда
SEТ ТZ=EST5EDT
определяет, что зона местного времени - EST (Eastern Standard Time, или Восточно-Европейское время), и это местное время на 5 часов отстает от гринвичского; а EDT - название зоны, используемое в тот период, когда действует летнее время. Пропуск зоны DST означает, что летнее время не действует никогда:
SET ТZ=EST5
Переменная daylight принимает ненулевое значение, если зона DST присутствует в установке TZ, в противном случае daylight становится равной О. Переменной tiтezoпe присваивается разница в секундах (вычисленная преобразованием часов в TZ) между временем по Гринвичу и местным временем. Первым элементом массива tzпaтe является строка, состоящая из трех букв зоны времени из TZ, вторым элементом является строка, определяющая зону DST. Если зона DST пропущена при установке TZ, то значением tzпame[1] будет пустая строка.
struct time{
unsigned char ti_min, ti_hour, ti_hund, ti_sec;
};
struct date{
int da_year;
unsigned char da_day, da_mon; /*январь - 1*/
};
struct tm{
int tm_sec, tm_min, tm_hour, tm_day,
tm_mon, /* январь - 0*/
tm_year, /* количество лет после 1900 года*/
tm_wday, /* воскресенье – 0, суббота - 6*/
tm_yday. /* 1 января - 0*/
tm_isdst; /* флаг наличия летнего времени*/
};
typedef long time_t
Рассмотрим остальные функции стандартной библиотеки, связанные с исчислением и преобразованием времени.
Так как представление периода времени в секундах не совсем привычно и не всегда удобно для человека, функция сtimе() преобразует величину типа loпg в текстовую строку.
char *ctime(long *time);
Возвращает строку из 26 символов. Параметр time аналогичен параметру timeptr в функции time.
Посмотрим, сказывается ли на работе этой функции значение переменной TZ:
#include <time.h>
#include <stdio.h>
void main()
{
long period;
time(&period);
printf("С 01/01/1970 прошло %ld секунд\n",period);
printf("Текущая дата и время: %s \n",ctime(&period));
}
Итак, все в порядке - дата и время выводятся правильно. Единственным недостатком работы всей группы функций времени является отсутствие ;'обратной силы" - любой день до 1 января 1970 года считается днем 1 января 1970 года.
Текущее время можно получить и более замысловатым путем. Довольно обыденным является представление времени в виде нескольких компонент - номера месяца, дня месяца и недели и так далее. Для такого представления существует специальная подробная структура - tm, описанная в том же hеаdеr-файле <time.h>.
Двефункции
struct tm *gmtime(long *timeptr);
struct tm *localtime(long *timeptr);
заполняют эту структуру. Первая из них делает это для Гринвичского меридиана, а вторая, как следует из названия, для местного часового пояса.
Функция
long mktime(struct tm *timeptr);
производит обратное действие - преобразует структуру tm в количество секунд после эпохи до момента, записанного в структуре. Одновременно она устанавливает день недели. При неправильной дате или времени возвращает -1. Для некоторых неправильных дат mktime пытается подправить эту дату в самой структуре tm и не возвращает признак ошибки -1.
Функция
char *asctime(struct tm *timeptr);
преобразует структуру tm в строку, эквивалентную возвращаемой функцией ctime().
Удобному представлению времени служит еще одна функция _strtime()
char *_strtime(char *time);
Функция _strtime() копирует текущее время в буфер, на который указывает аргумент tiтe, по формату "hh:mm:ss", где hh - две цифры, представляющие час в 24-часовой записи, mm - две цифры, представляющие минуты, а ss - две цифры, соответствующие секундам. Буфер должен быть выделен заранее и его длина должна составлять, по крайней мере, девять байт.
Функция clock() сообщает вызывающей программе процессорное время, затраченное с момента начала исполнения этой программы.
clock_t clock(void);
Функция может быть использована для определения временного интервала между двумя событиями. Для перевода количества тиков в секунды необходимо воспользоваться делителем CLK_TCK.
Функция utime( ) изменяет дату и время модификации файла.
int utime(char *pathname, struct utimebuf *timeptr);
Структура utimbuf объявлена в файле <sys\utime.h>
struct utimbuf {
time_t actime; /*время доступа к файлу*/
time_t modtime; /*время последней модификации файла*/
};
Поскольку файловая система DOS поддерживает только время модификации файла, функция utime поле actime игнорирует.
Если вместо адреса этой структуры функции utiтe() передается NULL, в качестве времени модификации используется текущее системное время. Файл должен иметь разрешение на запись. Код возврата: О при успешном изменении времени модификации файла и -1 при ошибке, определяемой с помощью переменной errno:
EACCES Pathname is directory ог read-only file
EMFILE Тоо many ореn files.
ENOENT File or path name not found
Вернемся опять к системным часам. Мы не будем вдаваться в подробное рассмотрение устройства таймера, отметим, что он вырабатывает системное прерывание INT 8Н, которое называется "тиком". Тик производится примерно 18,2 раза в секунду, то есть примерно каждые 55 миллисекунд (частота осциллятора системных часов - 1,19318 МГц, деленная на 65536). Пользуясь этим, мы можем производить необходимые машинно-независимые задержки времени, например для генерации звуков. Количество тиков (происшедших с последней полуночи) с помощью прерывания INT 1АН даст нам функция _bios_tiтeofday(). С ее помощью можно также установить новое число тиков. Если с момента последнего чтения или установления тиков миновала полночь, _bios_tiтeofday() возвращает 1, в противном случае - 0. При переходе через полночь необходимо прибавить к имеющимся тикам их количество в полных сутках - 1573040 (или 0x01800B0L).
unsigned _bios_timeofday(unsigntd service, long *timeval);
Переменная timeval указывает на количество тиков. Параметр service равен 0 при получении счетчика тиков и 1 при установлении нового количества тиков.
Текст _blos_tiтeofday() выглядит, вероятно, таким образом: