Как компьютер определяет человеческое время?
Представим ситуацию: Вы нашли в кладовой старый компьютер и решили попробовать его включить. Компьютер запускается, хотя стоял без дела 10 лет, но есть проблема: система уверена, что сегодня 1 января 1970 года. Эта ситуация поможет разобраться, как компьютеры хранят текущее время: ведь у них нет внутри пружин или маятников, как в часах, которые могли бы отсчитывать секунды. Каким образом они показывают верное время, если отключить их на несколько часов, и почему тот же принцип не работает, если компьютер простоял в кладовке 10 лет?
Маятник как первый секундомер
Прежде, чем ответить на эти вопросы, разберемся, как измеряется точное время. Это не такая простая задача, если под рукой нет часов. Мы можем отсчитывать дни, месяцы и годы или приблизительно разбить день на часы. Но найти ориентир, который подскажет количество прошедших секунд, гораздо сложнее.
Принято считать, что первым эту задачу в XVII веке решил итальянский ученый [Галилео Галилей] (https://ru.wikipedia.org/wiki/%D0%93%D0%B0%D0%BB%D0%B8%D0%BB%D0%B5%D0%B9,_%D0%93%D0%B0%D0%BB%D0%B8%D0%BB%D0%B5%D0%BE). Галилео нужно было точно знать, сколько длился тот или иной эксперимент. Его биографы
утверждают, что сначала он считал количество своих сердцебиений, но однажды наблюдал за раскачивающимся канделябром в кафедральном соборе Пизы и заметил, что каждое колебание канделябра занимает одинаковое количество сердцебиений. Это число не меняется при затухании амплитуды движений. Так он понял, что
колебания маятника помогают точно измерять время.
Можно ли использовать колебания для подсчета времени в компьютере?
Когда в 70-е-80-е годы инженеры впервые столкнулись с задачей отображения человеческого времени в компьютере, они, как ни странно, обратились к трудам итальянского ученого, но взглянули на них под другим углом. Маятник — не единственный возможный источник колебаний. У любого современного персонального компьютера есть «сердце» — центральное процессорное устройство (ЦПУ), или просто процессор. Он устроен очень сложно и выполняет множество функций, но в данном случае интересно одно из составляющих его устройств. Когда через него проходит ток, оно испускает равномерные электрические колебания с одинаковой частотой. Частота этих колебаний измеряется в герцах (Гц) и определяет количество операций, которое процессор может выполнить за одну секунду. К сожалению, если мы будем использовать единственную доступную нам операцию в секунду для увеличения счетчика колебаний, то наш процессор будет совершенно непригоден для чего-то ещё: ему будет некогда исполнять другие команды. Конечно, современные процессоры не такие медленные, их частота измеряется в гигагерцах (ГГц), то есть в 1 000 000 000 операций в секунду. Но даже при такой мощности для измерения времени пришлось бы считать каждое колебание процессора. Для решения проблемы был придуман кварцевый генератор. Он представляет собой тончайшую кремниевую пластину, которая под воздействием электрического тока равномерно расширяется и сжимается, генерируя слабый электрический заряд на поверхности. Таким образом, механические колебания пластины сопровождаются синхронными, равномерными колебаниями электрического заряда.
Полученные равномерные колебания принято называть таймером — они позволяют процессору синхронизировать производимые им операции во времени. Таким образом, у компьютера есть своё собственное понимание времени.
От компьютерного времени к человеческому
Получается, что внутри компьютера есть таймер, но он не измеряет человеческое время — оно для него довольно бесполезно. Как правило, эту задачу на себя берут операционные системы (ОС). Поскольку ОС знает, с какой частотой работает кварцевый генератор, она может измерить время, которое проходит между срабатываниями таймера (срабатывание таймера называют тик (tick) или джиффи (jiffy)). Если генератор
работает на частоте 100 Гц, то период между тиками равен 1/100 секунды или 10 миллисекундам. Операционная система создает в памяти переменную, которую обычно называют jiffies, и увеличивает её на единицу каждый раз, когда процессор дает сигнал о новом тике.
Соответственно, чтобы узнать, как долго включен компьютер, системе достаточно умножить размер периода между тиками на количество этих самых тиков. А чтобы узнать текущее время, нужно просто добавить прошедшее время к времени на момент старта системы. Но как узнать человеческое время на момент старта системы?
RTC или часы реального времени
Первые персональные компьютеры (например, IBM PC или Apple II), не умели следить за тем, сколько прошло время после выключения, а спрашивали его на старте. Для решения этой задачи снова пригодился кварцевый генератор. Устройству неважно, из какого источника получать электрический ток, — это натолкнуло разработчиков на мысль, что достаточно подключить генератор к обычной литиевой батарейке, чтобы получать равномерные электрические колебания.
Если в эту схему добавить бинарный счетчик, который увеличивается на каждое колебание, то мы получим устройство, которое может фиксировать человеческое время. Оно так и называется — RTC (Real Time Clock), или часы реального времени. Частота колебаний на выходе из RTC обычно 32 768 Гц или 2^15 Гц, что удобно использовать в бинарных счетчиках.
Старый компьютер, о котором шла речь в начале, перепутал время именно из-за того, что батарейка в его часах реального времени села: после запуска он не смог считать время из RTC и выставил стандартное стартовое время — 1 января 1970 года.
NTP или протокол сетевого времени
RTC позволяет компьютеру отсчитывать миллисекунды, даже когда он выключен. Но стандартные RTC имеют погрешность 1,7-8,6 секунд в день — то есть за год они могут потерять целый час.
Иногда мы сталкиваемся с этой проблемой в реальной жизни: например, когда нам приходится вручную настраивать время на наручных кварцевых часах или на микроволновке, где тоже используют RTC. Но нам никогда не приходится делать этого на компьютере. Да и на старте компьютер больше не спрашивает текущее время. Современные компьютеры настраивают время через интернет — для этого используется NTP (протокол сетевого времени). Этот протокол даже учитывает время передачи данных между источником и компьютером и компенсирует его. В публичной сети погрешность составляет всего 10 мс.
Как протоколы работают вместе?
Когда наш компьютер выключен, RTC продолжает работу и отсчитывает время. Когда мы нажимаем кнопку запуска, операционная система забирает время из RTC и начинает отслеживать время самостоятельно, используя таймер процессора. Время от времени операционная система получает точное время по NTP и поправляет свой внутренний счетчик. Когда мы выключаем компьютер, за дело снова берется RTC.
Таймстампы и эпохи
Как уже говорилось, для отслеживания человеческого времени операционная система создает в памяти компьютера переменную jiffy, в которой хранит количество тиков с момента старта системы. Но как с помощью неё показывать человеку календарь?
Unix-таймстамп и epoch
В 70-е годы прошлого века эту проблему решили инженеры из Bell Labs при разработке операционной системы Unix (она заложила фундамент для появления современных Linux и MacOS). Они ввели в систему переменную, которая, начиная с заданной даты, увеличивается на каждый тик генератора — её называли epoch.
Под эту переменную отводилось целое число со знаком (signed integer) размером в 32 бита (то есть от −2 147 483 648 (-2^31) до 2 147 483 647 (2^31−1)). Подавляющее большинство генераторов на тот момент работали на частоте 60 Гц, то есть отсчитывали 60 тиков в секунду, поэтому в переменной хранилось 1/60 секунды и она могла представлять временной промежуток не более 829 дней.
В версии Unix от 1971 года отсчет начинался с 1971-01-01 00:00
. На следующий год с 1972-01-01 00:00
. Переводить время каждый год было довольно неудобно, поэтому в четвертой версии Unix в 1973 году за epoch была взята дата 1970-01-01 00:00
, а в переменной стали хранить не 1/60 секунды, а полную секунду. Позже этот принцип стал международным стандартом и используется по сей день.
Если на Вашем компьютере установлена операционная система семейства Linux или MacOS, Вы можете увидеть текущий Unix-timestamp, введя в терминале:
Windows тоже считает время, отталкиваясь от конкретной даты, но использует для этого не абстрактный 1970 год, а 1601 — первый год Григорианского календаря.
Проблема 2038 года
Поскольку время хранится в целочисленной 32-битовой переменной и представляет собой определенное количество секунд с определенного момента, то самое большое количество секунд, которое мы можем использовать — это 2 147 483 647 (2^31−1). Если прибавить это количество к epoch, то мы получим 19 января 2038 03
. Что произойдет с системой, когда наступит этот день и пройдет еще одна секунда?
Чтобы ответить на этот вопрос, рассмотрим особенность 32-битовых переменных:
2 147 483 647 в двоичной системе записывается как
Если это число увеличить на единицу, то оно превратится в
Первая цифра у таких чисел означает знак: + или -. А значит, что в десятичной системе оно будет равно −2 147 483 648, то есть число перейдет от самого большого положительного к самому маленькому отрицательному. И система начнет показывать дату, равную разности epoch и двух млрд секунд, то есть 13 декабря 1901 года. Но не стоит волноваться, проблема уже решена — большинство систем используют 64 битовые числа для хранения времени. Этого хватит, чтобы не столкнуться с проблемой до 15
4 декабря 292 277 026 596 года.
Проблема 2000 года или Y2K bug
«640 килобайт памяти должно хватить кому угодно», — Билл Гейтс.
На самом деле в истории уже была такая ситуация. В 60-70-е годы прошлого столетия, когда люди только начали писать программное обеспечение для компьютеров, платы памяти стоили дорого и большинство компьютеров обходились несколькими килобайтами. Ради экономии памяти программисты решили записывать даты в формате ДД.ММ.ГГ.
Какую проблему это порождало? Допустим, у нас есть человек по имени Боб, у которого дата рождения записана как 01.11.19. Он родился в 1919 году, и ему чуть больше 100 лет. Есть человек по имени Фред и он родился 02.11.19, но ему всего два года, потому что он родился в 2019 году. Сталкиваясь с таким форматом дат, человек может исходить из контекста, но компьютер на такое не способен.
Программисты, которые писали программы в 60-70-е годы даже не предполагали, что их код может дожить до 2000 года, поэтому использование двух цифр вместо четырех было нормальной оптимизацией. Но когда приближался 2000 год, а многие компании всё ещё использовали тот же формат дат, началась паника: никто не понимал, что произойдёт, когда 99 год сменится на 00. Тогда в мире ходили самого разного рода слухи:
например, что банкоматы в этот момент начнут плеваться деньгами, а самолёты начнут падать. Баг был вовремя замечен и проблему в большинстве систем удалось исправить вовремя.
Об особенностях работы с локальным временем и часовыми поясами
Почему не стоит самостоятельно писать код, описывающий время
Представим, что абстрактный программист пишет приложение, которое должно каждый день в 12 часов дня отправлять пользователю уведомление, что наступил полдень. Программист живет в Лондоне и оно будет использоваться только там. Писать код он начал зимой, поэтому разницы с UTC(Всемирным координированным временем) нет.
Приложение работало всю зиму, но в первый день апреля уведомление ушло на час позже полудня. Дело в том, что разработчик просто прикрутил таймер, который срабатывает раз в 24 часа, а в последний день марта в Великобритании перевели время на час вперед — на летнее. Разработчик быстро исправил ситуацию, заменив таймер на планировщик задач — последний срабатывает в соответствии с локальным временем сервера, который находится в квартире разработчика.
Как ни странно, приложение пользуется популярностью. Спустя некоторое время разработчик получает отзыв от разгневанного пользователя из Нью-Йорка: тот пишет, что уведомления приходят вовсе не в полдень, а в 7 утра. Нью-Йорк находится в часовом поясе ETC (Eastern Time Zone), то есть UTC -05
.
Для разных часовых поясов нельзя просто ставить таймер, отталкиваясь от времени на сервере. Разработчик добавил выбор часового пояса при регистрации и переписал планировщик так, чтобы он стартовал для каждого пользователя раньше или позже на несколько часов в зависимости от данных, которые человек ввёл при регистрации.
Кажется, теперь все работает хорошо — но в ноябре разработчику начали писать пользователи из Москвы и жаловаться на уведомления, которые приходят в 11 утра. Дело в том, что в Великобритании время переводится с зимнего на летнее, а в России — нет. К тому моменту разработчик начал нервничать и решил просто записать в коде, что если пользователь из России, то делать переход во времени для него не нужно. Приложение тем временем только набирало популярность.
В марте, разработчику пишут из Сиднея — им уведомление приходит в два часа дня. Все потому, что в южном полушарии в марте происходит переход не на летнее время, а на зимнее. Разработчику не оставалось ничего иного, как дополнить свой код информацией о том, что в южном полушарии надо переводить время наоборот. Приложение уже стало популярным во всём мире и на разработчика посыпались претензии со всех уголков света. Например, пользователи из Палестины писали жалобы каждый раз, как их правительство переводило часы с летнего времени на зимнее, а случается это каждый раз в разное время. А ещё дело было в 2011 году, и 29 декабря пользователи из Самоа дружно пересекли линию перемены даты, совершив скачок в будущее и очутившись 31 декабря 2011 года. Еще через год разработчику позвонили из Международной службы вращения Земли и сообщили, что ему надо поправить приложение, потому что в этом году будет на
одну секунду больше, чем в прошлом из-за добавления дополнительной секунды. Записывать
все это в код — совершенно безумная идея. Поэтому разработчик закрыл проект и больше никогда не писал код.
Пара советов, как работать с локальным временем
Что бы мог сделать разработчик, чтобы избежать такого печального финала? Все просто — достаточно было учесть опыт тех, кто уже решал эту проблему и выкладывал решение в открытый доступ. В каждом языке программирования есть как минимум одна библиотека, которая умеет работать с человеческим временем и часовыми поясами. Более того, существует база данных, в которой хранится информация о локальном времени, в том числе в исторической перспективе.
Что бы мог сделать разработчик:
- Переложить ответственность за подсчет времени на библиотеки и базы данных;
- Хранить не разницу во времени с локальным временем пользователя, а его часовой
пояс в соответствии с классификацией IANA,
то есть не +3, а "Europe/Moscow";
- Настроить планировщик так, чтобы он выбирал задачи из списка в соответствии с
часовым поясом.
Пара советов, как НЕ работать с локальным временем
Существует много сценариев, при которых крайне важна последовательность событий, на которую не могут повлиять ни изменения часовых поясов, ни переход на летнее время.
Предположим, нужно написать сервис, обслуживающий банковские транзакции. Его функционал крайне простой: один пользователь отправляет другому деньги банковским переводом, любой из них может получить информацию обо всех переводах. В данном случае нужно хранить информацию о том, кто, кому, когда и сколько денег перевёл. При этом показывать эту информацию нужно в локальном времени пользователя при
условии, что транзакции в любом часовом поясе идут одна за другой.
Учтём опыт разработчика из примера выше и не будем описывать время самостоятельно, а используем библиотеку. Тут мы столкнемся с рядом вопросов:
- В каком часовом поясе хранить транзакции, если клиенты живут в разных странах,
а база данных находится в третьей стране;
- Как учитывать переход на летнее и зимнее время;
- Как учитывать дополнительную секунду;
- Что делать, если пользователь переехал из одной страны в другую.
И лучшим решением будет не делать ничего. Не нужно учитывать временные зоны клиентов, не нужно учитывать переходы на летнее и зимнее время, не нужно учитывать переезды пользователей. Когда нужно абсолютное время, достаточно использовать UTC.
Время в UTC поддерживается почти всеми библиотеками и почти всеми база и данных. Используя UTC, мы можем получать транзакцию в локальном времени пользователя, переводить это время в UTC и сохранять у себя. И транзакции всегда будут идти одна за другой.
Правда, существует проблема: UTC учитывает дополнительные секунды, а значит, в какой-то момент мы можем получить такие транзакции:
- Маша Феде перевела 100 рублей 2016-12-31 в 23:59
- Федя Юре перевёл 100 рублей 2016-12-31 в 23:59
- Юра отправил Майклу 1$ 2016-12-31 в 23:59
- Майкл отправил Теду 99$ 2017-01-01 в 00:00
Скорее всего программа, работающая с UTC, не сможет сказать, что произошло раньше — вторая транзакция, третья или четвертая. Google, которой очень важна абсолютная последовательность событий во времени, очень интересно решила эту проблему. В своих серверах точного времени они добавляют дополнительную секунду не за раз, а равномерно распределяют её в течение дня в начале года. Это называется leap smear. Таким образом, раз в год, когда астрономы объявляют дополнительную секунду, в течение суток сервера точного времени Google немного отстают от UTC, чтобы набрать дополнительную секунду.
Выводы
- В любом современном ПК есть процессор, который умеет производить колебания с
равными интервалами.
- Зная период этих колебаний, операционная система компьютера может считать
человеческое время.
- С помощью протокола NTP операционная система может обращаться к серверам
точного времени для скорректировать своих часов.
- Когда компьютер выключен, время считает специальное устройство — RTC, которое
питается от батареи.
- Epoch — это некая дата, от которой принято отсчитывать время вперед или назад.
Разные операционные системы и языки программирования могут выбирать свой собственный
epoch.
- Unix-timestamp — это количество секунд, прошедших с unix-epoch, то есть
с 00 1 января 1970 года.
- Никогда не пишите самостоятельно код, который описывает человеческое время.
- Если вам нужна точная последовательность действий во времени, не используйте
часовые зоны. Храните время в UTC и по требованию конвертируйте его.
Дополнительные материалы
<!DOCTYPE html>
<html class="h-100" data-bs-theme="light" data-mantine-color-scheme="light" lang="ru" prefix="og: https://ogp.me/ns#">
<head>
<meta content="width=device-width, initial-scale=1.0" name="viewport">
<meta content="IE=Edge" http-equiv="X-UA-Compatible">
<link crossorigin="true" href="https://cdn.hexlet.io" rel="preconnect">
<link href="https://mc.yandex.ru" rel="preconnect">
<meta content="aa2vrdtq64dub8knuf83lwywit311w" name="facebook-domain-verification">
<link href="/favicon.ico" rel="icon" sizes="any">
<link href="/favicon.svg" rel="icon" type="image/svg+xml">
<link href="/apple-touch-icon.png" rel="apple-touch-icon">
<link href="/manifest.webmanifest" rel="manifest">
<script>
//<![CDATA[
window.gon={};gon.ym_counter="25559621";gon.is_bot=true;gon.applications={};gon.current_user={"id":null,"last_viewed_notification_id":null,"email":null,"state":null,"first_name":"","last_name":"","created_at":"2026-02-26 16:43:24 UTC","current_program":null,"current_team":null,"full_name":"","guest":true,"can_use_paid_features":false,"is_hexlet_employee":false,"sanitized_phone_number":"","can_subscribe":true,"can_renew_education":false};gon.token="0AI3JBdS1HSlkJb53iJkdpdyx3_FC6pbUEGf7Z8XlJY_0_wT5Sx5FBPTsmHSLZQBV3vq1c08VPntoQW5zRBz-A";gon.locale="ru";gon.language="ru";gon.theme="light";gon.rails_env="production";gon.mobile=false;gon.google={"analytics_key":"UA-1360700-51","optimize_key":"GTM-5QDVFPF"};gon.captcha={"google_v3_site_key":"6LenGbgZAAAAAM7HbrDbn5JlizCSzPcS767c9vaY","yandex_site_key":"ysc1_Vyob5ZPPUdPBsu0ykt8bVFdzsfpoVjQChLGl2b4g19647a89","verification_failed":null};gon.social_signin=false;gon.typoreporter_google_form_id="1FAIpQLSeibfGq-KvWQ2Fyru-zkFFRVTLBuzXAHAoEyN1p49FtDmNoNA";
//]]>
</script>
<meta charset="utf-8">
<title>Что такое время | Hexlet Guides</title>
<meta name="description" content="Как современный компьютер хранит информацию, как он отсчитывает секунды, почему почти никогда не ошибается и как с этим временем работать">
<link rel="canonical" href="https://ru.hexlet.io/blog/posts/time">
<meta property="og:title" content="Что такое время | Hexlet Guides">
<meta property="og:description" content="Как современный компьютер хранит информацию, как он отсчитывает секунды, почему почти никогда не ошибается и как с этим временем работать">
<meta property="og:image" content="https://ru.hexlet.io/rails/active_storage/representations/proxy/eyJfcmFpbHMiOnsiZGF0YSI6NDIzLCJwdXIiOiJibG9iX2lkIn19--5c6c5168a4dcfa6bb0bed8ae50c0877c3db687e3/eyJfcmFpbHMiOnsiZGF0YSI6eyJmb3JtYXQiOiJ3ZWJwIiwicmVzaXplX2FuZF9wYWQiOlsxNDU2LDcyOF0sInNhdmVyIjp7InF1YWxpdHkiOjg1fX0sInB1ciI6InZhcmlhdGlvbiJ9fQ==--b2ecf6121248a24883cda19469732b47b57f4e80/time.png">
<meta name="csrf-param" content="authenticity_token" />
<meta name="csrf-token" content="gyzseeqR9S4qFXdOYbsJxHQgoUGlwdo1_fqkqniWcWls_SdOGO9YTpxWU9ZttPmztCmM6632JJdAGj7-KpGWBw" />
<script src="/vite/assets/inertia-INZxX8jp.js" crossorigin="anonymous" type="module"></script><link rel="modulepreload" href="/vite/assets/chunk-DsPFFUou.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/preload-helper-BJ4cLWpC.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/init-nkZBEvfU.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/ahoy-DrlRQ-1D.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/analytics-6pOtQ3OW.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/ErrorFallbackBlock-naDSYSy9.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/Surface-DL2bpZA-.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/gon-D3e4yh1x.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/mantine-CGMYrt2Y.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/utils-DRqSHbQE.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/routes-CCH8ilKF.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/extends-C-EagtpE.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/inheritsLoose-BBd-DCVI.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/objectWithoutPropertiesLoose-DRHXDhjp.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/index.esm-DAqKOkZ0.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/Button-CGPUux8l.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/CloseButton-D1euiPao.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/Group-BX48WcuU.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/Loader-BQEY8g6v.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/Modal-Cy3HByv7.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/OptionalPortal-1Hza5P2w.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/Stack-CtjJzfw4.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/Textarea-Ck64llAy.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/Box-B5-OOzBf.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/DirectionProvider-Dc9zdUke.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/events-DJQOhap0.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/use-reduced-motion-D2owz4wa.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/use-disclosure-zKtK5W1r.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/use-hotkeys-Cnc_Rwkb.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/random-id-DOQyszCZ.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/notifications.store-C-3AFSMn.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/exports-C_MrNx_T.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/axios-BEvgo0ym.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/dayjs.min-BkKovM-s.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/i18next-BlSq9s7B.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/client-U9M77rxp.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/react-dom-DaLxUz_h.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/useTranslation-Bx1Cdrkz.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/compiler-runtime-6XxiPFnt.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/jsx-runtime-CwjcCKJi.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/react-CkL4ZRHB.js" as="script" crossorigin="anonymous">
<link rel="stylesheet" href="/vite/assets/application-BqhCP46M.js" />
<script src="/vite/assets/application-Df9RExpe.js" crossorigin="anonymous" type="module"></script><link rel="modulepreload" href="/vite/assets/chunk-DsPFFUou.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/autocomplete-VMNbxKGl.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/routes-CCH8ilKF.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/createPopper-C3aM9r1M.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/js.cookie-D1-O8zkX.js" as="script" crossorigin="anonymous"><link rel="stylesheet" href="/vite/assets/application-C8HjmMaq.css" media="screen" />
<script>
window.ym = function(){(ym.a=ym.a||[]).push(arguments)};
window.addEventListener('load', function() {
setTimeout(function() {
ym.l = 1*new Date();
ym(window.gon.ym_counter, "init", {
clickmap: true,
trackLinks: true,
accurateTrackBounce: true,
webvisor: true
});
// Загружаем скрипт
var k = document.createElement('script');
k.async = 1;
k.src = 'https://mc.yandex.ru/metrika/tag.js';
document.head.appendChild(k);
ym(window.gon.ym_counter, 'getClientID', function(clientID) {
window.ymClientId = clientID;
});
}, 1500);
});
</script>
<!-- Google Tag Manager - deferred -->
<script>
// dataLayer stub сразу — пуши работают до загрузки скрипта
window.dataLayer = window.dataLayer || [];
// Сам скрипт — отложенно после load
window.addEventListener('load', function() {
setTimeout(function() {
dataLayer.push({'gtm.start': new Date().getTime(), event: 'gtm.js'});
var j = document.createElement('script');
j.async = true;
j.src = 'https://www.googletagmanager.com/gtm.js?id=GTM-WK88TH';
document.head.appendChild(j);
}, 1500);
});
</script>
<!-- End Google Tag Manager -->
</head>
<body>
<noscript>
<div>
<img alt="" src="https://mc.yandex.ru/watch/25559621" style="position:absolute; left:-9999px;">
</div>
</noscript>
<header class="sticky-top bg-body">
<nav class="navbar navbar-expand-lg">
<div class="container-xxl">
<a class="navbar-brand" href="/"><img alt="Логотип Хекслета" height="24" src="https://ru.hexlet.io/vite/assets/logo_ru_light-BpiEA1LT.svg" width="96">
</a><button aria-controls="collapsable" aria-expanded="false" aria-label="Меню" class="navbar-toggler border-0 mb-0 mt-1" data-bs-target="#collapsable" data-bs-toggle="collapse">
<span class="navbar-toggler-icon"></span>
</button>
<div class="collapse navbar-collapse" id="collapsable">
<ul class="navbar-nav mb-lg-0 mt-lg-1">
<li class="nav-item dropdown">
<button aria-haspopup class="btn nav-link" data-bs-toggle="dropdown" type="button">
Все курсы
<span class="bi bi-chevron-down align-middle ms-1"></span>
</button>
<ul class="dropdown-menu">
<li>
<a class="dropdown-item d-flex py-2" href="/courses"><div class="fw-bold me-auto">Все что есть</div>
<div class="text-muted">117</div>
</a></li>
<li>
<hr class="dropdown-divider">
</li>
<li class="dropdown-item">
<b>Популярные категории</b>
</li>
<li>
<a class="dropdown-item py-2" href="/courses_devops">Курсы по DevOps
</a></li>
<li>
<a class="dropdown-item py-2" href="/courses_data_analytics">Курсы по аналитике данных
</a></li>
<li>
<a class="dropdown-item py-2" href="/courses_programming">Курсы по программированию
</a></li>
<li>
<a class="dropdown-item py-2" href="/courses_testing">Курсы по тестированию
</a></li>
<li>
<hr class="dropdown-divider">
</li>
<li class="dropdown-item">
<b>Популярные курсы</b>
</li>
<li>
<a class="dropdown-item py-2" href="/programs/devops-engineer-from-scratch">DevOps-инженер с нуля
</a></li>
<li>
<a class="dropdown-item py-2" href="/programs/go">Go-разработчик
</a></li>
<li>
<a class="dropdown-item py-2" href="/programs/java">Java-разработчик
</a></li>
<li>
<a class="dropdown-item py-2" href="/programs/python">Python-разработчик
</a></li>
<li>
<a class="dropdown-item py-2" href="/programs/qa-auto-engineer-java">Автоматизатор тестирования на Java
</a></li>
<li>
<a class="dropdown-item py-2" href="/programs/data-analytics">Аналитик данных
</a></li>
<li>
<a class="dropdown-item py-2" href="/programs/frontend">Фронтенд-разработчик
</a></li>
</ul>
</li>
<li class="nav-item dropdown">
<button aria-haspopup class="btn nav-link" data-bs-toggle="dropdown" type="button">
О Хекслете
<span class="bi bi-chevron-down align-middle"></span>
</button>
<ul class="dropdown-menu bg-body">
<li>
<a class="dropdown-item py-2" href="/pages/about">О нас
</a></li>
<li>
<a class="dropdown-item py-2" href="/blog">Блог
</a></li>
<li>
<span class="dropdown-item py-2 external-link" data-href="https://special.hexlet.io/hse-research" role="button">Результаты (Исследование)
</span></li>
<li>
<span class="dropdown-item py-2 external-link" data-href="https://career.hexlet.io" role="button">Хекслет Карьера
</span></li>
<li>
<a class="dropdown-item py-2" href="/testimonials">Отзывы студентов
</a></li>
<li>
<span class="dropdown-item py-2 external-link" data-href="https://t.me/hexlet_help_bot" role="button">Поддержка (В ТГ)
</span></li>
<li>
<span class="dropdown-item py-2 external-link" data-href="https://special.hexlet.io/referal-program/?promo_creative=priglasite-druzei&promo_name=referal-program&promo_position=promo_position&promo_start=010724&promo_type=link" role="button">Реферальная программа
</span></li>
<li>
<span class="dropdown-item py-2 external-link" data-href="https://special.hexlet.io/certificate" role="button">Подарочные сертификаты
</span></li>
<li>
<span class="dropdown-item py-2 external-link" data-href="https://hh.ru/employer/4307094" role="button">Вакансии
</span></li>
<li>
<span class="dropdown-item d-flex external-link" rel="noopener noreferrer nofollow" data-href="https://b2b.hexlet.io" data-target="_blank" role="button">Компаниям
</span></li>
<li>
<span class="dropdown-item d-flex external-link" rel="noopener noreferrer nofollow" data-href="https://hexly.ru/" data-target="_blank" role="button">Колледж
</span></li>
<li>
<span class="dropdown-item d-flex external-link" rel="noopener noreferrer nofollow" data-href="https://hexlyschool.ru/" data-target="_blank" role="button">Частная школа
</span></li>
</ul>
</li>
<li><a class="nav-link" href="/subscription/new">Подписка</a></li>
</ul>
<ul class="navbar-nav flex-lg-row align-items-lg-center gap-2 ms-auto">
<li>
<a class="nav-link" aria-label="Переключить тему" href="/theme/switch?new_theme=dark"><span aria-hidden="true" class="bi bi-moon"></span>
</a></li>
<li>
<span data-target="_self" class="nav-link external-link" data-href="/u/new" role="button"><span>Регистрация</span>
</span></li>
<li>
<span data-target="_self" class="nav-link external-link" data-href="https://ru.hexlet.io/session/new" role="button"><span>Вход</span>
</span></li>
</ul>
</div>
</div>
</nav>
</header>
<div class="x-container-xxxl">
</div>
<main class="mb-6 min-vh-100 h-100">
<link rel="preload" as="image" href="https://hexlet.io/rails/active_storage/representations/proxy/eyJfcmFpbHMiOnsiZGF0YSI6NDIzLCJwdXIiOiJibG9iX2lkIn19--5c6c5168a4dcfa6bb0bed8ae50c0877c3db687e3/eyJfcmFpbHMiOnsiZGF0YSI6eyJmb3JtYXQiOiJ3ZWJwIiwicmVzaXplX2FuZF9wYWQiOlsxNDU2LDcyOF0sInNhdmVyIjp7InF1YWxpdHkiOjg1fX0sInB1ciI6InZhcmlhdGlvbiJ9fQ==--b2ecf6121248a24883cda19469732b47b57f4e80/time.png"/><link rel="preload" as="image" href="https://hexlet.io/rails/active_storage/representations/proxy/eyJfcmFpbHMiOnsiZGF0YSI6NDAyOCwicHVyIjoiYmxvYl9pZCJ9fQ==--ae9eed98663dd1201759d042a5ba7ca790866156/eyJfcmFpbHMiOnsiZGF0YSI6eyJmb3JtYXQiOiJ3ZWJwIiwicmVzaXplX3RvX2xpbWl0IjpbNDAwLDQwMF0sInNhdmVyIjp7InF1YWxpdHkiOjg1fX0sInB1ciI6InZhcmlhdGlvbiJ9fQ==--5b6f46dacd1af664f27558553a58076185091823/Programming-bro.png"/><link rel="preload" as="image" href="https://hexlet.io/rails/active_storage/representations/proxy/eyJfcmFpbHMiOnsiZGF0YSI6MzY2MSwicHVyIjoiYmxvYl9pZCJ9fQ==--e9c2b6bde361adaac625a7f47d8b9671c17f3ddb/eyJfcmFpbHMiOnsiZGF0YSI6eyJmb3JtYXQiOiJ3ZWJwIiwicmVzaXplX3RvX2xpbWl0IjpbNDAwLDQwMF0sInNhdmVyIjp7InF1YWxpdHkiOjg1fX0sInB1ciI6InZhcmlhdGlvbiJ9fQ==--5b6f46dacd1af664f27558553a58076185091823/Mathematics-bro.png"/><link rel="preload" as="image" href="https://hexlet.io/rails/active_storage/representations/proxy/eyJfcmFpbHMiOnsiZGF0YSI6Mzc1OCwicHVyIjoiYmxvYl9pZCJ9fQ==--023ea18f500b1c4c91617fa96bbc52df8395da39/eyJfcmFpbHMiOnsiZGF0YSI6eyJmb3JtYXQiOiJ3ZWJwIiwicmVzaXplX3RvX2xpbWl0IjpbNDAwLDQwMF0sInNhdmVyIjp7InF1YWxpdHkiOjg1fX0sInB1ciI6InZhcmlhdGlvbiJ9fQ==--5b6f46dacd1af664f27558553a58076185091823/Software%20engineer-bro.png"/><link rel="preload" as="image" href="https://hexlet.io/rails/active_storage/representations/proxy/eyJfcmFpbHMiOnsiZGF0YSI6NDAxNiwicHVyIjoiYmxvYl9pZCJ9fQ==--eb66b9b5e26fafa32844ce0f4522c3ed84544040/eyJfcmFpbHMiOnsiZGF0YSI6eyJmb3JtYXQiOiJ3ZWJwIiwicmVzaXplX3RvX2xpbWl0IjpbNDAwLDQwMF0sInNhdmVyIjp7InF1YWxpdHkiOjg1fX0sInB1ciI6InZhcmlhdGlvbiJ9fQ==--5b6f46dacd1af664f27558553a58076185091823/Hand%20coding-rafiki.png"/><link rel="preload" as="image" href="https://hexlet.io/rails/active_storage/representations/proxy/eyJfcmFpbHMiOnsiZGF0YSI6MzY0NywicHVyIjoiYmxvYl9pZCJ9fQ==--fd23784224c1dfaefc6e1cd556a43d164e3c412f/eyJfcmFpbHMiOnsiZGF0YSI6eyJmb3JtYXQiOiJ3ZWJwIiwicmVzaXplX3RvX2xpbWl0IjpbNDAwLDQwMF0sInNhdmVyIjp7InF1YWxpdHkiOjg1fX0sInB1ciI6InZhcmlhdGlvbiJ9fQ==--5b6f46dacd1af664f27558553a58076185091823/Mathematics-amico.png"/><link rel="preload" as="image" href="/vite/assets/development-BVihs_d5.png"/><div id="app" data-page="{"component":"web/blog/posts/show","props":{"errors":{},"locale":"ru","language":"ru","httpsHost":"https://ru.hexlet.io","host":"ru.hexlet.io","colorScheme":"light","auth":{"user":{"id":null,"last_viewed_notification_id":null,"email":null,"state":null,"first_name":"","last_name":"","created_at":"2026-02-26T16:43:24.175Z","current_program":null,"current_team":null,"full_name":"","guest":true,"can_use_paid_features":false,"is_hexlet_employee":false,"sanitized_phone_number":"","can_subscribe":true,"can_renew_education":false}},"cloudflareTurnstileSiteKey":"0x4AAAAAAA15KmeFXzd2H0Xo","vkIdClientId":"51586979","yandexIdClientId":"88d071f1d3384eb4bd1deb37910235c7","formAuthToken":"IkZm2LNVV8en4qrQ8EddoEpdc2Vn4hoHKeNRpl3myzzNl63vQSv6pxGhjkj8SK3XilRez2_V5KWUA8vyD-EsUg","post":{"model_name":"BlogPost","category":{"id":4,"name":"Код","slug":"code","state":"published","created_at":"2016-08-23T13:33:44.258Z"},"creator":{"public_name":"Валерия Белякова","id":431834,"is_tutor":false},"tags":[{"id":1119,"slug":"bez-steka","name":"Без стека"}],"id":2781,"title":"Что такое время","slug":"time","state":"published","summary":"Как современный компьютер хранит информацию, как он отсчитывает секунды, почему почти никогда не ошибается и как с этим временем работать","votes_count":0,"created_at":"2024-11-18T16:04:32.533Z","published_at":"2024-12-18T13:54:19.292Z","body":"## Как компьютер определяет человеческое время?\r\n\r\nПредставим ситуацию: Вы нашли в кладовой старый компьютер и решили попробовать его включить. Компьютер запускается, хотя стоял без дела 10 лет, но есть проблема: система уверена, что сегодня 1 января 1970 года. Эта ситуация поможет разобраться, как компьютеры хранят текущее время: ведь у них нет внутри пружин или маятников, как в часах, которые могли бы отсчитывать секунды. Каким образом они показывают верное время, если отключить их на несколько часов, и почему тот же принцип не работает, если компьютер простоял в кладовке 10 лет?\r\n\r\n::programs\r\n\r\n### Маятник как первый секундомер\r\n\r\nПрежде, чем ответить на эти вопросы, разберемся, как измеряется точное время. Это не такая простая задача, если под рукой нет часов. Мы можем отсчитывать дни, месяцы и годы или приблизительно разбить день на часы. Но найти ориентир, который подскажет количество прошедших секунд, гораздо сложнее.\r\n\r\nПринято считать, что первым эту задачу в XVII веке решил итальянский ученый [Галилео Галилей] (https://ru.wikipedia.org/wiki/%D0%93%D0%B0%D0%BB%D0%B8%D0%BB%D0%B5%D0%B9,_%D0%93%D0%B0%D0%BB%D0%B8%D0%BB%D0%B5%D0%BE). Галилео нужно было точно знать, сколько длился тот или иной эксперимент. Его биографы\r\nутверждают, что сначала он считал количество своих сердцебиений, но однажды наблюдал за раскачивающимся канделябром в кафедральном соборе Пизы и заметил, что каждое колебание канделябра занимает одинаковое количество сердцебиений. Это число не меняется при затухании амплитуды движений. Так он понял, что\r\nколебания маятника помогают точно измерять время.\r\n\r\n### Можно ли использовать колебания для подсчета времени в компьютере?\r\n\r\nКогда в 70-е-80-е годы инженеры впервые столкнулись с задачей отображения человеческого времени в компьютере, они, как ни странно, обратились к трудам итальянского ученого, но взглянули на них под другим углом. Маятник — не единственный возможный источник колебаний. У любого современного персонального компьютера есть «сердце» — центральное процессорное устройство (ЦПУ), или просто процессор. Он устроен очень сложно и выполняет множество функций, но в данном случае интересно одно из составляющих его устройств. Когда через него проходит ток, оно испускает равномерные электрические колебания с одинаковой [частотой](https://en.wikipedia.org/wiki/Clock_rate). Частота этих колебаний измеряется в герцах (Гц) и определяет количество операций, которое процессор может выполнить за одну секунду. К сожалению, если мы будем использовать единственную доступную нам операцию в секунду для увеличения счетчика колебаний, то наш процессор будет совершенно непригоден для чего-то ещё: ему будет некогда исполнять другие команды. Конечно, современные процессоры не такие медленные, их частота измеряется в гигагерцах (ГГц), то есть в 1 000 000 000 операций в секунду. Но даже при такой мощности для измерения времени пришлось бы считать каждое колебание процессора. Для решения проблемы был придуман кварцевый генератор. Он представляет собой тончайшую кремниевую пластину, которая под воздействием электрического тока равномерно расширяется и сжимается, [генерируя слабый электрический заряд на поверхности](https://ru.wikipedia.org/wiki/%D0%9F%D1%8C%D0%B5%D0%B7%D0%BE%D1%8D%D0%BB%D0%B5%D0%BA%D1%82%D1%80%D0%B8%D1%87%D0%B5%D1%81%D0%BA%D0%B8%D0%B9_%D1%8D%D1%84%D1%84%D0%B5%D0%BA%D1%82). Таким образом, механические колебания пластины сопровождаются синхронными, равномерными колебаниями электрического заряда.\r\n\r\n\r\n\r\nПолученные равномерные колебания принято называть таймером — они позволяют процессору синхронизировать производимые им операции во времени. Таким образом, у компьютера есть своё собственное понимание времени.\r\n\r\n::posts\r\n\r\n### От компьютерного времени к человеческому\r\n\r\nПолучается, что внутри компьютера есть таймер, но он не измеряет человеческое время — оно для него довольно бесполезно. Как правило, эту задачу на себя берут операционные системы (ОС). Поскольку ОС знает, с какой частотой работает кварцевый генератор, она может измерить время, которое проходит между срабатываниями таймера (срабатывание таймера называют тик (tick) или джиффи (jiffy)). Если генератор\r\nработает на частоте 100 Гц, то период между тиками равен 1/100 секунды или 10 миллисекундам. Операционная система создает в памяти переменную, которую обычно называют jiffies, и увеличивает её на единицу каждый раз, когда процессор дает сигнал о новом тике.\r\n\r\nСоответственно, чтобы узнать, как долго включен компьютер, системе достаточно умножить размер периода между тиками на количество этих самых тиков. А чтобы узнать текущее время, нужно просто добавить прошедшее время к времени на момент старта системы. Но как узнать человеческое время на момент старта системы?\r\n\r\n### RTC или часы реального времени\r\n\r\nПервые персональные компьютеры (например, [IBM PC](https://en.wikipedia.org/wiki/IBM_Personal_Computer#Design_process) или [Apple II](https://en.wikipedia.org/wiki/Apple_II)), не умели следить за тем, сколько прошло время после выключения, а [спрашивали его на старте](https://www.youtube.com/watch?v=X3aqJQPQKhs). Для решения этой задачи снова пригодился кварцевый генератор. Устройству неважно, из какого источника получать электрический ток, — это натолкнуло разработчиков на мысль, что достаточно подключить генератор к обычной литиевой батарейке, чтобы получать равномерные электрические колебания.\r\n\r\nЕсли в эту схему добавить бинарный счетчик, который увеличивается на каждое колебание, то мы получим устройство, которое может фиксировать человеческое время. Оно так и называется — RTC (Real Time Clock), или часы реального времени. Частота колебаний на выходе из RTC обычно 32 768 Гц или 2^15 Гц, что удобно использовать в бинарных счетчиках.\r\n\r\n\r\n\r\nСтарый компьютер, о котором шла речь в начале, перепутал время именно из-за того, что батарейка в его часах реального времени села: после запуска он не смог считать время из RTC и выставил стандартное стартовое время — 1 января 1970 года.\r\n\r\n\r\n\r\n### NTP или протокол сетевого времени\r\n\r\nRTC позволяет компьютеру отсчитывать миллисекунды, даже когда он выключен. Но стандартные RTC имеют погрешность 1,7-8,6 секунд в день — то есть за год они могут потерять целый час.\r\n\r\nИногда мы сталкиваемся с этой проблемой в реальной жизни: например, когда нам приходится вручную настраивать время на наручных кварцевых часах или на микроволновке, где тоже используют RTC. Но нам никогда не приходится делать этого на компьютере. Да и на старте компьютер больше не спрашивает текущее время. Современные компьютеры настраивают время через интернет — для этого используется NTP (протокол сетевого времени). Этот протокол даже учитывает время передачи данных между источником и компьютером и компенсирует его. В публичной сети погрешность составляет всего 10 мс.\r\n\r\n### Как протоколы работают вместе?\r\n\r\n\r\n\r\nКогда наш компьютер выключен, RTC продолжает работу и отсчитывает время. Когда мы нажимаем кнопку запуска, операционная система забирает время из RTC и начинает отслеживать время самостоятельно, используя таймер процессора. Время от времени операционная система получает точное время по NTP и поправляет свой внутренний счетчик. Когда мы выключаем компьютер, за дело снова берется RTC.\r\n\r\n## Таймстампы и эпохи\r\n\r\nКак уже говорилось, для отслеживания человеческого времени операционная система создает в памяти компьютера переменную jiffy, в которой хранит количество тиков с момента старта системы. Но как с помощью неё показывать человеку календарь?\r\n\r\n### Unix-таймстамп и epoch\r\n\r\nВ 70-е годы прошлого века эту проблему решили инженеры из Bell Labs при разработке операционной системы Unix (она заложила фундамент для появления современных Linux и MacOS). Они ввели в систему переменную, которая, начиная с заданной даты, увеличивается на каждый тик генератора — её называли epoch.\r\n\r\nПод эту переменную отводилось целое число со знаком (signed integer) размером в 32 бита (то есть от −2 147 483 648 (-2^31) до 2 147 483 647 (2^31−1)). Подавляющее большинство генераторов на тот момент работали на частоте 60 Гц, то есть отсчитывали 60 тиков в секунду, поэтому в переменной хранилось 1/60 секунды и она могла представлять временной промежуток не более 829 дней.\r\n\r\nВ [версии Unix от 1971 года](http://man.cat-v.org/unix-1st/2/sys-time) отсчет начинался с 1971-01-01 00:00:00. [На следующий год](https://minnie.tuhs.org/cgi-bin/utree.pl?file=V3/man/man2/time.2) с 1972-01-01 00:00:00. Переводить время каждый год было довольно неудобно, поэтому в [четвертой версии Unix](https://minnie.tuhs.org/cgi-bin/utree.pl?file=V4/man/man2/time.2) в 1973 году за epoch была взята дата 1970-01-01 00:00:00, а в переменной стали хранить не 1/60 секунды, а полную секунду. Позже этот принцип стал международным стандартом и используется по сей день.\r\n\r\nЕсли на Вашем компьютере установлена операционная система семейства Linux или MacOS, Вы можете увидеть текущий Unix-timestamp, введя в терминале:\r\n\r\n```shell\r\ndate +\"%s\"\r\n```\r\n\r\nWindows тоже считает время, отталкиваясь от конкретной даты, но использует для этого не абстрактный 1970 год, а 1601 — первый год Григорианского календаря.\r\n\r\n\r\n\r\n### Проблема 2038 года\r\n\r\nПоскольку время хранится в целочисленной 32-битовой переменной и представляет собой определенное количество секунд с определенного момента, то самое большое количество секунд, которое мы можем использовать — это 2 147 483 647 (2^31−1). Если прибавить это количество к epoch, то мы получим 19 января 2038 03:14. Что произойдет с системой, когда наступит этот день и пройдет еще одна секунда?\r\n\r\nЧтобы ответить на этот вопрос, рассмотрим особенность 32-битовых переменных: \r\n\r\n2 147 483 647 в двоичной системе записывается как\r\n\r\n```js\r\n01111111111111111111111111111111 // (1 ноль и 31 единица)\r\n```\r\n\r\nЕсли это число увеличить на единицу, то оно превратится в\r\n\r\n```js\r\n10000000000000000000000000000000 // (1 единица и 31 ноль)\r\n```\r\n\r\nПервая цифра у таких чисел означает знак: `+` или `-`. А значит, что в десятичной системе оно будет равно −2 147 483 648, то есть число перейдет от самого большого положительного к самому маленькому отрицательному. И система начнет показывать дату, равную разности epoch и двух млрд секунд, то есть 13 декабря 1901 года. Но не стоит волноваться, проблема уже решена — большинство систем используют 64 битовые числа для хранения времени. Этого хватит, чтобы не столкнуться с проблемой до 15:40 4 декабря 292 277 026 596 года.\r\n\r\n### Проблема 2000 года или Y2K bug\r\n\r\n> «640 килобайт памяти должно хватить кому угодно», — Билл Гейтс.\r\n\r\nНа самом деле в истории уже была такая ситуация. В 60-70-е годы прошлого столетия, когда люди только начали писать программное обеспечение для компьютеров, платы памяти стоили дорого и большинство компьютеров обходились несколькими килобайтами. Ради экономии памяти программисты решили записывать даты в формате ДД.ММ.ГГ.\r\n\r\nКакую проблему это порождало? Допустим, у нас есть человек по имени Боб, у которого дата рождения записана как 01.11.19. Он родился в 1919 году, и ему чуть больше 100 лет. Есть человек по имени Фред и он родился 02.11.19, но ему всего два года, потому что он родился в 2019 году. Сталкиваясь с таким форматом дат, человек может исходить из контекста, но компьютер на такое не способен.\r\n\r\nПрограммисты, которые писали программы в 60-70-е годы даже не предполагали, что их код может дожить до 2000 года, поэтому использование двух цифр вместо четырех было нормальной оптимизацией. Но когда приближался 2000 год, а многие компании всё ещё использовали тот же формат дат, началась паника: никто не понимал, что произойдёт, когда 99 год сменится на 00. Тогда в мире ходили самого разного рода слухи:\r\nнапример, что банкоматы в этот момент начнут плеваться деньгами, а самолёты начнут падать. Баг был вовремя замечен и проблему в большинстве систем удалось исправить вовремя.\r\n\r\n## Об особенностях работы с локальным временем и часовыми поясами\r\n\r\n### Почему не стоит самостоятельно писать код, описывающий время\r\n\r\nПредставим, что абстрактный программист пишет приложение, которое должно каждый день в 12 часов дня отправлять пользователю уведомление, что наступил полдень. Программист живет в Лондоне и оно будет использоваться только там. Писать код он начал зимой, поэтому разницы с [UTC](https://ru.wikipedia.org/wiki/%D0%92%D1%81%D0%B5%D0%BC%D0%B8%D1%80%D0%BD%D0%BE%D0%B5_%D0%BA%D0%BE%D0%BE%D1%80%D0%B4%D0%B8%D0%BD%D0%B8%D1%80%D0%BE%D0%B2%D0%B0%D0%BD%D0%BD%D0%BE%D0%B5_%D0%B2%D1%80%D0%B5%D0%BC%D1%8F)(Всемирным координированным временем) нет.\r\n\r\nПриложение работало всю зиму, но в первый день апреля уведомление ушло на час позже полудня. Дело в том, что разработчик просто прикрутил таймер, который срабатывает раз в 24 часа, а в последний день марта в Великобритании перевели время на час вперед — на летнее. Разработчик быстро исправил ситуацию, заменив таймер на планировщик задач — последний срабатывает в соответствии с локальным временем сервера, который находится в квартире разработчика.\r\n\r\nКак ни странно, приложение пользуется популярностью. Спустя некоторое время разработчик получает отзыв от разгневанного пользователя из Нью-Йорка: тот пишет, что уведомления приходят вовсе не в полдень, а в 7 утра. Нью-Йорк находится в часовом поясе ETC (Eastern Time Zone), то есть UTC -05:00.\r\n\r\nДля разных часовых поясов нельзя просто ставить таймер, отталкиваясь от времени на сервере. Разработчик добавил выбор часового пояса при регистрации и переписал планировщик так, чтобы он стартовал для каждого пользователя раньше или позже на несколько часов в зависимости от данных, которые человек ввёл при регистрации.\r\n\r\nКажется, теперь все работает хорошо — но в ноябре разработчику начали писать пользователи из Москвы и жаловаться на уведомления, которые приходят в 11 утра. Дело в том, что в Великобритании время переводится с зимнего на летнее, а в России — нет. К тому моменту разработчик начал нервничать и решил просто записать в коде, что если пользователь из России, то делать переход во времени для него не нужно. Приложение тем временем только набирало популярность.\r\n\r\nВ марте, разработчику пишут из Сиднея — им уведомление приходит в два часа дня. Все потому, что в южном полушарии в марте происходит переход не на летнее время, а на зимнее. Разработчику не оставалось ничего иного, как дополнить свой код информацией о том, что в южном полушарии надо переводить время наоборот. Приложение уже стало популярным во всём мире и на разработчика посыпались претензии со всех уголков света. Например, пользователи из Палестины писали жалобы каждый раз, как их правительство переводило часы с летнего времени на зимнее, а случается это каждый раз в разное время. А ещё дело было в 2011 году, и 29 декабря пользователи из Самоа дружно пересекли [линию перемены даты](https://ru.wikipedia.org/wiki/%D0%9B%D0%B8%D0%BD%D0%B8%D1%8F_%D0%BF%D0%B5%D1%80%D0%B5%D0%BC%D0%B5%D0%BD%D1%8B_%D0%B4%D0%B0%D1%82%D1%8B), совершив скачок в будущее и очутившись 31 декабря 2011 года. Еще через год разработчику позвонили из [Международной службы вращения Земли](https://ru.wikipedia.org/wiki/%D0%9C%D0%B5%D0%B6%D0%B4%D1%83%D0%BD%D0%B0%D1%80%D0%BE%D0%B4%D0%BD%D0%B0%D1%8F_%D1%81%D0%BB%D1%83%D0%B6%D0%B1%D0%B0_%D0%B2%D1%80%D0%B0%D1%89%D0%B5%D0%BD%D0%B8%D1%8F_%D0%97%D0%B5%D0%BC%D0%BB%D0%B8) и сообщили, что ему надо поправить приложение, потому что в этом году будет на\r\nодну секунду больше, чем в прошлом из-за добавления дополнительной секунды. Записывать\r\nвсе это в код — совершенно безумная идея. Поэтому разработчик закрыл проект и больше никогда не писал код.\r\n\r\n### Пара советов, как работать с локальным временем\r\n\r\nЧто бы мог сделать разработчик, чтобы избежать такого печального финала? Все просто — достаточно было учесть опыт тех, кто уже решал эту проблему и выкладывал решение в открытый доступ. В каждом языке программирования есть как минимум одна библиотека, которая умеет работать с человеческим временем и часовыми поясами. Более того, существует база данных, в которой хранится информация о локальном времени, в том числе в исторической перспективе.\r\n\r\nЧто бы мог сделать разработчик:\r\n\r\n- Переложить ответственность за подсчет времени на библиотеки и базы данных;\r\n- Хранить не разницу во времени с локальным временем пользователя, а его часовой\r\nпояс в соответствии с [классификацией IANA](https://en.wikipedia.org/wiki/List_of_tz_database_time_zones),\r\n то есть не +3, а \"Europe/Moscow\";\r\n- Настроить планировщик так, чтобы он выбирал задачи из списка в соответствии с\r\nчасовым поясом.\r\n\r\n### Пара советов, как НЕ работать с локальным временем\r\n\r\nСуществует много сценариев, при которых крайне важна последовательность событий, на которую не могут повлиять ни изменения часовых поясов, ни переход на летнее время.\r\n\r\nПредположим, нужно написать сервис, обслуживающий банковские транзакции. Его функционал крайне простой: один пользователь отправляет другому деньги банковским переводом, любой из них может получить информацию обо всех переводах. В данном случае нужно хранить информацию о том, кто, кому, когда и сколько денег перевёл. При этом показывать эту информацию нужно в локальном времени пользователя при\r\nусловии, что транзакции в любом часовом поясе идут одна за другой.\r\n\r\nУчтём опыт разработчика из примера выше и не будем описывать время самостоятельно, а используем библиотеку. Тут мы столкнемся с рядом вопросов:\r\n\r\n- В каком часовом поясе хранить транзакции, если клиенты живут в разных странах,\r\nа база данных находится в третьей стране;\r\n- Как учитывать переход на летнее и зимнее время;\r\n- Как учитывать дополнительную секунду;\r\n- Что делать, если пользователь переехал из одной страны в другую.\r\n\r\nИ лучшим решением будет не делать ничего. Не нужно учитывать временные зоны клиентов, не нужно учитывать переходы на летнее и зимнее время, не нужно учитывать переезды пользователей. Когда нужно абсолютное время, достаточно использовать UTC.\r\n\r\nВремя в UTC поддерживается почти всеми библиотеками и почти всеми база и данных. Используя UTC, мы можем получать транзакцию в локальном времени пользователя, переводить это время в UTC и сохранять у себя. И транзакции всегда будут идти одна за другой.\r\n\r\nПравда, существует проблема: UTC учитывает дополнительные секунды, а значит, в какой-то момент мы можем получить такие транзакции:\r\n\r\n- Маша Феде перевела 100 рублей 2016-12-31 в 23:59:58\r\n- Федя Юре перевёл 100 рублей 2016-12-31 в 23:59:59\r\n- Юра отправил Майклу 1$ 2016-12-31 в 23:59:60\r\n- Майкл отправил Теду 99$ 2017-01-01 в 00:00:00\r\n\r\nСкорее всего программа, работающая с UTC, не сможет сказать, что произошло раньше — вторая транзакция, третья или четвертая. Google, которой очень важна абсолютная последовательность событий во времени, очень интересно решила эту проблему. В своих серверах точного времени они добавляют дополнительную секунду не за раз, а равномерно распределяют её в течение дня в начале года. Это называется [leap smear](https://developers.google.com/time/smear). Таким образом, раз в год, когда астрономы объявляют дополнительную секунду, в течение суток сервера точного времени Google немного отстают от UTC, чтобы набрать дополнительную секунду.\r\n\r\n\r\n\r\n## Выводы\r\n\r\n- В любом современном ПК есть процессор, который умеет производить колебания с\r\nравными интервалами.\r\n- Зная период этих колебаний, операционная система компьютера может считать\r\nчеловеческое время.\r\n- С помощью протокола NTP операционная система может обращаться к серверам\r\nточного времени для скорректировать своих часов.\r\n- Когда компьютер выключен, время считает специальное устройство — RTC, которое\r\nпитается от батареи.\r\n- Epoch — это некая дата, от которой принято отсчитывать время вперед или назад.\r\nРазные операционные системы и языки программирования могут выбирать свой собственный\r\nepoch.\r\n- Unix-timestamp — это количество секунд, прошедших с unix-epoch, то есть\r\nс 00:00 1 января 1970 года.\r\n- Никогда не пишите самостоятельно код, который описывает человеческое время.\r\n- Если вам нужна точная последовательность действий во времени, не используйте\r\nчасовые зоны. Храните время в UTC и по требованию конвертируйте его.\r\n\r\n## Дополнительные материалы\r\n\r\n- [Как компьютеры складывают числа](https://www.youtube.com/watch?v=YuSgZ173Utg)","reading_time":10,"url":"https://ru.hexlet.io/blog/posts/time","cover_thumb_variant":"https://hexlet.io/rails/active_storage/representations/proxy/eyJfcmFpbHMiOnsiZGF0YSI6NDIzLCJwdXIiOiJibG9iX2lkIn19--5c6c5168a4dcfa6bb0bed8ae50c0877c3db687e3/eyJfcmFpbHMiOnsiZGF0YSI6eyJmb3JtYXQiOiJ3ZWJwIiwicmVzaXplX3RvX2xpbWl0IjpbMTAwLDUwXSwic2F2ZXIiOnsicXVhbGl0eSI6ODV9fSwicHVyIjoidmFyaWF0aW9uIn19--f0d4342fcdbdffa15c37fb02bfb423ac88d5c0c9/time.png","cover_list_variant":"https://hexlet.io/rails/active_storage/representations/proxy/eyJfcmFpbHMiOnsiZGF0YSI6NDIzLCJwdXIiOiJibG9iX2lkIn19--5c6c5168a4dcfa6bb0bed8ae50c0877c3db687e3/eyJfcmFpbHMiOnsiZGF0YSI6eyJmb3JtYXQiOiJ3ZWJwIiwicmVzaXplX2FuZF9wYWQiOls3MDgsMzU0XSwic2F2ZXIiOnsicXVhbGl0eSI6ODV9fSwicHVyIjoidmFyaWF0aW9uIn19--324dc52aa55ebe818c2a887ebcb832b9ad1c0381/time.png","cover_main_variant":"https://hexlet.io/rails/active_storage/representations/proxy/eyJfcmFpbHMiOnsiZGF0YSI6NDIzLCJwdXIiOiJibG9iX2lkIn19--5c6c5168a4dcfa6bb0bed8ae50c0877c3db687e3/eyJfcmFpbHMiOnsiZGF0YSI6eyJmb3JtYXQiOiJ3ZWJwIiwicmVzaXplX2FuZF9wYWQiOlsxNDU2LDcyOF0sInNhdmVyIjp7InF1YWxpdHkiOjg1fX0sInB1ciI6InZhcmlhdGlvbiJ9fQ==--b2ecf6121248a24883cda19469732b47b57f4e80/time.png","related_stacks_count":5},"relatedPosts":[{"model_name":"BlogPost","id":2772,"title":"Что такое Makefile и как начать его использовать","slug":"makefile-as-task-runner","summary":"Этот гайд расскажет, как использование инструмента Makefile позволит свести процесс разворачивания проекта к нескольким коротким и понятным командам","created_at":"2024-11-13T16:54:35.412Z","published_at":"2024-12-18T13:54:20.591Z","cover_list_variant":"https://hexlet.io/rails/active_storage/representations/proxy/eyJfcmFpbHMiOnsiZGF0YSI6NDU5LCJwdXIiOiJibG9iX2lkIn19--97454b3b2dea7bca2f8da19e43e311b790c71130/eyJfcmFpbHMiOnsiZGF0YSI6eyJmb3JtYXQiOiJ3ZWJwIiwicmVzaXplX2FuZF9wYWQiOls3MDgsMzU0XSwic2F2ZXIiOnsicXVhbGl0eSI6ODV9fSwicHVyIjoidmFyaWF0aW9uIn19--324dc52aa55ebe818c2a887ebcb832b9ad1c0381/cover.jpg"},{"model_name":"BlogPost","id":2780,"title":"Что такое REST API?","slug":"rest-api","summary":"REST API применяется везде, где есть необходимость предоставления данных с сервера пользователю веб-приложения или сайта. Всё о REST API: от истории к принципам","created_at":"2024-11-18T15:51:47.262Z","published_at":"2024-12-18T13:54:20.467Z","cover_list_variant":"https://hexlet.io/rails/active_storage/representations/proxy/eyJfcmFpbHMiOnsiZGF0YSI6NDI3LCJwdXIiOiJibG9iX2lkIn19--836a0585b4618daeee1ff29040ebb7a9c5aaf365/eyJfcmFpbHMiOnsiZGF0YSI6eyJmb3JtYXQiOiJ3ZWJwIiwicmVzaXplX2FuZF9wYWQiOls3MDgsMzU0XSwic2F2ZXIiOnsicXVhbGl0eSI6ODV9fSwicHVyIjoidmFyaWF0aW9uIn19--324dc52aa55ebe818c2a887ebcb832b9ad1c0381/%D0%A7%D1%82%D0%BE%D1%82%D0%B0%D0%BA%D0%BE%D0%B5RESTAPI_.png"},{"model_name":"BlogPost","id":2775,"title":"Что такое хостинг и домен сайта простыми словами","slug":"hosting","summary":"Чтобы создать собственный сайт, вам понадобится домен, хостинг и курсы программирования. Рассказываем просто о сложном процессе","created_at":"2024-11-15T08:21:52.376Z","published_at":"2024-12-18T13:54:20.205Z","cover_list_variant":"https://hexlet.io/rails/active_storage/representations/proxy/eyJfcmFpbHMiOnsiZGF0YSI6NDQ3LCJwdXIiOiJibG9iX2lkIn19--ffcfe2a0a0c94c3c08e708d19a79ea94dce9b54b/eyJfcmFpbHMiOnsiZGF0YSI6eyJmb3JtYXQiOiJ3ZWJwIiwicmVzaXplX2FuZF9wYWQiOls3MDgsMzU0XSwic2F2ZXIiOnsicXVhbGl0eSI6ODV9fSwicHVyIjoidmFyaWF0aW9uIn19--324dc52aa55ebe818c2a887ebcb832b9ad1c0381/servers.png"}],"category":{"id":4,"name":"Код","slug":"code","state":"published","created_at":"2016-08-23T13:33:44.258Z"},"mainStackCategory":{"id":2,"name":"Курсы по веб-разработке","slug":"web_development","short_name":"Веб-разработка","order":190,"state":"published","category_slug":"courses_web_development"},"categories":[{"id":6,"name":"Мотивация","slug":"motivation","state":"published","created_at":"2016-10-06T18:31:38.903Z"},{"id":3,"name":"Истории успеха","slug":"success","state":"published","created_at":"2016-07-30T12:57:18.308Z"},{"id":14,"name":"Дневник студента","slug":"student-diary","state":"published","created_at":"2019-02-25T13:27:09.471Z"},{"id":4,"name":"Код","slug":"code","state":"published","created_at":"2016-08-23T13:33:44.258Z"},{"id":12,"name":"Карьера","slug":"career","state":"published","created_at":"2017-07-21T15:42:21.481Z"}],"relatedLandings":[{"stack":{"id":34,"slug":"algorithms","title":"Алгоритмы и структуры данных","audience":"for_programmers","start_type":"anytime","pricing_model":"subscription","priority":"medium","kind":"track","state":"published","stack_state":"finished","order":4000,"duration_in_months":2},"id":56,"slug":"algorithms","title":"Алгоритмы и структуры данных","subtitle":"Навык, который увеличит ваши шансы пройти алгоритмическое интервью в международные компании на 80%","subtitle_for_lists":"Алгоритмы для собеседований","locale":"ru","current":true,"duration_in_months_text":"2 месяца","stack_slug":"algorithms","price_text":"от 3 900 ₽","duration_text":"2 месяца","cover_list_variant":"https://hexlet.io/rails/active_storage/representations/proxy/eyJfcmFpbHMiOnsiZGF0YSI6NDAyOCwicHVyIjoiYmxvYl9pZCJ9fQ==--ae9eed98663dd1201759d042a5ba7ca790866156/eyJfcmFpbHMiOnsiZGF0YSI6eyJmb3JtYXQiOiJ3ZWJwIiwicmVzaXplX3RvX2xpbWl0IjpbNDAwLDQwMF0sInNhdmVyIjp7InF1YWxpdHkiOjg1fX0sInB1ciI6InZhcmlhdGlvbiJ9fQ==--5b6f46dacd1af664f27558553a58076185091823/Programming-bro.png"},{"stack":{"id":50,"slug":"discrete-mathematics","title":"Дискретная математика","audience":"for_beginners","start_type":"anytime","pricing_model":"subscription","priority":"medium","kind":"track","state":"published","stack_state":"finished","order":4650,"duration_in_months":1},"id":88,"slug":"discrete-mathematics","title":"Дискретная математика","subtitle":"Навык дискретной математики для укрепления теоретических знаний и лучшего понимания алгоритмов и структур данных","subtitle_for_lists":"Дискретная математика для программистов","locale":"ru","current":true,"duration_in_months_text":"1 месяц","stack_slug":"discrete-mathematics","price_text":"от 3 900 ₽","duration_text":"1 месяц","cover_list_variant":"https://hexlet.io/rails/active_storage/representations/proxy/eyJfcmFpbHMiOnsiZGF0YSI6MzY2MSwicHVyIjoiYmxvYl9pZCJ9fQ==--e9c2b6bde361adaac625a7f47d8b9671c17f3ddb/eyJfcmFpbHMiOnsiZGF0YSI6eyJmb3JtYXQiOiJ3ZWJwIiwicmVzaXplX3RvX2xpbWl0IjpbNDAwLDQwMF0sInNhdmVyIjp7InF1YWxpdHkiOjg1fX0sInB1ciI6InZhcmlhdGlvbiJ9fQ==--5b6f46dacd1af664f27558553a58076185091823/Mathematics-bro.png"},{"stack":{"id":37,"slug":"python-sicp","title":"СИКП на Python","audience":"for_programmers","start_type":"anytime","pricing_model":"subscription","priority":"medium","kind":"track","state":"published","stack_state":"finished","order":4150,"duration_in_months":1},"id":62,"slug":"python-sicp","title":"СИКП на Python","subtitle":"Навык понимать код на фундаментальном уровне, уверенно проходить собеседования и решать сложные задачи","subtitle_for_lists":"Изучите Python на глубоком уровне для решения сложных задач","locale":"ru","current":true,"duration_in_months_text":"1 месяц","stack_slug":"python-sicp","price_text":"от 3 900 ₽","duration_text":"1 месяц","cover_list_variant":"https://hexlet.io/rails/active_storage/representations/proxy/eyJfcmFpbHMiOnsiZGF0YSI6Mzc1OCwicHVyIjoiYmxvYl9pZCJ9fQ==--023ea18f500b1c4c91617fa96bbc52df8395da39/eyJfcmFpbHMiOnsiZGF0YSI6eyJmb3JtYXQiOiJ3ZWJwIiwicmVzaXplX3RvX2xpbWl0IjpbNDAwLDQwMF0sInNhdmVyIjp7InF1YWxpdHkiOjg1fX0sInB1ciI6InZhcmlhdGlvbiJ9fQ==--5b6f46dacd1af664f27558553a58076185091823/Software%20engineer-bro.png"},{"stack":{"id":36,"slug":"java-sicp","title":"СИКП на Java","audience":"for_programmers","start_type":"anytime","pricing_model":"subscription","priority":"medium","kind":"track","state":"published","stack_state":"finished","order":4100,"duration_in_months":1},"id":60,"slug":"java-sicp","title":"СИКП на Java","subtitle":"Навык понимать программы на фундаментальном уровне, уверенно проходить собеседования и решать сложные задачи","subtitle_for_lists":"Изучите фундаментальные принципы программирования на Java","locale":"ru","current":true,"duration_in_months_text":"1 месяц","stack_slug":"java-sicp","price_text":"от 3 900 ₽","duration_text":"1 месяц","cover_list_variant":"https://hexlet.io/rails/active_storage/representations/proxy/eyJfcmFpbHMiOnsiZGF0YSI6NDAxNiwicHVyIjoiYmxvYl9pZCJ9fQ==--eb66b9b5e26fafa32844ce0f4522c3ed84544040/eyJfcmFpbHMiOnsiZGF0YSI6eyJmb3JtYXQiOiJ3ZWJwIiwicmVzaXplX3RvX2xpbWl0IjpbNDAwLDQwMF0sInNhdmVyIjp7InF1YWxpdHkiOjg1fX0sInB1ciI6InZhcmlhdGlvbiJ9fQ==--5b6f46dacd1af664f27558553a58076185091823/Hand%20coding-rafiki.png"},{"stack":{"id":117,"slug":"linear-algebra","title":"Линейная алгебра","audience":"for_beginners","start_type":"anytime","pricing_model":"subscription","priority":"medium","kind":"track","state":"published","stack_state":"finished","order":4600,"duration_in_months":1},"id":192,"slug":"linear-algebra","title":"Линейная алгебра","subtitle":"Навык работы с векторами и матрицами для решения систем уравнений и выполнения матричных разложений","subtitle_for_lists":"Освоите работу с векторами, матрицами и системами уравнений","locale":"ru","current":true,"duration_in_months_text":"1 месяц","stack_slug":"linear-algebra","price_text":"от 3 900 ₽","duration_text":"1 месяц","cover_list_variant":"https://hexlet.io/rails/active_storage/representations/proxy/eyJfcmFpbHMiOnsiZGF0YSI6MzY0NywicHVyIjoiYmxvYl9pZCJ9fQ==--fd23784224c1dfaefc6e1cd556a43d164e3c412f/eyJfcmFpbHMiOnsiZGF0YSI6eyJmb3JtYXQiOiJ3ZWJwIiwicmVzaXplX3RvX2xpbWl0IjpbNDAwLDQwMF0sInNhdmVyIjp7InF1YWxpdHkiOjg1fX0sInB1ciI6InZhcmlhdGlvbiJ9fQ==--5b6f46dacd1af664f27558553a58076185091823/Mathematics-amico.png"}]},"url":"/blog/posts/time","version":"0b0c6d4ebbd40fd58630a0dd89cc25544ccdf24e","encryptHistory":false,"clearHistory":false}"><style data-mantine-styles="true">:root, :host{--mantine-font-family: Arial, sans-serif;--mantine-font-family-headings: Arial, sans-serif;--mantine-heading-font-weight: normal;--mantine-radius-default: 0rem;--mantine-primary-color-filled: var(--mantine-color-indigo-filled);--mantine-primary-color-filled-hover: var(--mantine-color-indigo-filled-hover);--mantine-primary-color-light: var(--mantine-color-indigo-light);--mantine-primary-color-light-hover: var(--mantine-color-indigo-light-hover);--mantine-primary-color-light-color: var(--mantine-color-indigo-light-color);--mantine-spacing-xxl: calc(4rem * var(--mantine-scale));--mantine-font-size-xs: 12px;--mantine-font-size-sm: 14px;--mantine-font-size-md: 16px;--mantine-font-size-lg: clamp(16.0000px, calc(15.2727px + 0.2273vw), 18.0000px);--mantine-font-size-xl: clamp(16.0000px, calc(14.5455px + 0.4545vw), 20.0000px);--mantine-font-size-display-3: clamp(32.0000px, calc(26.1818px + 1.8182vw), 48.0000px);--mantine-font-size-display-2: clamp(36.0000px, calc(25.8182px + 3.1818vw), 64.0000px);--mantine-font-size-display-1: clamp(40.0000px, calc(25.4545px + 4.5455vw), 80.0000px);--mantine-font-size-h1: clamp(28.0000px, calc(23.6364px + 1.3636vw), 40.0000px);--mantine-font-size-h2: clamp(24.0000px, calc(21.0909px + 0.9091vw), 32.0000px);--mantine-font-size-h3: clamp(20.0000px, calc(17.0909px + 0.9091vw), 28.0000px);--mantine-font-size-h4: clamp(16.0000px, calc(13.0909px + 0.9091vw), 24.0000px);--mantine-font-size-h5: clamp(16.0000px, calc(14.5455px + 0.4545vw), 20.0000px);--mantine-font-size-h6: 1rem;--mantine-primary-color-0: var(--mantine-color-indigo-0);--mantine-primary-color-1: var(--mantine-color-indigo-1);--mantine-primary-color-2: var(--mantine-color-indigo-2);--mantine-primary-color-3: var(--mantine-color-indigo-3);--mantine-primary-color-4: var(--mantine-color-indigo-4);--mantine-primary-color-5: var(--mantine-color-indigo-5);--mantine-primary-color-6: var(--mantine-color-indigo-6);--mantine-primary-color-7: var(--mantine-color-indigo-7);--mantine-primary-color-8: var(--mantine-color-indigo-8);--mantine-primary-color-9: var(--mantine-color-indigo-9);--mantine-color-red-0: #ffeaea;--mantine-color-red-1: #fed4d4;--mantine-color-red-2: #f4a7a8;--mantine-color-red-3: #ec7878;--mantine-color-red-4: #e55050;--mantine-color-red-5: #e03131;--mantine-color-red-6: #e02829;--mantine-color-red-7: #c71a1c;--mantine-color-red-8: #b21218;--mantine-color-red-9: #9c0411;--mantine-color-violet-0: #fce9ff;--mantine-color-violet-1: #f1cfff;--mantine-color-violet-2: #e09bff;--mantine-color-violet-3: #d16fff;--mantine-color-violet-4: #be37fe;--mantine-color-violet-5: #b51afe;--mantine-color-violet-6: #b009ff;--mantine-color-violet-7: #9b00e4;--mantine-color-violet-8: #8a00cc;--mantine-color-violet-9: #7800b3;--mantine-color-indigo-0: #edecff;--mantine-color-indigo-1: #d6d5fe;--mantine-color-indigo-2: #aaa9f4;--mantine-color-indigo-3: #7b79eb;--mantine-color-indigo-4: #5451e4;--mantine-color-indigo-5: #3b37e0;--mantine-color-indigo-6: #2d2adf;--mantine-color-indigo-7: #1f1ec7;--mantine-color-indigo-8: #1819b2;--mantine-color-indigo-9: #0c149e;--mantine-color-cyan-0: #dffdff;--mantine-color-cyan-1: #caf5ff;--mantine-color-cyan-2: #99e8ff;--mantine-color-cyan-3: #64daff;--mantine-color-cyan-4: #3ccffe;--mantine-color-cyan-5: #24c8fe;--mantine-color-cyan-6: #00c2ff;--mantine-color-cyan-7: #00ade4;--mantine-color-cyan-8: #009acd;--mantine-color-cyan-9: #0085b5;--mantine-color-green-0: #e9fdec;--mantine-color-green-1: #d7f6dc;--mantine-color-green-2: #b0eab9;--mantine-color-green-3: #86df94;--mantine-color-green-4: #62d574;--mantine-color-green-5: #4ccf5f;--mantine-color-green-6: #3fcc54;--mantine-color-green-7: #2fb344;--mantine-color-green-8: #25a03b;--mantine-color-green-9: #138a2e;--mantine-color-yellow-0: #fff7e2;--mantine-color-yellow-1: #ffeecd;--mantine-color-yellow-2: #ffdc9c;--mantine-color-yellow-3: #ffc966;--mantine-color-yellow-4: #feb93a;--mantine-color-yellow-5: #feae1e;--mantine-color-yellow-6: #ffa90f;--mantine-color-yellow-8: #ca8200;--mantine-color-yellow-9: #af7000;--mantine-h1-font-size: clamp(28.0000px, calc(23.6364px + 1.3636vw), 40.0000px);--mantine-h1-font-weight: normal;--mantine-h2-font-size: clamp(24.0000px, calc(21.0909px + 0.9091vw), 32.0000px);--mantine-h2-font-weight: normal;--mantine-h3-font-size: clamp(20.0000px, calc(17.0909px + 0.9091vw), 28.0000px);--mantine-h3-font-weight: normal;--mantine-h4-font-size: clamp(16.0000px, calc(13.0909px + 0.9091vw), 24.0000px);--mantine-h4-font-weight: normal;--mantine-h5-font-size: clamp(16.0000px, calc(14.5455px + 0.4545vw), 20.0000px);--mantine-h5-font-weight: normal;--mantine-h6-font-size: 1rem;--mantine-h6-font-weight: normal;}
:root[data-mantine-color-scheme="dark"], :host([data-mantine-color-scheme="dark"]){--mantine-color-anchor: var(--mantine-color-text);--mantine-color-dimmed: #495057;--mantine-color-dark-filled: var(--mantine-color-dark-5);--mantine-color-dark-filled-hover: var(--mantine-color-dark-6);--mantine-color-dark-light: rgba(105, 105, 105, 0.15);--mantine-color-dark-light-hover: rgba(105, 105, 105, 0.2);--mantine-color-dark-light-color: var(--mantine-color-dark-0);--mantine-color-dark-outline: var(--mantine-color-dark-1);--mantine-color-dark-outline-hover: rgba(184, 184, 184, 0.05);--mantine-color-gray-filled: var(--mantine-color-gray-5);--mantine-color-gray-filled-hover: var(--mantine-color-gray-6);--mantine-color-gray-light: rgba(222, 226, 230, 0.15);--mantine-color-gray-light-hover: rgba(222, 226, 230, 0.2);--mantine-color-gray-light-color: var(--mantine-color-gray-0);--mantine-color-gray-outline: var(--mantine-color-gray-1);--mantine-color-gray-outline-hover: rgba(241, 243, 245, 0.05);--mantine-color-red-filled: var(--mantine-color-red-5);--mantine-color-red-filled-hover: var(--mantine-color-red-6);--mantine-color-red-light: rgba(236, 120, 120, 0.15);--mantine-color-red-light-hover: rgba(236, 120, 120, 0.2);--mantine-color-red-light-color: var(--mantine-color-red-0);--mantine-color-red-outline: var(--mantine-color-red-1);--mantine-color-red-outline-hover: rgba(254, 212, 212, 0.05);--mantine-color-pink-filled: var(--mantine-color-pink-5);--mantine-color-pink-filled-hover: var(--mantine-color-pink-6);--mantine-color-pink-light: rgba(250, 162, 193, 0.15);--mantine-color-pink-light-hover: rgba(250, 162, 193, 0.2);--mantine-color-pink-light-color: var(--mantine-color-pink-0);--mantine-color-pink-outline: var(--mantine-color-pink-1);--mantine-color-pink-outline-hover: rgba(255, 222, 235, 0.05);--mantine-color-grape-filled: var(--mantine-color-grape-5);--mantine-color-grape-filled-hover: var(--mantine-color-grape-6);--mantine-color-grape-light: rgba(229, 153, 247, 0.15);--mantine-color-grape-light-hover: rgba(229, 153, 247, 0.2);--mantine-color-grape-light-color: var(--mantine-color-grape-0);--mantine-color-grape-outline: var(--mantine-color-grape-1);--mantine-color-grape-outline-hover: rgba(243, 217, 250, 0.05);--mantine-color-violet-filled: var(--mantine-color-violet-5);--mantine-color-violet-filled-hover: var(--mantine-color-violet-6);--mantine-color-violet-light: rgba(209, 111, 255, 0.15);--mantine-color-violet-light-hover: rgba(209, 111, 255, 0.2);--mantine-color-violet-light-color: var(--mantine-color-violet-0);--mantine-color-violet-outline: var(--mantine-color-violet-1);--mantine-color-violet-outline-hover: rgba(241, 207, 255, 0.05);--mantine-color-indigo-filled: var(--mantine-color-indigo-5);--mantine-color-indigo-filled-hover: var(--mantine-color-indigo-6);--mantine-color-indigo-light: rgba(123, 121, 235, 0.15);--mantine-color-indigo-light-hover: rgba(123, 121, 235, 0.2);--mantine-color-indigo-light-color: var(--mantine-color-indigo-0);--mantine-color-indigo-outline: var(--mantine-color-indigo-1);--mantine-color-indigo-outline-hover: rgba(214, 213, 254, 0.05);--mantine-color-blue-filled: var(--mantine-color-blue-5);--mantine-color-blue-filled-hover: var(--mantine-color-blue-6);--mantine-color-blue-light: rgba(116, 192, 252, 0.15);--mantine-color-blue-light-hover: rgba(116, 192, 252, 0.2);--mantine-color-blue-light-color: var(--mantine-color-blue-0);--mantine-color-blue-outline: var(--mantine-color-blue-1);--mantine-color-blue-outline-hover: rgba(208, 235, 255, 0.05);--mantine-color-cyan-filled: var(--mantine-color-cyan-5);--mantine-color-cyan-filled-hover: var(--mantine-color-cyan-6);--mantine-color-cyan-light: rgba(100, 218, 255, 0.15);--mantine-color-cyan-light-hover: rgba(100, 218, 255, 0.2);--mantine-color-cyan-light-color: var(--mantine-color-cyan-0);--mantine-color-cyan-outline: var(--mantine-color-cyan-1);--mantine-color-cyan-outline-hover: rgba(202, 245, 255, 0.05);--mantine-color-teal-filled: var(--mantine-color-teal-5);--mantine-color-teal-filled-hover: var(--mantine-color-teal-6);--mantine-color-teal-light: rgba(99, 230, 190, 0.15);--mantine-color-teal-light-hover: rgba(99, 230, 190, 0.2);--mantine-color-teal-light-color: var(--mantine-color-teal-0);--mantine-color-teal-outline: var(--mantine-color-teal-1);--mantine-color-teal-outline-hover: rgba(195, 250, 232, 0.05);--mantine-color-green-filled: var(--mantine-color-green-5);--mantine-color-green-filled-hover: var(--mantine-color-green-6);--mantine-color-green-light: rgba(134, 223, 148, 0.15);--mantine-color-green-light-hover: rgba(134, 223, 148, 0.2);--mantine-color-green-light-color: var(--mantine-color-green-0);--mantine-color-green-outline: var(--mantine-color-green-1);--mantine-color-green-outline-hover: rgba(215, 246, 220, 0.05);--mantine-color-lime-filled: var(--mantine-color-lime-5);--mantine-color-lime-filled-hover: var(--mantine-color-lime-6);--mantine-color-lime-light: rgba(192, 235, 117, 0.15);--mantine-color-lime-light-hover: rgba(192, 235, 117, 0.2);--mantine-color-lime-light-color: var(--mantine-color-lime-0);--mantine-color-lime-outline: var(--mantine-color-lime-1);--mantine-color-lime-outline-hover: rgba(233, 250, 200, 0.05);--mantine-color-yellow-filled: var(--mantine-color-yellow-5);--mantine-color-yellow-filled-hover: var(--mantine-color-yellow-6);--mantine-color-yellow-light: rgba(255, 201, 102, 0.15);--mantine-color-yellow-light-hover: rgba(255, 201, 102, 0.2);--mantine-color-yellow-light-color: var(--mantine-color-yellow-0);--mantine-color-yellow-outline: var(--mantine-color-yellow-1);--mantine-color-yellow-outline-hover: rgba(255, 238, 205, 0.05);--mantine-color-orange-filled: var(--mantine-color-orange-5);--mantine-color-orange-filled-hover: var(--mantine-color-orange-6);--mantine-color-orange-light: rgba(255, 192, 120, 0.15);--mantine-color-orange-light-hover: rgba(255, 192, 120, 0.2);--mantine-color-orange-light-color: var(--mantine-color-orange-0);--mantine-color-orange-outline: var(--mantine-color-orange-1);--mantine-color-orange-outline-hover: rgba(255, 232, 204, 0.05);--app-cta-gradient: linear-gradient(90deg, var(--mantine-color-blue-9) 0%, var(--mantine-color-cyan-7) 100%);--app-color-surface: #2e2e2e;}
:root[data-mantine-color-scheme="light"], :host([data-mantine-color-scheme="light"]){--mantine-color-anchor: var(--mantine-color-text);--mantine-color-dimmed: #495057;--mantine-color-red-light: rgba(224, 40, 41, 0.1);--mantine-color-red-light-hover: rgba(224, 40, 41, 0.12);--mantine-color-red-outline-hover: rgba(224, 40, 41, 0.05);--mantine-color-violet-light: rgba(176, 9, 255, 0.1);--mantine-color-violet-light-hover: rgba(176, 9, 255, 0.12);--mantine-color-violet-outline-hover: rgba(176, 9, 255, 0.05);--mantine-color-indigo-light: rgba(45, 42, 223, 0.1);--mantine-color-indigo-light-hover: rgba(45, 42, 223, 0.12);--mantine-color-indigo-outline-hover: rgba(45, 42, 223, 0.05);--mantine-color-cyan-light: rgba(0, 194, 255, 0.1);--mantine-color-cyan-light-hover: rgba(0, 194, 255, 0.12);--mantine-color-cyan-outline-hover: rgba(0, 194, 255, 0.05);--mantine-color-green-light: rgba(63, 204, 84, 0.1);--mantine-color-green-light-hover: rgba(63, 204, 84, 0.12);--mantine-color-green-outline-hover: rgba(63, 204, 84, 0.05);--mantine-color-yellow-light: rgba(255, 169, 15, 0.1);--mantine-color-yellow-light-hover: rgba(255, 169, 15, 0.12);--mantine-color-yellow-outline-hover: rgba(255, 169, 15, 0.05);--app-color-surface: #f1f3f5;--app-cta-gradient: linear-gradient(90deg, var(--mantine-color-blue-filled) 0%, var(--mantine-color-cyan-5) 100%);}</style><style data-mantine-styles="classes">@media (max-width: 35.99375em) {.mantine-visible-from-xs {display: none !important;}}@media (min-width: 36em) {.mantine-hidden-from-xs {display: none !important;}}@media (max-width: 47.99375em) {.mantine-visible-from-sm {display: none !important;}}@media (min-width: 48em) {.mantine-hidden-from-sm {display: none !important;}}@media (max-width: 61.99375em) {.mantine-visible-from-md {display: none !important;}}@media (min-width: 62em) {.mantine-hidden-from-md {display: none !important;}}@media (max-width: 74.99375em) {.mantine-visible-from-lg {display: none !important;}}@media (min-width: 75em) {.mantine-hidden-from-lg {display: none !important;}}@media (max-width: 87.99375em) {.mantine-visible-from-xl {display: none !important;}}@media (min-width: 88em) {.mantine-hidden-from-xl {display: none !important;}}</style><script type="application/ld+json">{"@context":"https://schema.org","@type":"Article","author":"Валерия Белякова","name":"Что такое время","datePublished":"2024-12-18T13:54:19.292Z","headline":"Как современный компьютер хранит информацию, как он отсчитывает секунды, почему почти никогда не ошибается и как с этим временем работать","image":"https://hexlet.io/rails/active_storage/representations/proxy/eyJfcmFpbHMiOnsiZGF0YSI6NDIzLCJwdXIiOiJibG9iX2lkIn19--5c6c5168a4dcfa6bb0bed8ae50c0877c3db687e3/eyJfcmFpbHMiOnsiZGF0YSI6eyJmb3JtYXQiOiJ3ZWJwIiwicmVzaXplX2FuZF9wYWQiOlsxNDU2LDcyOF0sInNhdmVyIjp7InF1YWxpdHkiOjg1fX0sInB1ciI6InZhcmlhdGlvbiJ9fQ==--b2ecf6121248a24883cda19469732b47b57f4e80/time.png","interactionStatistic":[{"@type":"InteractionCounter","interactionType":{"@type":"LikeAction"},"userInteractionCount":0}]}</script><div style="--container-size:var(--container-size-lg);margin-top:var(--mantine-spacing-xl);height:100%" class="m_7485cace mantine-Container-root" data-size="lg" data-strategy="block"><script type="application/ld+json">{"@context":"https://schema.org","@type":"BreadcrumbList","itemListElement":[{"position":1,"@type":"ListItem","item":{"@id":"/blog","name":"Блог Хекслета"}},{"position":2,"@type":"ListItem","item":{"@id":"/blog/categories/code","name":"Код"}},{"position":3,"@type":"ListItem","item":{"@id":"/blog/posts/time","name":"Что такое время"}}]}</script><div style="margin-bottom:var(--mantine-spacing-xs)" class="m_8b3717df mantine-Breadcrumbs-root"><a style="--text-fz:var(--mantine-font-size-sm);--text-lh:var(--mantine-line-height-sm);white-space:normal;color:inherit" class="mantine-focus-auto m_849cf0da m_f678d540 mantine-Breadcrumbs-breadcrumb m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-size="sm" data-underline="hover" href="/"><div style="color:inherit" class="m_4451eb3a mantine-Center-root"><svg xmlns="http://www.w3.org/2000/svg" width="15" height="15" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="tabler-icon tabler-icon-home-link "><path d="M20.085 11.085l-8.085 -8.085l-9 9h2v7a2 2 0 0 0 2 2h4.5"></path><path d="M9 21v-6a2 2 0 0 1 2 -2h2a2 2 0 0 1 1.807 1.143"></path><path d="M20 21a1 1 0 1 0 2 0a1 1 0 1 0 -2 0"></path><path d="M20 16a1 1 0 1 0 2 0a1 1 0 1 0 -2 0"></path><path d="M15 19a1 1 0 1 0 2 0a1 1 0 1 0 -2 0"></path><path d="M21 16l-5 3l5 2"></path></svg></div></a><div class="m_3b8f2208 mantine-Breadcrumbs-separator">/</div><a style="--text-fz:var(--mantine-font-size-sm);--text-lh:var(--mantine-line-height-sm);white-space:normal;color:inherit" class="mantine-focus-auto m_849cf0da m_f678d540 mantine-Breadcrumbs-breadcrumb m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-size="sm" data-underline="hover" href="/blog">Блог Хекслета</a><div class="m_3b8f2208 mantine-Breadcrumbs-separator">/</div><a style="--text-fz:var(--mantine-font-size-sm);--text-lh:var(--mantine-line-height-sm);white-space:normal;color:inherit" class="mantine-focus-auto m_849cf0da m_f678d540 mantine-Breadcrumbs-breadcrumb m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-size="sm" data-underline="hover" href="/blog/categories/code">Код</a><div class="m_3b8f2208 mantine-Breadcrumbs-separator">/</div><p style="--text-fz:var(--mantine-font-size-sm);--text-lh:var(--mantine-line-height-sm);white-space:normal;color:var(--mantine-color-dimmed)" class="mantine-focus-auto m_f678d540 mantine-Breadcrumbs-breadcrumb m_b6d8b162 mantine-Text-root" data-size="sm">Что такое время</p></div><style data-mantine-styles="inline">.__m__-_R_eub_{margin-bottom:var(--mantine-spacing-xs);}@media(min-width: 36em){.__m__-_R_eub_{margin-bottom:var(--mantine-spacing-xs);}}</style><div style="--group-gap:var(--mantine-spacing-md);--group-align:center;--group-justify:space-between;--group-wrap:wrap" class="m_4081bf90 mantine-Group-root __m__-_R_eub_"><style data-mantine-styles="inline">.__m__-_R_deub_{width:100%;}@media(min-width: 36em){.__m__-_R_deub_{width:70%;}}@media(min-width: 75em){.__m__-_R_deub_{width:75%;}}</style><div class="__m__-_R_deub_"><div style="--group-gap:var(--mantine-spacing-md);--group-align:center;--group-justify:start;--group-wrap:nowrap" class="m_4081bf90 mantine-Group-root"><h1 style="--title-fw:var(--mantine-h1-font-weight);--title-lh:var(--mantine-h1-line-height);--title-fz:var(--mantine-h1-font-size)" class="m_8a5d1357 mantine-Title-root" data-order="1">Что такое время</h1></div></div></div><div style="position:absolute;top:calc(18.75rem * var(--mantine-scale))" class=""></div><style data-mantine-styles="inline">.__m__-_R_2iub_{--grid-gutter:var(--mantine-spacing-xl);}</style><div class="m_410352e9 mantine-Grid-root __m__-_R_2iub_"><div class="m_dee7bd2f mantine-Grid-inner"><style data-mantine-styles="inline">.__m__-_R_dmiub_{--col-flex-grow:auto;--col-flex-basis:100%;--col-max-width:100%;}@media(min-width: 48em){.__m__-_R_dmiub_{--col-flex-grow:auto;--col-flex-basis:83.33333333333334%;--col-max-width:83.33333333333334%;}}@media(min-width: 62em){.__m__-_R_dmiub_{--col-flex-grow:auto;--col-flex-basis:66.66666666666667%;--col-max-width:66.66666666666667%;}}</style><div class="m_96bdd299 mantine-Grid-col __m__-_R_dmiub_"><div style="--stack-gap:var(--mantine-spacing-md);--stack-align:stretch;--stack-justify:flex-start;margin-bottom:var(--mantine-spacing-xl)" class="m_6d731127 mantine-Stack-root"><div class=""><div style="--group-gap:var(--mantine-spacing-xs);--group-align:center;--group-justify:flex-start;--group-wrap:wrap;margin-bottom:var(--mantine-spacing-xl)" class="m_4081bf90 mantine-Group-root"><button style="--badge-height:var(--badge-height-sm);--badge-padding-x:var(--badge-padding-x-sm);--badge-fz:var(--badge-fz-sm);--badge-bg:var(--mantine-color-default);--badge-color:var(--mantine-color-default-color);--badge-bd:calc(0.0625rem * var(--mantine-scale)) solid var(--mantine-color-default-border);cursor:pointer;color:inherit" class="m_347db0ec mantine-Badge-root" data-variant="default" data-size="sm" type="button" aria-label="Без стека"><span class="m_5add502a mantine-Badge-label">Без стека</span></button></div><div style="--group-gap:calc(0.625rem * var(--mantine-scale));--group-align:center;--group-justify:flex-start;--group-wrap:wrap;margin-bottom:var(--mantine-spacing-sm);color:var(--mantine-color-gray-text)" class="m_4081bf90 mantine-Group-root"><div style="--group-gap:calc(0.1875rem * var(--mantine-scale));--group-align:center;--group-justify:flex-start;--group-wrap:wrap;margin-inline-end:var(--mantine-spacing-lg)" class="m_4081bf90 mantine-Group-root">18 декабря 2024 г.</div><div style="--group-gap:calc(0.1875rem * var(--mantine-scale));--group-align:center;--group-justify:flex-start;--group-wrap:wrap" class="m_4081bf90 mantine-Group-root"><div style="--ti-size:var(--ti-size-xs);--ti-bg:transparent;--ti-color:var(--mantine-color-indigo-light-color);--ti-bd:calc(0.0625rem * var(--mantine-scale)) solid transparent;color:inherit" class="m_7341320d mantine-ThemeIcon-root" data-variant="transparent" data-size="xs"><svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="tabler-icon tabler-icon-clock "><path d="M3 12a9 9 0 1 0 18 0a9 9 0 0 0 -18 0"></path><path d="M12 7v5l3 3"></path></svg></div>10 минут</div><div style="--group-gap:calc(0.1875rem * var(--mantine-scale));--group-align:center;--group-justify:flex-start;--group-wrap:wrap" class="m_4081bf90 mantine-Group-root"><div style="--ti-size:var(--ti-size-xs);--ti-bg:transparent;--ti-color:var(--mantine-color-indigo-light-color);--ti-bd:calc(0.0625rem * var(--mantine-scale)) solid transparent;color:inherit" class="m_7341320d mantine-ThemeIcon-root" data-variant="transparent" data-size="xs"><svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="tabler-icon tabler-icon-thumb-up "><path d="M7 11v8a1 1 0 0 1 -1 1h-2a1 1 0 0 1 -1 -1v-7a1 1 0 0 1 1 -1h3a4 4 0 0 0 4 -4v-1a2 2 0 0 1 4 0v5h3a2 2 0 0 1 2 2l-1 5a2 3 0 0 1 -2 2h-7a3 3 0 0 1 -3 -3"></path></svg></div></div></div><div style="--ar-ratio:2" class="m_71ac47fc mantine-AspectRatio-root"><img style="--image-radius:var(--mantine-radius-md);--image-object-fit:cover;width:100%;height:100%" class="m_9e117634 mantine-Image-root" src="https://hexlet.io/rails/active_storage/representations/proxy/eyJfcmFpbHMiOnsiZGF0YSI6NDIzLCJwdXIiOiJibG9iX2lkIn19--5c6c5168a4dcfa6bb0bed8ae50c0877c3db687e3/eyJfcmFpbHMiOnsiZGF0YSI6eyJmb3JtYXQiOiJ3ZWJwIiwicmVzaXplX2FuZF9wYWQiOlsxNDU2LDcyOF0sInNhdmVyIjp7InF1YWxpdHkiOjg1fX0sInB1ciI6InZhcmlhdGlvbiJ9fQ==--b2ecf6121248a24883cda19469732b47b57f4e80/time.png" alt="Что такое время"/></div></div><div role="link" tabindex="0" style="cursor:pointer"><button style="display:block;width:100%" class="mantine-focus-auto m_87cf2631 mantine-UnstyledButton-root" type="button" aria-label="Присоединяйтесь к нашему Telegram-сообществу"><div style="background-color:light-dark(var(--mantine-color-gray-1), var(--mantine-color-dark-6))" class="m_e615b15f mantine-Card-root m_1b7284a3 mantine-Paper-root"><div style="--group-gap:var(--mantine-spacing-md);--group-align:center;--group-justify:flex-start;--group-wrap:wrap" class="m_4081bf90 mantine-Group-root"><div style="--ti-bg:transparent;--ti-color:var(--mantine-color-indigo-light-color);--ti-bd:calc(0.0625rem * var(--mantine-scale)) solid transparent;margin-inline-end:auto;color:inherit" class="m_7341320d mantine-ThemeIcon-root" data-variant="transparent"><svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="1.2" stroke-linecap="round" stroke-linejoin="round" class="tabler-icon tabler-icon-brand-telegram "><path d="M15 10l-4 4l6 6l4 -16l-18 7l4 2l2 6l3 -4"></path></svg></div>Присоединяйтесь к нашему Telegram-сообществу</div></div></button></div><div style="margin-bottom:var(--mantine-spacing-xl)" class="m_d08caa0 mantine-Typography-root"><h2 id="heading-2-1">Как компьютер определяет человеческое время?</h2>
<p>Представим ситуацию: Вы нашли в кладовой старый компьютер и решили попробовать его включить. Компьютер запускается, хотя стоял без дела 10 лет, но есть проблема: система уверена, что сегодня 1 января 1970 года. Эта ситуация поможет разобраться, как компьютеры хранят текущее время: ведь у них нет внутри пружин или маятников, как в часах, которые могли бы отсчитывать секунды. Каким образом они показывают верное время, если отключить их на несколько часов, и почему тот же принцип не работает, если компьютер простоял в кладовке 10 лет?</p>
<style data-mantine-styles="inline">.__m__-_R_5derddmiub_{--carousel-slide-gap:var(--mantine-spacing-xs);--carousel-slide-size:70%;}@media(min-width: 36em){.__m__-_R_5derddmiub_{--carousel-slide-gap:var(--mantine-spacing-xl);--carousel-slide-size:50%;}}</style><div style="--carousel-control-size:calc(2.5rem * var(--mantine-scale));--carousel-controls-offset:var(--mantine-spacing-sm);margin-bottom:var(--mantine-spacing-lg);padding-block:var(--mantine-spacing-sm);background:var(--app-color-surface)" class="m_17884d0f mantine-Carousel-root responsiveClassName" data-orientation="horizontal" data-include-gap-in-size="true"><div class="m_39bc3463 mantine-Carousel-controls" data-orientation="horizontal"><button class="mantine-focus-auto m_64f58e10 mantine-Carousel-control m_87cf2631 mantine-UnstyledButton-root" type="button" data-inactive="true" data-type="previous" tabindex="-1"><svg viewBox="0 0 15 15" fill="none" xmlns="http://www.w3.org/2000/svg" style="transform:rotate(90deg);width:calc(1rem * var(--mantine-scale));height:calc(1rem * var(--mantine-scale));display:block"><path d="M3.13523 6.15803C3.3241 5.95657 3.64052 5.94637 3.84197 6.13523L7.5 9.56464L11.158 6.13523C11.3595 5.94637 11.6759 5.95657 11.8648 6.15803C12.0536 6.35949 12.0434 6.67591 11.842 6.86477L7.84197 10.6148C7.64964 10.7951 7.35036 10.7951 7.15803 10.6148L3.15803 6.86477C2.95657 6.67591 2.94637 6.35949 3.13523 6.15803Z" fill="currentColor" fill-rule="evenodd" clip-rule="evenodd"></path></svg></button><button class="mantine-focus-auto m_64f58e10 mantine-Carousel-control m_87cf2631 mantine-UnstyledButton-root" type="button" data-inactive="true" data-type="next" tabindex="-1"><svg viewBox="0 0 15 15" fill="none" xmlns="http://www.w3.org/2000/svg" style="transform:rotate(-90deg);width:calc(1rem * var(--mantine-scale));height:calc(1rem * var(--mantine-scale));display:block"><path d="M3.13523 6.15803C3.3241 5.95657 3.64052 5.94637 3.84197 6.13523L7.5 9.56464L11.158 6.13523C11.3595 5.94637 11.6759 5.95657 11.8648 6.15803C12.0536 6.35949 12.0434 6.67591 11.842 6.86477L7.84197 10.6148C7.64964 10.7951 7.35036 10.7951 7.15803 10.6148L3.15803 6.86477C2.95657 6.67591 2.94637 6.35949 3.13523 6.15803Z" fill="currentColor" fill-rule="evenodd" clip-rule="evenodd"></path></svg></button></div><div class="m_a2dae653 mantine-Carousel-viewport" data-type="media"><div class="m_fcd81474 mantine-Carousel-container __m__-_R_5derddmiub_" data-orientation="horizontal"><div class="m_d98df724 mantine-Carousel-slide" data-orientation="horizontal"><div tabindex="0" style="cursor:pointer;height:100%"><a style="text-decoration:none" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="/programs/algorithms?promo_name=programs_list&promo_position=blog_post&promo_creative=catalog_card&promo_type=card" target="_blank"><div style="height:100%" class="m_e615b15f mantine-Card-root m_1b7284a3 mantine-Paper-root" data-with-border="true"><div style="--group-gap:calc(0.25rem * var(--mantine-scale));--group-align:center;--group-justify:flex-start;--group-wrap:nowrap" class="m_4081bf90 mantine-Group-root"><span style="font-size:var(--mantine-font-size-sm)" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">2 месяца</span><span class="mantine-focus-auto m_b6d8b162 mantine-Text-root">·</span><span style="font-size:var(--mantine-font-size-sm)" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">Для продвинутых</span></div><p style="margin-bottom:var(--mantine-spacing-sm);font-size:var(--mantine-font-size-h5);font-weight:bold" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">Алгоритмы и структуры данных</p><p class="mantine-focus-auto m_b6d8b162 mantine-Text-root">Алгоритмы для собеседований</p><div style="margin-top:auto" class=""><div class="m_4451eb3a mantine-Center-root"><img style="opacity:0.8;width:70%" class="m_9e117634 mantine-Image-root mantine-visible-from-xs" src="https://hexlet.io/rails/active_storage/representations/proxy/eyJfcmFpbHMiOnsiZGF0YSI6NDAyOCwicHVyIjoiYmxvYl9pZCJ9fQ==--ae9eed98663dd1201759d042a5ba7ca790866156/eyJfcmFpbHMiOnsiZGF0YSI6eyJmb3JtYXQiOiJ3ZWJwIiwicmVzaXplX3RvX2xpbWl0IjpbNDAwLDQwMF0sInNhdmVyIjp7InF1YWxpdHkiOjg1fX0sInB1ciI6InZhcmlhdGlvbiJ9fQ==--5b6f46dacd1af664f27558553a58076185091823/Programming-bro.png" alt="Алгоритмы и структуры данных" loading="eager"/></div><div style="--group-gap:var(--mantine-spacing-md);--group-align:end;--group-justify:space-between;--group-wrap:wrap;margin-top:var(--mantine-spacing-xs)" class="m_4081bf90 mantine-Group-root"><p style="font-size:var(--mantine-font-size-xl)" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">от 3 900 ₽</p><p style="font-size:var(--mantine-font-size-sm)" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">Посмотреть →</p></div></div></div></a></div></div><div class="m_d98df724 mantine-Carousel-slide" data-orientation="horizontal"><div tabindex="0" style="cursor:pointer;height:100%"><a style="text-decoration:none" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="/programs/discrete-mathematics?promo_name=programs_list&promo_position=blog_post&promo_creative=catalog_card&promo_type=card" target="_blank"><div style="height:100%" class="m_e615b15f mantine-Card-root m_1b7284a3 mantine-Paper-root" data-with-border="true"><div style="--group-gap:calc(0.25rem * var(--mantine-scale));--group-align:center;--group-justify:flex-start;--group-wrap:nowrap" class="m_4081bf90 mantine-Group-root"><span style="font-size:var(--mantine-font-size-sm)" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">1 месяц</span><span class="mantine-focus-auto m_b6d8b162 mantine-Text-root">·</span><span style="font-size:var(--mantine-font-size-sm)" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">С нуля</span></div><p style="margin-bottom:var(--mantine-spacing-sm);font-size:var(--mantine-font-size-h5);font-weight:bold" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">Дискретная математика</p><p class="mantine-focus-auto m_b6d8b162 mantine-Text-root">Дискретная математика для программистов</p><div style="margin-top:auto" class=""><div class="m_4451eb3a mantine-Center-root"><img style="opacity:0.8;width:70%" class="m_9e117634 mantine-Image-root mantine-visible-from-xs" src="https://hexlet.io/rails/active_storage/representations/proxy/eyJfcmFpbHMiOnsiZGF0YSI6MzY2MSwicHVyIjoiYmxvYl9pZCJ9fQ==--e9c2b6bde361adaac625a7f47d8b9671c17f3ddb/eyJfcmFpbHMiOnsiZGF0YSI6eyJmb3JtYXQiOiJ3ZWJwIiwicmVzaXplX3RvX2xpbWl0IjpbNDAwLDQwMF0sInNhdmVyIjp7InF1YWxpdHkiOjg1fX0sInB1ciI6InZhcmlhdGlvbiJ9fQ==--5b6f46dacd1af664f27558553a58076185091823/Mathematics-bro.png" alt="Дискретная математика" loading="eager"/></div><div style="--group-gap:var(--mantine-spacing-md);--group-align:end;--group-justify:space-between;--group-wrap:wrap;margin-top:var(--mantine-spacing-xs)" class="m_4081bf90 mantine-Group-root"><p style="font-size:var(--mantine-font-size-xl)" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">от 3 900 ₽</p><p style="font-size:var(--mantine-font-size-sm)" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">Посмотреть →</p></div></div></div></a></div></div><div class="m_d98df724 mantine-Carousel-slide" data-orientation="horizontal"><div tabindex="0" style="cursor:pointer;height:100%"><a style="text-decoration:none" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="/programs/python-sicp?promo_name=programs_list&promo_position=blog_post&promo_creative=catalog_card&promo_type=card" target="_blank"><div style="height:100%" class="m_e615b15f mantine-Card-root m_1b7284a3 mantine-Paper-root" data-with-border="true"><div style="--group-gap:calc(0.25rem * var(--mantine-scale));--group-align:center;--group-justify:flex-start;--group-wrap:nowrap" class="m_4081bf90 mantine-Group-root"><span style="font-size:var(--mantine-font-size-sm)" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">1 месяц</span><span class="mantine-focus-auto m_b6d8b162 mantine-Text-root">·</span><span style="font-size:var(--mantine-font-size-sm)" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">Для продвинутых</span></div><p style="margin-bottom:var(--mantine-spacing-sm);font-size:var(--mantine-font-size-h5);font-weight:bold" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">СИКП на Python</p><p class="mantine-focus-auto m_b6d8b162 mantine-Text-root">Изучите Python на глубоком уровне для решения сложных задач</p><div style="margin-top:auto" class=""><div class="m_4451eb3a mantine-Center-root"><img style="opacity:0.8;width:70%" class="m_9e117634 mantine-Image-root mantine-visible-from-xs" src="https://hexlet.io/rails/active_storage/representations/proxy/eyJfcmFpbHMiOnsiZGF0YSI6Mzc1OCwicHVyIjoiYmxvYl9pZCJ9fQ==--023ea18f500b1c4c91617fa96bbc52df8395da39/eyJfcmFpbHMiOnsiZGF0YSI6eyJmb3JtYXQiOiJ3ZWJwIiwicmVzaXplX3RvX2xpbWl0IjpbNDAwLDQwMF0sInNhdmVyIjp7InF1YWxpdHkiOjg1fX0sInB1ciI6InZhcmlhdGlvbiJ9fQ==--5b6f46dacd1af664f27558553a58076185091823/Software%20engineer-bro.png" alt="СИКП на Python" loading="eager"/></div><div style="--group-gap:var(--mantine-spacing-md);--group-align:end;--group-justify:space-between;--group-wrap:wrap;margin-top:var(--mantine-spacing-xs)" class="m_4081bf90 mantine-Group-root"><p style="font-size:var(--mantine-font-size-xl)" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">от 3 900 ₽</p><p style="font-size:var(--mantine-font-size-sm)" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">Посмотреть →</p></div></div></div></a></div></div><div class="m_d98df724 mantine-Carousel-slide" data-orientation="horizontal"><div tabindex="0" style="cursor:pointer;height:100%"><a style="text-decoration:none" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="/programs/java-sicp?promo_name=programs_list&promo_position=blog_post&promo_creative=catalog_card&promo_type=card" target="_blank"><div style="height:100%" class="m_e615b15f mantine-Card-root m_1b7284a3 mantine-Paper-root" data-with-border="true"><div style="--group-gap:calc(0.25rem * var(--mantine-scale));--group-align:center;--group-justify:flex-start;--group-wrap:nowrap" class="m_4081bf90 mantine-Group-root"><span style="font-size:var(--mantine-font-size-sm)" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">1 месяц</span><span class="mantine-focus-auto m_b6d8b162 mantine-Text-root">·</span><span style="font-size:var(--mantine-font-size-sm)" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">Для продвинутых</span></div><p style="margin-bottom:var(--mantine-spacing-sm);font-size:var(--mantine-font-size-h5);font-weight:bold" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">СИКП на Java</p><p class="mantine-focus-auto m_b6d8b162 mantine-Text-root">Изучите фундаментальные принципы программирования на Java</p><div style="margin-top:auto" class=""><div class="m_4451eb3a mantine-Center-root"><img style="opacity:0.8;width:70%" class="m_9e117634 mantine-Image-root mantine-visible-from-xs" src="https://hexlet.io/rails/active_storage/representations/proxy/eyJfcmFpbHMiOnsiZGF0YSI6NDAxNiwicHVyIjoiYmxvYl9pZCJ9fQ==--eb66b9b5e26fafa32844ce0f4522c3ed84544040/eyJfcmFpbHMiOnsiZGF0YSI6eyJmb3JtYXQiOiJ3ZWJwIiwicmVzaXplX3RvX2xpbWl0IjpbNDAwLDQwMF0sInNhdmVyIjp7InF1YWxpdHkiOjg1fX0sInB1ciI6InZhcmlhdGlvbiJ9fQ==--5b6f46dacd1af664f27558553a58076185091823/Hand%20coding-rafiki.png" alt="СИКП на Java" loading="eager"/></div><div style="--group-gap:var(--mantine-spacing-md);--group-align:end;--group-justify:space-between;--group-wrap:wrap;margin-top:var(--mantine-spacing-xs)" class="m_4081bf90 mantine-Group-root"><p style="font-size:var(--mantine-font-size-xl)" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">от 3 900 ₽</p><p style="font-size:var(--mantine-font-size-sm)" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">Посмотреть →</p></div></div></div></a></div></div><div class="m_d98df724 mantine-Carousel-slide" data-orientation="horizontal"><div tabindex="0" style="cursor:pointer;height:100%"><a style="text-decoration:none" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="/programs/linear-algebra?promo_name=programs_list&promo_position=blog_post&promo_creative=catalog_card&promo_type=card" target="_blank"><div style="height:100%" class="m_e615b15f mantine-Card-root m_1b7284a3 mantine-Paper-root" data-with-border="true"><div style="--group-gap:calc(0.25rem * var(--mantine-scale));--group-align:center;--group-justify:flex-start;--group-wrap:nowrap" class="m_4081bf90 mantine-Group-root"><span style="font-size:var(--mantine-font-size-sm)" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">1 месяц</span><span class="mantine-focus-auto m_b6d8b162 mantine-Text-root">·</span><span style="font-size:var(--mantine-font-size-sm)" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">С нуля</span></div><p style="margin-bottom:var(--mantine-spacing-sm);font-size:var(--mantine-font-size-h5);font-weight:bold" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">Линейная алгебра</p><p class="mantine-focus-auto m_b6d8b162 mantine-Text-root">Освоите работу с векторами, матрицами и системами уравнений</p><div style="margin-top:auto" class=""><div class="m_4451eb3a mantine-Center-root"><img style="opacity:0.8;width:70%" class="m_9e117634 mantine-Image-root mantine-visible-from-xs" src="https://hexlet.io/rails/active_storage/representations/proxy/eyJfcmFpbHMiOnsiZGF0YSI6MzY0NywicHVyIjoiYmxvYl9pZCJ9fQ==--fd23784224c1dfaefc6e1cd556a43d164e3c412f/eyJfcmFpbHMiOnsiZGF0YSI6eyJmb3JtYXQiOiJ3ZWJwIiwicmVzaXplX3RvX2xpbWl0IjpbNDAwLDQwMF0sInNhdmVyIjp7InF1YWxpdHkiOjg1fX0sInB1ciI6InZhcmlhdGlvbiJ9fQ==--5b6f46dacd1af664f27558553a58076185091823/Mathematics-amico.png" alt="Линейная алгебра" loading="eager"/></div><div style="--group-gap:var(--mantine-spacing-md);--group-align:end;--group-justify:space-between;--group-wrap:wrap;margin-top:var(--mantine-spacing-xs)" class="m_4081bf90 mantine-Group-root"><p style="font-size:var(--mantine-font-size-xl)" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">от 3 900 ₽</p><p style="font-size:var(--mantine-font-size-sm)" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">Посмотреть →</p></div></div></div></a></div></div><div class="m_d98df724 mantine-Carousel-slide" data-orientation="horizontal"><div tabindex="0" style="cursor:pointer;height:100%"><a style="text-decoration:none" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="/courses?promo_name=programs_list&promo_position=blog_post&promo_creative=catalog_card&promo_type=card"><div style="height:100%" class="m_e615b15f mantine-Card-root m_1b7284a3 mantine-Paper-root" data-with-border="true"><h2 style="--title-fw:var(--mantine-h2-font-weight);--title-lh:var(--mantine-h2-line-height);--title-fz:var(--mantine-h2-font-size);margin-bottom:var(--mantine-spacing-md);font-size:var(--mantine-font-size-h3)" class="m_8a5d1357 mantine-Title-root" data-order="2" data-responsive="true">Каталог</h2><p style="margin-bottom:auto" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">Полный список доступных курсов по разным направлениям</p><div style="margin-top:auto" class=""><div class="m_4451eb3a mantine-Center-root"><img style="opacity:0.8;width:70%" class="m_9e117634 mantine-Image-root mantine-visible-from-xs" src="/vite/assets/development-BVihs_d5.png" alt="Orientation"/></div></div></div></a></div></div></div></div></div>
<h3 id="heading-3-2">Маятник как первый секундомер</h3>
<p>Прежде, чем ответить на эти вопросы, разберемся, как измеряется точное время. Это не такая простая задача, если под рукой нет часов. Мы можем отсчитывать дни, месяцы и годы или приблизительно разбить день на часы. Но найти ориентир, который подскажет количество прошедших секунд, гораздо сложнее.</p>
<p>Принято считать, что первым эту задачу в XVII веке решил итальянский ученый [Галилео Галилей] (<a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="https://ru.wikipedia.org/wiki/%D0%93%D0%B0%D0%BB%D0%B8%D0%BB%D0%B5%D0%B9,_%D0%93%D0%B0%D0%BB%D0%B8%D0%BB%D0%B5%D0%BE" rel="noopener noreferrer" target="_blank">https://ru.wikipedia.org/wiki/%D0%93%D0%B0%D0%BB%D0%B8%D0%BB%D0%B5%D0%B9,_%D0%93%D0%B0%D0%BB%D0%B8%D0%BB%D0%B5%D0%BE</a>). Галилео нужно было точно знать, сколько длился тот или иной эксперимент. Его биографы
утверждают, что сначала он считал количество своих сердцебиений, но однажды наблюдал за раскачивающимся канделябром в кафедральном соборе Пизы и заметил, что каждое колебание канделябра занимает одинаковое количество сердцебиений. Это число не меняется при затухании амплитуды движений. Так он понял, что
колебания маятника помогают точно измерять время.</p>
<h3 id="heading-3-3">Можно ли использовать колебания для подсчета времени в компьютере?</h3>
<p>Когда в 70-е-80-е годы инженеры впервые столкнулись с задачей отображения человеческого времени в компьютере, они, как ни странно, обратились к трудам итальянского ученого, но взглянули на них под другим углом. Маятник — не единственный возможный источник колебаний. У любого современного персонального компьютера есть «сердце» — центральное процессорное устройство (ЦПУ), или просто процессор. Он устроен очень сложно и выполняет множество функций, но в данном случае интересно одно из составляющих его устройств. Когда через него проходит ток, оно испускает равномерные электрические колебания с одинаковой <a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="https://en.wikipedia.org/wiki/Clock_rate" rel="noopener noreferrer" target="_blank">частотой</a>. Частота этих колебаний измеряется в герцах (Гц) и определяет количество операций, которое процессор может выполнить за одну секунду. К сожалению, если мы будем использовать единственную доступную нам операцию в секунду для увеличения счетчика колебаний, то наш процессор будет совершенно непригоден для чего-то ещё: ему будет некогда исполнять другие команды. Конечно, современные процессоры не такие медленные, их частота измеряется в гигагерцах (ГГц), то есть в 1 000 000 000 операций в секунду. Но даже при такой мощности для измерения времени пришлось бы считать каждое колебание процессора. Для решения проблемы был придуман кварцевый генератор. Он представляет собой тончайшую кремниевую пластину, которая под воздействием электрического тока равномерно расширяется и сжимается, <a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="https://ru.wikipedia.org/wiki/%D0%9F%D1%8C%D0%B5%D0%B7%D0%BE%D1%8D%D0%BB%D0%B5%D0%BA%D1%82%D1%80%D0%B8%D1%87%D0%B5%D1%81%D0%BA%D0%B8%D0%B9_%D1%8D%D1%84%D1%84%D0%B5%D0%BA%D1%82" rel="noopener noreferrer" target="_blank">генерируя слабый электрический заряд на поверхности</a>. Таким образом, механические колебания пластины сопровождаются синхронными, равномерными колебаниями электрического заряда.</p>
<p><img style="--image-object-fit:contain;width:auto" class="m_9e117634 mantine-Image-root" src="https://imgur.com/ll9vqr4.png" alt="oscillator" loading="lazy"/></p>
<p>Полученные равномерные колебания принято называть таймером — они позволяют процессору синхронизировать производимые им операции во времени. Таким образом, у компьютера есть своё собственное понимание времени.</p>
<style data-mantine-styles="inline">.__m__-_R_lderddmiub_{--carousel-slide-gap:var(--mantine-spacing-xs);--carousel-slide-size:80%;}@media(min-width: 36em){.__m__-_R_lderddmiub_{--carousel-slide-gap:var(--mantine-spacing-xl);--carousel-slide-size:50%;}}</style><div style="--carousel-control-size:calc(2.5rem * var(--mantine-scale));--carousel-controls-offset:var(--mantine-spacing-sm);margin-bottom:var(--mantine-spacing-lg);padding-block:var(--mantine-spacing-sm);background:var(--app-color-surface)" class="m_17884d0f mantine-Carousel-root responsiveClassName" data-orientation="horizontal" data-include-gap-in-size="true"><div class="m_39bc3463 mantine-Carousel-controls" data-orientation="horizontal"><button class="mantine-focus-auto m_64f58e10 mantine-Carousel-control m_87cf2631 mantine-UnstyledButton-root" type="button" data-inactive="true" data-type="previous" tabindex="-1"><svg viewBox="0 0 15 15" fill="none" xmlns="http://www.w3.org/2000/svg" style="transform:rotate(90deg);width:calc(1rem * var(--mantine-scale));height:calc(1rem * var(--mantine-scale));display:block"><path d="M3.13523 6.15803C3.3241 5.95657 3.64052 5.94637 3.84197 6.13523L7.5 9.56464L11.158 6.13523C11.3595 5.94637 11.6759 5.95657 11.8648 6.15803C12.0536 6.35949 12.0434 6.67591 11.842 6.86477L7.84197 10.6148C7.64964 10.7951 7.35036 10.7951 7.15803 10.6148L3.15803 6.86477C2.95657 6.67591 2.94637 6.35949 3.13523 6.15803Z" fill="currentColor" fill-rule="evenodd" clip-rule="evenodd"></path></svg></button><button class="mantine-focus-auto m_64f58e10 mantine-Carousel-control m_87cf2631 mantine-UnstyledButton-root" type="button" data-inactive="true" data-type="next" tabindex="-1"><svg viewBox="0 0 15 15" fill="none" xmlns="http://www.w3.org/2000/svg" style="transform:rotate(-90deg);width:calc(1rem * var(--mantine-scale));height:calc(1rem * var(--mantine-scale));display:block"><path d="M3.13523 6.15803C3.3241 5.95657 3.64052 5.94637 3.84197 6.13523L7.5 9.56464L11.158 6.13523C11.3595 5.94637 11.6759 5.95657 11.8648 6.15803C12.0536 6.35949 12.0434 6.67591 11.842 6.86477L7.84197 10.6148C7.64964 10.7951 7.35036 10.7951 7.15803 10.6148L3.15803 6.86477C2.95657 6.67591 2.94637 6.35949 3.13523 6.15803Z" fill="currentColor" fill-rule="evenodd" clip-rule="evenodd"></path></svg></button></div><div class="m_a2dae653 mantine-Carousel-viewport" data-type="media"><div class="m_fcd81474 mantine-Carousel-container __m__-_R_lderddmiub_" data-orientation="horizontal"><div class="m_d98df724 mantine-Carousel-slide" data-orientation="horizontal"><div tabindex="0" style="cursor:pointer;height:100%"><a style="text-decoration:none" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="/blog/posts/makefile-as-task-runner"><div style="padding-top:0rem;height:100%" class="m_e615b15f mantine-Card-root m_1b7284a3 mantine-Paper-root" data-with-border="true"><div style="margin-bottom:var(--mantine-spacing-sm)" class="m_599a2148 mantine-Card-section" data-first-section="true"><div style="--ar-ratio:2" class="m_71ac47fc mantine-AspectRatio-root"><img class="m_9e117634 mantine-Image-root" src="https://hexlet.io/rails/active_storage/representations/proxy/eyJfcmFpbHMiOnsiZGF0YSI6NDU5LCJwdXIiOiJibG9iX2lkIn19--97454b3b2dea7bca2f8da19e43e311b790c71130/eyJfcmFpbHMiOnsiZGF0YSI6eyJmb3JtYXQiOiJ3ZWJwIiwicmVzaXplX2FuZF9wYWQiOls3MDgsMzU0XSwic2F2ZXIiOnsicXVhbGl0eSI6ODV9fSwicHVyIjoidmFyaWF0aW9uIn19--324dc52aa55ebe818c2a887ebcb832b9ad1c0381/cover.jpg" loading="lazy" alt="Что такое Makefile и как начать его использовать"/></div></div><p style="margin-bottom:var(--mantine-spacing-xs);font-size:var(--mantine-font-size-lg);font-weight:bold" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">Что такое Makefile и как начать его использовать</p><p style="margin-bottom:auto" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">Этот гайд расскажет, как использование инструмента Makefile позволит свести процесс разворачивани...</p><div style="--group-gap:var(--mantine-spacing-md);--group-align:center;--group-justify:space-between;--group-wrap:wrap;margin-top:var(--mantine-spacing-lg);font-size:var(--mantine-font-size-sm)" class="m_4081bf90 mantine-Group-root">18 декабря 2024 г.<p style="font-size:inherit" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">Посмотреть →</p></div></div></a></div></div><div class="m_d98df724 mantine-Carousel-slide" data-orientation="horizontal"><div tabindex="0" style="cursor:pointer;height:100%"><a style="text-decoration:none" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="/blog/posts/rest-api"><div style="padding-top:0rem;height:100%" class="m_e615b15f mantine-Card-root m_1b7284a3 mantine-Paper-root" data-with-border="true"><div style="margin-bottom:var(--mantine-spacing-sm)" class="m_599a2148 mantine-Card-section" data-first-section="true"><div style="--ar-ratio:2" class="m_71ac47fc mantine-AspectRatio-root"><img class="m_9e117634 mantine-Image-root" src="https://hexlet.io/rails/active_storage/representations/proxy/eyJfcmFpbHMiOnsiZGF0YSI6NDI3LCJwdXIiOiJibG9iX2lkIn19--836a0585b4618daeee1ff29040ebb7a9c5aaf365/eyJfcmFpbHMiOnsiZGF0YSI6eyJmb3JtYXQiOiJ3ZWJwIiwicmVzaXplX2FuZF9wYWQiOls3MDgsMzU0XSwic2F2ZXIiOnsicXVhbGl0eSI6ODV9fSwicHVyIjoidmFyaWF0aW9uIn19--324dc52aa55ebe818c2a887ebcb832b9ad1c0381/%D0%A7%D1%82%D0%BE%D1%82%D0%B0%D0%BA%D0%BE%D0%B5RESTAPI_.png" loading="lazy" alt="Что такое REST API?"/></div></div><p style="margin-bottom:var(--mantine-spacing-xs);font-size:var(--mantine-font-size-lg);font-weight:bold" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">Что такое REST API?</p><p style="margin-bottom:auto" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">REST API применяется везде, где есть необходимость предоставления данных с сервера пользователю в...</p><div style="--group-gap:var(--mantine-spacing-md);--group-align:center;--group-justify:space-between;--group-wrap:wrap;margin-top:var(--mantine-spacing-lg);font-size:var(--mantine-font-size-sm)" class="m_4081bf90 mantine-Group-root">18 декабря 2024 г.<p style="font-size:inherit" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">Посмотреть →</p></div></div></a></div></div><div class="m_d98df724 mantine-Carousel-slide" data-orientation="horizontal"><div tabindex="0" style="cursor:pointer;height:100%"><a style="text-decoration:none" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="/blog/posts/hosting"><div style="padding-top:0rem;height:100%" class="m_e615b15f mantine-Card-root m_1b7284a3 mantine-Paper-root" data-with-border="true"><div style="margin-bottom:var(--mantine-spacing-sm)" class="m_599a2148 mantine-Card-section" data-first-section="true"><div style="--ar-ratio:2" class="m_71ac47fc mantine-AspectRatio-root"><img class="m_9e117634 mantine-Image-root" src="https://hexlet.io/rails/active_storage/representations/proxy/eyJfcmFpbHMiOnsiZGF0YSI6NDQ3LCJwdXIiOiJibG9iX2lkIn19--ffcfe2a0a0c94c3c08e708d19a79ea94dce9b54b/eyJfcmFpbHMiOnsiZGF0YSI6eyJmb3JtYXQiOiJ3ZWJwIiwicmVzaXplX2FuZF9wYWQiOls3MDgsMzU0XSwic2F2ZXIiOnsicXVhbGl0eSI6ODV9fSwicHVyIjoidmFyaWF0aW9uIn19--324dc52aa55ebe818c2a887ebcb832b9ad1c0381/servers.png" loading="lazy" alt="Что такое хостинг и домен сайта простыми словами"/></div></div><p style="margin-bottom:var(--mantine-spacing-xs);font-size:var(--mantine-font-size-lg);font-weight:bold" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">Что такое хостинг и домен сайта простыми словами</p><p style="margin-bottom:auto" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">Чтобы создать собственный сайт, вам понадобится домен, хостинг и курсы программирования. Рассказы...</p><div style="--group-gap:var(--mantine-spacing-md);--group-align:center;--group-justify:space-between;--group-wrap:wrap;margin-top:var(--mantine-spacing-lg);font-size:var(--mantine-font-size-sm)" class="m_4081bf90 mantine-Group-root">18 декабря 2024 г.<p style="font-size:inherit" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">Посмотреть →</p></div></div></a></div></div></div></div></div>
<h3 id="heading-3-4">От компьютерного времени к человеческому</h3>
<p>Получается, что внутри компьютера есть таймер, но он не измеряет человеческое время — оно для него довольно бесполезно. Как правило, эту задачу на себя берут операционные системы (ОС). Поскольку ОС знает, с какой частотой работает кварцевый генератор, она может измерить время, которое проходит между срабатываниями таймера (срабатывание таймера называют тик (tick) или джиффи (jiffy)). Если генератор
работает на частоте 100 Гц, то период между тиками равен 1/100 секунды или 10 миллисекундам. Операционная система создает в памяти переменную, которую обычно называют jiffies, и увеличивает её на единицу каждый раз, когда процессор дает сигнал о новом тике.</p>
<p>Соответственно, чтобы узнать, как долго включен компьютер, системе достаточно умножить размер периода между тиками на количество этих самых тиков. А чтобы узнать текущее время, нужно просто добавить прошедшее время к времени на момент старта системы. Но как узнать человеческое время на момент старта системы?</p>
<h3 id="heading-3-5">RTC или часы реального времени</h3>
<p>Первые персональные компьютеры (например, <a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="https://en.wikipedia.org/wiki/IBM_Personal_Computer#Design_process" rel="noopener noreferrer" target="_blank">IBM PC</a> или <a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="https://en.wikipedia.org/wiki/Apple_II" rel="noopener noreferrer" target="_blank">Apple II</a>), не умели следить за тем, сколько прошло время после выключения, а <a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="https://www.youtube.com/watch?v=X3aqJQPQKhs" rel="noopener noreferrer" target="_blank">спрашивали его на старте</a>. Для решения этой задачи снова пригодился кварцевый генератор. Устройству неважно, из какого источника получать электрический ток, — это натолкнуло разработчиков на мысль, что достаточно подключить генератор к обычной литиевой батарейке, чтобы получать равномерные электрические колебания.</p>
<p>Если в эту схему добавить бинарный счетчик, который увеличивается на каждое колебание, то мы получим устройство, которое может фиксировать человеческое время. Оно так и называется — RTC (Real Time Clock), или часы реального времени. Частота колебаний на выходе из RTC обычно 32 768 Гц или 2^15 Гц, что удобно использовать в бинарных счетчиках.</p>
<p><img style="--image-object-fit:contain;width:auto" class="m_9e117634 mantine-Image-root" src="https://imgur.com/11i2B4B.png" alt="RTC" loading="lazy"/></p>
<p>Старый компьютер, о котором шла речь в начале, перепутал время именно из-за того, что батарейка в его часах реального времени села: после запуска он не смог считать время из RTC и выставил стандартное стартовое время — 1 января 1970 года.</p>
<h3 id="heading-3-6">NTP или протокол сетевого времени</h3>
<p>RTC позволяет компьютеру отсчитывать миллисекунды, даже когда он выключен. Но стандартные RTC имеют погрешность 1,7-8,6 секунд в день — то есть за год они могут потерять целый час.</p>
<p>Иногда мы сталкиваемся с этой проблемой в реальной жизни: например, когда нам приходится вручную настраивать время на наручных кварцевых часах или на микроволновке, где тоже используют RTC. Но нам никогда не приходится делать этого на компьютере. Да и на старте компьютер больше не спрашивает текущее время. Современные компьютеры настраивают время через интернет — для этого используется NTP (протокол сетевого времени). Этот протокол даже учитывает время передачи данных между источником и компьютером и компенсирует его. В публичной сети погрешность составляет всего 10 мс.</p>
<h3 id="heading-3-7">Как протоколы работают вместе?</h3>
<p><img style="--image-object-fit:contain;width:auto" class="m_9e117634 mantine-Image-root" src="https://imgur.com/tW5eYQi.png" alt="computer timekeeping" loading="lazy"/></p>
<p>Когда наш компьютер выключен, RTC продолжает работу и отсчитывает время. Когда мы нажимаем кнопку запуска, операционная система забирает время из RTC и начинает отслеживать время самостоятельно, используя таймер процессора. Время от времени операционная система получает точное время по NTP и поправляет свой внутренний счетчик. Когда мы выключаем компьютер, за дело снова берется RTC.</p>
<h2 id="heading-2-8">Таймстампы и эпохи</h2>
<p>Как уже говорилось, для отслеживания человеческого времени операционная система создает в памяти компьютера переменную jiffy, в которой хранит количество тиков с момента старта системы. Но как с помощью неё показывать человеку календарь?</p>
<h3 id="heading-3-9">Unix-таймстамп и epoch</h3>
<p>В 70-е годы прошлого века эту проблему решили инженеры из Bell Labs при разработке операционной системы Unix (она заложила фундамент для появления современных Linux и MacOS). Они ввели в систему переменную, которая, начиная с заданной даты, увеличивается на каждый тик генератора — её называли epoch.</p>
<p>Под эту переменную отводилось целое число со знаком (signed integer) размером в 32 бита (то есть от −2 147 483 648 (-2^31) до 2 147 483 647 (2^31−1)). Подавляющее большинство генераторов на тот момент работали на частоте 60 Гц, то есть отсчитывали 60 тиков в секунду, поэтому в переменной хранилось 1/60 секунды и она могла представлять временной промежуток не более 829 дней.</p>
<p>В <a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="http://man.cat-v.org/unix-1st/2/sys-time" rel="noopener noreferrer" target="_blank">версии Unix от 1971 года</a> отсчет начинался с 1971-01-01 00:00</p><div></div>. <a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="https://minnie.tuhs.org/cgi-bin/utree.pl?file=V3/man/man2/time.2" rel="noopener noreferrer" target="_blank">На следующий год</a> с 1972-01-01 00:00<div></div>. Переводить время каждый год было довольно неудобно, поэтому в <a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="https://minnie.tuhs.org/cgi-bin/utree.pl?file=V4/man/man2/time.2" rel="noopener noreferrer" target="_blank">четвертой версии Unix</a> в 1973 году за epoch была взята дата 1970-01-01 00:00<div></div>, а в переменной стали хранить не 1/60 секунды, а полную секунду. Позже этот принцип стал международным стандартом и используется по сей день.<p></p>
<p>Если на Вашем компьютере установлена операционная система семейства Linux или MacOS, Вы можете увидеть текущий Unix-timestamp, введя в терминале:</p>
<div style="margin-bottom:var(--mantine-spacing-lg)" class="m_e597c321 mantine-CodeHighlight-codeHighlight" dir="ltr"><div class="m_be7e9c9c mantine-CodeHighlight-controls"><button style="--ai-bg:transparent;--ai-hover:transparent;--ai-color:inherit;--ai-bd:none" class="mantine-focus-auto mantine-active m_d498bab7 mantine-CodeHighlight-control m_8d3f4000 mantine-ActionIcon-root m_87cf2631 mantine-UnstyledButton-root" data-variant="none" type="button" aria-label="Copy code"><span class="m_8d3afb97 mantine-ActionIcon-icon"><svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24" stroke-width="2" stroke="currentColor" fill="none" stroke-linecap="round" stroke-linejoin="round"><path stroke="none" d="M0 0h24v24H0z" fill="none"></path><path d="M8 8m0 2a2 2 0 0 1 2 -2h8a2 2 0 0 1 2 2v8a2 2 0 0 1 -2 2h-8a2 2 0 0 1 -2 -2z"></path><path d="M16 8v-2a2 2 0 0 0 -2 -2h-8a2 2 0 0 0 -2 2v8a2 2 0 0 0 2 2h2"></path></svg></span></button></div><div style="--scrollarea-scrollbar-size:calc(0.25rem * var(--mantine-scale));--sa-corner-width:0px;--sa-corner-height:0px" class="m_f744fd40 mantine-CodeHighlight-scrollarea m_d57069b5 mantine-ScrollArea-root" dir="ltr"><div style="overflow-x:hidden;overflow-y:hidden;overscroll-behavior-inline:none" class="m_c0783ff9 mantine-ScrollArea-viewport" data-scrollbars="xy"><div class="m_b1336c6 mantine-ScrollArea-content"><pre class="m_2c47c4fd mantine-CodeHighlight-pre" style="padding:0"><code class="m_5caae6d3 mantine-CodeHighlight-code">date +"%s"</code></pre></div></div></div><button class="mantine-focus-auto m_c9378bc2 mantine-CodeHighlight-showCodeButton m_87cf2631 mantine-UnstyledButton-root" data-hidden="true" type="button">Expand code</button></div>
<p>Windows тоже считает время, отталкиваясь от конкретной даты, но использует для этого не абстрактный 1970 год, а 1601 — первый год Григорианского календаря.</p>
<h3 id="heading-3-10">Проблема 2038 года</h3>
<p>Поскольку время хранится в целочисленной 32-битовой переменной и представляет собой определенное количество секунд с определенного момента, то самое большое количество секунд, которое мы можем использовать — это 2 147 483 647 (2^31−1). Если прибавить это количество к epoch, то мы получим 19 января 2038 03</p><div></div>. Что произойдет с системой, когда наступит этот день и пройдет еще одна секунда?<p></p>
<p>Чтобы ответить на этот вопрос, рассмотрим особенность 32-битовых переменных:</p>
<p>2 147 483 647 в двоичной системе записывается как</p>
<div style="margin-bottom:var(--mantine-spacing-lg)" class="m_e597c321 mantine-CodeHighlight-codeHighlight" dir="ltr"><div class="m_be7e9c9c mantine-CodeHighlight-controls"><button style="--ai-bg:transparent;--ai-hover:transparent;--ai-color:inherit;--ai-bd:none" class="mantine-focus-auto mantine-active m_d498bab7 mantine-CodeHighlight-control m_8d3f4000 mantine-ActionIcon-root m_87cf2631 mantine-UnstyledButton-root" data-variant="none" type="button" aria-label="Copy code"><span class="m_8d3afb97 mantine-ActionIcon-icon"><svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24" stroke-width="2" stroke="currentColor" fill="none" stroke-linecap="round" stroke-linejoin="round"><path stroke="none" d="M0 0h24v24H0z" fill="none"></path><path d="M8 8m0 2a2 2 0 0 1 2 -2h8a2 2 0 0 1 2 2v8a2 2 0 0 1 -2 2h-8a2 2 0 0 1 -2 -2z"></path><path d="M16 8v-2a2 2 0 0 0 -2 -2h-8a2 2 0 0 0 -2 2v8a2 2 0 0 0 2 2h2"></path></svg></span></button></div><div style="--scrollarea-scrollbar-size:calc(0.25rem * var(--mantine-scale));--sa-corner-width:0px;--sa-corner-height:0px" class="m_f744fd40 mantine-CodeHighlight-scrollarea m_d57069b5 mantine-ScrollArea-root" dir="ltr"><div style="overflow-x:hidden;overflow-y:hidden;overscroll-behavior-inline:none" class="m_c0783ff9 mantine-ScrollArea-viewport" data-scrollbars="xy"><div class="m_b1336c6 mantine-ScrollArea-content"><pre class="m_2c47c4fd mantine-CodeHighlight-pre" style="padding:0"><code class="m_5caae6d3 mantine-CodeHighlight-code">01111111111111111111111111111111 // (1 ноль и 31 единица)</code></pre></div></div></div><button class="mantine-focus-auto m_c9378bc2 mantine-CodeHighlight-showCodeButton m_87cf2631 mantine-UnstyledButton-root" data-hidden="true" type="button">Expand code</button></div>
<p>Если это число увеличить на единицу, то оно превратится в</p>
<div style="margin-bottom:var(--mantine-spacing-lg)" class="m_e597c321 mantine-CodeHighlight-codeHighlight" dir="ltr"><div class="m_be7e9c9c mantine-CodeHighlight-controls"><button style="--ai-bg:transparent;--ai-hover:transparent;--ai-color:inherit;--ai-bd:none" class="mantine-focus-auto mantine-active m_d498bab7 mantine-CodeHighlight-control m_8d3f4000 mantine-ActionIcon-root m_87cf2631 mantine-UnstyledButton-root" data-variant="none" type="button" aria-label="Copy code"><span class="m_8d3afb97 mantine-ActionIcon-icon"><svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24" stroke-width="2" stroke="currentColor" fill="none" stroke-linecap="round" stroke-linejoin="round"><path stroke="none" d="M0 0h24v24H0z" fill="none"></path><path d="M8 8m0 2a2 2 0 0 1 2 -2h8a2 2 0 0 1 2 2v8a2 2 0 0 1 -2 2h-8a2 2 0 0 1 -2 -2z"></path><path d="M16 8v-2a2 2 0 0 0 -2 -2h-8a2 2 0 0 0 -2 2v8a2 2 0 0 0 2 2h2"></path></svg></span></button></div><div style="--scrollarea-scrollbar-size:calc(0.25rem * var(--mantine-scale));--sa-corner-width:0px;--sa-corner-height:0px" class="m_f744fd40 mantine-CodeHighlight-scrollarea m_d57069b5 mantine-ScrollArea-root" dir="ltr"><div style="overflow-x:hidden;overflow-y:hidden;overscroll-behavior-inline:none" class="m_c0783ff9 mantine-ScrollArea-viewport" data-scrollbars="xy"><div class="m_b1336c6 mantine-ScrollArea-content"><pre class="m_2c47c4fd mantine-CodeHighlight-pre" style="padding:0"><code class="m_5caae6d3 mantine-CodeHighlight-code">10000000000000000000000000000000 // (1 единица и 31 ноль)</code></pre></div></div></div><button class="mantine-focus-auto m_c9378bc2 mantine-CodeHighlight-showCodeButton m_87cf2631 mantine-UnstyledButton-root" data-hidden="true" type="button">Expand code</button></div>
<p>Первая цифра у таких чисел означает знак: <code style="margin-bottom:var(--mantine-spacing-lg)" class="m_dfe9c588 mantine-InlineCodeHighlight-inlineCodeHighlight m_e597c321 mantine-CodeHighlight-codeHighlight m_dfe9c588 mantine-InlineCodeHighlight-inlineCodeHighlight">+</code> или <code style="margin-bottom:var(--mantine-spacing-lg)" class="m_dfe9c588 mantine-InlineCodeHighlight-inlineCodeHighlight m_e597c321 mantine-CodeHighlight-codeHighlight m_dfe9c588 mantine-InlineCodeHighlight-inlineCodeHighlight">-</code>. А значит, что в десятичной системе оно будет равно −2 147 483 648, то есть число перейдет от самого большого положительного к самому маленькому отрицательному. И система начнет показывать дату, равную разности epoch и двух млрд секунд, то есть 13 декабря 1901 года. Но не стоит волноваться, проблема уже решена — большинство систем используют 64 битовые числа для хранения времени. Этого хватит, чтобы не столкнуться с проблемой до 15</p><div></div> 4 декабря 292 277 026 596 года.<p></p>
<h3 id="heading-3-11">Проблема 2000 года или Y2K bug</h3>
<blockquote>
<p>«640 килобайт памяти должно хватить кому угодно», — Билл Гейтс.</p>
</blockquote>
<p>На самом деле в истории уже была такая ситуация. В 60-70-е годы прошлого столетия, когда люди только начали писать программное обеспечение для компьютеров, платы памяти стоили дорого и большинство компьютеров обходились несколькими килобайтами. Ради экономии памяти программисты решили записывать даты в формате ДД.ММ.ГГ.</p>
<p>Какую проблему это порождало? Допустим, у нас есть человек по имени Боб, у которого дата рождения записана как 01.11.19. Он родился в 1919 году, и ему чуть больше 100 лет. Есть человек по имени Фред и он родился 02.11.19, но ему всего два года, потому что он родился в 2019 году. Сталкиваясь с таким форматом дат, человек может исходить из контекста, но компьютер на такое не способен.</p>
<p>Программисты, которые писали программы в 60-70-е годы даже не предполагали, что их код может дожить до 2000 года, поэтому использование двух цифр вместо четырех было нормальной оптимизацией. Но когда приближался 2000 год, а многие компании всё ещё использовали тот же формат дат, началась паника: никто не понимал, что произойдёт, когда 99 год сменится на 00. Тогда в мире ходили самого разного рода слухи:
например, что банкоматы в этот момент начнут плеваться деньгами, а самолёты начнут падать. Баг был вовремя замечен и проблему в большинстве систем удалось исправить вовремя.</p>
<h2 id="heading-2-12">Об особенностях работы с локальным временем и часовыми поясами</h2>
<h3 id="heading-3-13">Почему не стоит самостоятельно писать код, описывающий время</h3>
<p>Представим, что абстрактный программист пишет приложение, которое должно каждый день в 12 часов дня отправлять пользователю уведомление, что наступил полдень. Программист живет в Лондоне и оно будет использоваться только там. Писать код он начал зимой, поэтому разницы с <a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="https://ru.wikipedia.org/wiki/%D0%92%D1%81%D0%B5%D0%BC%D0%B8%D1%80%D0%BD%D0%BE%D0%B5_%D0%BA%D0%BE%D0%BE%D1%80%D0%B4%D0%B8%D0%BD%D0%B8%D1%80%D0%BE%D0%B2%D0%B0%D0%BD%D0%BD%D0%BE%D0%B5_%D0%B2%D1%80%D0%B5%D0%BC%D1%8F" rel="noopener noreferrer" target="_blank">UTC</a>(Всемирным координированным временем) нет.</p>
<p>Приложение работало всю зиму, но в первый день апреля уведомление ушло на час позже полудня. Дело в том, что разработчик просто прикрутил таймер, который срабатывает раз в 24 часа, а в последний день марта в Великобритании перевели время на час вперед — на летнее. Разработчик быстро исправил ситуацию, заменив таймер на планировщик задач — последний срабатывает в соответствии с локальным временем сервера, который находится в квартире разработчика.</p>
<p>Как ни странно, приложение пользуется популярностью. Спустя некоторое время разработчик получает отзыв от разгневанного пользователя из Нью-Йорка: тот пишет, что уведомления приходят вовсе не в полдень, а в 7 утра. Нью-Йорк находится в часовом поясе ETC (Eastern Time Zone), то есть UTC -05</p><div></div>.<p></p>
<p>Для разных часовых поясов нельзя просто ставить таймер, отталкиваясь от времени на сервере. Разработчик добавил выбор часового пояса при регистрации и переписал планировщик так, чтобы он стартовал для каждого пользователя раньше или позже на несколько часов в зависимости от данных, которые человек ввёл при регистрации.</p>
<p>Кажется, теперь все работает хорошо — но в ноябре разработчику начали писать пользователи из Москвы и жаловаться на уведомления, которые приходят в 11 утра. Дело в том, что в Великобритании время переводится с зимнего на летнее, а в России — нет. К тому моменту разработчик начал нервничать и решил просто записать в коде, что если пользователь из России, то делать переход во времени для него не нужно. Приложение тем временем только набирало популярность.</p>
<p>В марте, разработчику пишут из Сиднея — им уведомление приходит в два часа дня. Все потому, что в южном полушарии в марте происходит переход не на летнее время, а на зимнее. Разработчику не оставалось ничего иного, как дополнить свой код информацией о том, что в южном полушарии надо переводить время наоборот. Приложение уже стало популярным во всём мире и на разработчика посыпались претензии со всех уголков света. Например, пользователи из Палестины писали жалобы каждый раз, как их правительство переводило часы с летнего времени на зимнее, а случается это каждый раз в разное время. А ещё дело было в 2011 году, и 29 декабря пользователи из Самоа дружно пересекли <a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="https://ru.wikipedia.org/wiki/%D0%9B%D0%B8%D0%BD%D0%B8%D1%8F_%D0%BF%D0%B5%D1%80%D0%B5%D0%BC%D0%B5%D0%BD%D1%8B_%D0%B4%D0%B0%D1%82%D1%8B" rel="noopener noreferrer" target="_blank">линию перемены даты</a>, совершив скачок в будущее и очутившись 31 декабря 2011 года. Еще через год разработчику позвонили из <a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="https://ru.wikipedia.org/wiki/%D0%9C%D0%B5%D0%B6%D0%B4%D1%83%D0%BD%D0%B0%D1%80%D0%BE%D0%B4%D0%BD%D0%B0%D1%8F_%D1%81%D0%BB%D1%83%D0%B6%D0%B1%D0%B0_%D0%B2%D1%80%D0%B0%D1%89%D0%B5%D0%BD%D0%B8%D1%8F_%D0%97%D0%B5%D0%BC%D0%BB%D0%B8" rel="noopener noreferrer" target="_blank">Международной службы вращения Земли</a> и сообщили, что ему надо поправить приложение, потому что в этом году будет на
одну секунду больше, чем в прошлом из-за добавления дополнительной секунды. Записывать
все это в код — совершенно безумная идея. Поэтому разработчик закрыл проект и больше никогда не писал код.</p>
<h3 id="heading-3-14">Пара советов, как работать с локальным временем</h3>
<p>Что бы мог сделать разработчик, чтобы избежать такого печального финала? Все просто — достаточно было учесть опыт тех, кто уже решал эту проблему и выкладывал решение в открытый доступ. В каждом языке программирования есть как минимум одна библиотека, которая умеет работать с человеческим временем и часовыми поясами. Более того, существует база данных, в которой хранится информация о локальном времени, в том числе в исторической перспективе.</p>
<p>Что бы мог сделать разработчик:</p>
<ul>
<li>Переложить ответственность за подсчет времени на библиотеки и базы данных;</li>
<li>Хранить не разницу во времени с локальным временем пользователя, а его часовой
пояс в соответствии с <a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="https://en.wikipedia.org/wiki/List_of_tz_database_time_zones" rel="noopener noreferrer" target="_blank">классификацией IANA</a>,
то есть не +3, а "Europe/Moscow";</li>
<li>Настроить планировщик так, чтобы он выбирал задачи из списка в соответствии с
часовым поясом.</li>
</ul>
<h3 id="heading-3-15">Пара советов, как НЕ работать с локальным временем</h3>
<p>Существует много сценариев, при которых крайне важна последовательность событий, на которую не могут повлиять ни изменения часовых поясов, ни переход на летнее время.</p>
<p>Предположим, нужно написать сервис, обслуживающий банковские транзакции. Его функционал крайне простой: один пользователь отправляет другому деньги банковским переводом, любой из них может получить информацию обо всех переводах. В данном случае нужно хранить информацию о том, кто, кому, когда и сколько денег перевёл. При этом показывать эту информацию нужно в локальном времени пользователя при
условии, что транзакции в любом часовом поясе идут одна за другой.</p>
<p>Учтём опыт разработчика из примера выше и не будем описывать время самостоятельно, а используем библиотеку. Тут мы столкнемся с рядом вопросов:</p>
<ul>
<li>В каком часовом поясе хранить транзакции, если клиенты живут в разных странах,
а база данных находится в третьей стране;</li>
<li>Как учитывать переход на летнее и зимнее время;</li>
<li>Как учитывать дополнительную секунду;</li>
<li>Что делать, если пользователь переехал из одной страны в другую.</li>
</ul>
<p>И лучшим решением будет не делать ничего. Не нужно учитывать временные зоны клиентов, не нужно учитывать переходы на летнее и зимнее время, не нужно учитывать переезды пользователей. Когда нужно абсолютное время, достаточно использовать UTC.</p>
<p>Время в UTC поддерживается почти всеми библиотеками и почти всеми база и данных. Используя UTC, мы можем получать транзакцию в локальном времени пользователя, переводить это время в UTC и сохранять у себя. И транзакции всегда будут идти одна за другой.</p>
<p>Правда, существует проблема: UTC учитывает дополнительные секунды, а значит, в какой-то момент мы можем получить такие транзакции:</p>
<ul>
<li>Маша Феде перевела 100 рублей 2016-12-31 в 23:59<div></div></li>
<li>Федя Юре перевёл 100 рублей 2016-12-31 в 23:59<div></div></li>
<li>Юра отправил Майклу 1$ 2016-12-31 в 23:59<div></div></li>
<li>Майкл отправил Теду 99$ 2017-01-01 в 00:00<div></div></li>
</ul>
<p>Скорее всего программа, работающая с UTC, не сможет сказать, что произошло раньше — вторая транзакция, третья или четвертая. Google, которой очень важна абсолютная последовательность событий во времени, очень интересно решила эту проблему. В своих серверах точного времени они добавляют дополнительную секунду не за раз, а равномерно распределяют её в течение дня в начале года. Это называется <a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="https://developers.google.com/time/smear" rel="noopener noreferrer" target="_blank">leap smear</a>. Таким образом, раз в год, когда астрономы объявляют дополнительную секунду, в течение суток сервера точного времени Google немного отстают от UTC, чтобы набрать дополнительную секунду.</p>
<h2 id="heading-2-16">Выводы</h2>
<ul>
<li>В любом современном ПК есть процессор, который умеет производить колебания с
равными интервалами.</li>
<li>Зная период этих колебаний, операционная система компьютера может считать
человеческое время.</li>
<li>С помощью протокола NTP операционная система может обращаться к серверам
точного времени для скорректировать своих часов.</li>
<li>Когда компьютер выключен, время считает специальное устройство — RTC, которое
питается от батареи.</li>
<li>Epoch — это некая дата, от которой принято отсчитывать время вперед или назад.
Разные операционные системы и языки программирования могут выбирать свой собственный
epoch.</li>
<li>Unix-timestamp — это количество секунд, прошедших с unix-epoch, то есть
с 00<div></div> 1 января 1970 года.</li>
<li>Никогда не пишите самостоятельно код, который описывает человеческое время.</li>
<li>Если вам нужна точная последовательность действий во времени, не используйте
часовые зоны. Храните время в UTC и по требованию конвертируйте его.</li>
</ul>
<h2 id="heading-2-17">Дополнительные материалы</h2>
<ul>
<li><a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="https://www.youtube.com/watch?v=YuSgZ173Utg" rel="noopener noreferrer" target="_blank">Как компьютеры складывают числа</a></li>
</ul></div><div class=""><div style="--group-gap:var(--mantine-spacing-md);--group-align:center;--group-justify:space-between;--group-wrap:wrap;margin-bottom:var(--mantine-spacing-lg)" class="m_4081bf90 mantine-Group-root"><div class="m_4451eb3a mantine-Center-root" data-inline="true"><div style="--ti-size:var(--ti-size-xs);--ti-bg:transparent;--ti-color:var(--mantine-color-indigo-light-color);--ti-bd:calc(0.0625rem * var(--mantine-scale)) solid transparent;margin-inline-end:var(--mantine-spacing-xs);color:inherit" class="m_7341320d mantine-ThemeIcon-root" data-variant="transparent" data-size="xs"><svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="tabler-icon tabler-icon-user "><path d="M8 7a4 4 0 1 0 8 0a4 4 0 0 0 -8 0"></path><path d="M6 21v-2a4 4 0 0 1 4 -4h4a4 4 0 0 1 4 4v2"></path></svg></div><p style="margin-inline-end:var(--mantine-spacing-xl)" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">Валерия Белякова</p><p class="mantine-focus-auto m_b6d8b162 mantine-Text-root">год назад</p></div><div style="align-items:center" class="m_8bffd616 mantine-Flex-root __m__-_R_5dirddmiub_"><a style="display:inline-flex" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="/blog/posts/time/votes"><div style="--ti-size:var(--ti-size-sm);--ti-bg:transparent;--ti-color:var(--mantine-color-indigo-light-color);--ti-bd:calc(0.0625rem * var(--mantine-scale)) solid transparent;margin-inline-end:var(--mantine-spacing-xs);color:inherit" class="m_7341320d mantine-ThemeIcon-root" data-variant="transparent" data-size="sm"><svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="tabler-icon tabler-icon-thumb-up "><path d="M7 11v8a1 1 0 0 1 -1 1h-2a1 1 0 0 1 -1 -1v-7a1 1 0 0 1 1 -1h3a4 4 0 0 0 4 -4v-1a2 2 0 0 1 4 0v5h3a2 2 0 0 1 2 2l-1 5a2 3 0 0 1 -2 2h-7a3 3 0 0 1 -3 -3"></path></svg></div></a><p class="mantine-focus-auto m_b6d8b162 mantine-Text-root">0</p></div></div></div><div style="background-color:var(--mantine-color-indigo-light);border:calc(0.0625rem * var(--mantine-scale)) solid transparent;padding:var(--mantine-spacing-xl)" class="m_e615b15f mantine-Card-root m_1b7284a3 mantine-Paper-root"><p style="margin-bottom:var(--mantine-spacing-sm);font-size:var(--mantine-font-size-h4)" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">Читайте также:</p><ul style="margin-inline-start:var(--mantine-spacing-lg)" class="m_abbac491 mantine-List-root"><li style="margin-bottom:var(--mantine-spacing-sm)" class="m_abb6bec2 mantine-List-item" data-with-icon="true"><div class="m_75cd9f71 mantine-List-itemWrapper"><span class="m_60f83e5b mantine-List-itemIcon"><div class="m_4451eb3a mantine-Center-root"><div style="--ti-size:var(--ti-size-xs);--ti-bg:transparent;--ti-color:var(--mantine-color-indigo-light-color);--ti-bd:calc(0.0625rem * var(--mantine-scale)) solid transparent;color:inherit" class="m_7341320d mantine-ThemeIcon-root" data-variant="transparent" data-size="xs"><svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="tabler-icon tabler-icon-chevron-compact-right "><path d="M11 4l3 8l-3 8"></path></svg></div></div></span><span class="mantine-List-itemLabel"><a style="color:inherit" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="/blog/posts/makefile-as-task-runner">Что такое Makefile и как начать его использовать</a></span></div></li><li style="margin-bottom:var(--mantine-spacing-sm)" class="m_abb6bec2 mantine-List-item" data-with-icon="true"><div class="m_75cd9f71 mantine-List-itemWrapper"><span class="m_60f83e5b mantine-List-itemIcon"><div class="m_4451eb3a mantine-Center-root"><div style="--ti-size:var(--ti-size-xs);--ti-bg:transparent;--ti-color:var(--mantine-color-indigo-light-color);--ti-bd:calc(0.0625rem * var(--mantine-scale)) solid transparent;color:inherit" class="m_7341320d mantine-ThemeIcon-root" data-variant="transparent" data-size="xs"><svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="tabler-icon tabler-icon-chevron-compact-right "><path d="M11 4l3 8l-3 8"></path></svg></div></div></span><span class="mantine-List-itemLabel"><a style="color:inherit" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="/blog/posts/rest-api">Что такое REST API?</a></span></div></li><li style="margin-bottom:var(--mantine-spacing-sm)" class="m_abb6bec2 mantine-List-item" data-with-icon="true"><div class="m_75cd9f71 mantine-List-itemWrapper"><span class="m_60f83e5b mantine-List-itemIcon"><div class="m_4451eb3a mantine-Center-root"><div style="--ti-size:var(--ti-size-xs);--ti-bg:transparent;--ti-color:var(--mantine-color-indigo-light-color);--ti-bd:calc(0.0625rem * var(--mantine-scale)) solid transparent;color:inherit" class="m_7341320d mantine-ThemeIcon-root" data-variant="transparent" data-size="xs"><svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="tabler-icon tabler-icon-chevron-compact-right "><path d="M11 4l3 8l-3 8"></path></svg></div></div></span><span class="mantine-List-itemLabel"><a style="color:inherit" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="/blog/posts/hosting">Что такое хостинг и домен сайта простыми словами</a></span></div></li></ul></div><div style="margin-block:var(--mantine-spacing-xl)" class="m_3eebeb36 mantine-Divider-root" data-orientation="horizontal" role="separator"></div></div><div></div></div><style data-mantine-styles="inline">.__m__-_R_lmiub_{--col-flex-grow:auto;--col-flex-basis:100%;--col-max-width:100%;}@media(min-width: 48em){.__m__-_R_lmiub_{--col-flex-grow:auto;--col-flex-basis:16.666666666666668%;--col-max-width:16.666666666666668%;}}@media(min-width: 62em){.__m__-_R_lmiub_{--col-flex-grow:auto;--col-flex-basis:33.333333333333336%;--col-max-width:33.333333333333336%;}}</style><div class="m_96bdd299 mantine-Grid-col __m__-_R_lmiub_ mantine-visible-from-md"><div style="background-color:var(--mantine-color-indigo-light);border:calc(0.0625rem * var(--mantine-scale)) solid transparent;margin-bottom:var(--mantine-spacing-xl);padding:var(--mantine-spacing-xl);width:100%" class="m_e615b15f mantine-Card-root m_1b7284a3 mantine-Paper-root"><div style="margin-bottom:var(--mantine-spacing-md)" class="m_4451eb3a mantine-Center-root" data-inline="true"><p style="font-size:var(--mantine-font-size-h4)" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">Категории</p></div><ul class="m_abbac491 mantine-List-root"><li style="margin-bottom:var(--mantine-spacing-xs)" class="m_abb6bec2 mantine-List-item" data-with-icon="true"><div class="m_75cd9f71 mantine-List-itemWrapper"><span class="m_60f83e5b mantine-List-itemIcon"><div class="m_4451eb3a mantine-Center-root"><div style="--ti-size:var(--ti-size-xs);--ti-bg:transparent;--ti-color:var(--mantine-color-indigo-light-color);--ti-bd:calc(0.0625rem * var(--mantine-scale)) solid transparent;color:inherit" class="m_7341320d mantine-ThemeIcon-root" data-variant="transparent" data-size="xs"><svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="1.2" stroke-linecap="round" stroke-linejoin="round" class="tabler-icon tabler-icon-chevron-compact-right "><path d="M11 4l3 8l-3 8"></path></svg></div></div></span><span class="mantine-List-itemLabel"><button style="color:inherit;text-decoration:underline" class="mantine-focus-auto m_87cf2631 mantine-UnstyledButton-root" type="button" aria-label="Мотивация">Мотивация</button></span></div></li><li style="margin-bottom:var(--mantine-spacing-xs)" class="m_abb6bec2 mantine-List-item" data-with-icon="true"><div class="m_75cd9f71 mantine-List-itemWrapper"><span class="m_60f83e5b mantine-List-itemIcon"><div class="m_4451eb3a mantine-Center-root"><div style="--ti-size:var(--ti-size-xs);--ti-bg:transparent;--ti-color:var(--mantine-color-indigo-light-color);--ti-bd:calc(0.0625rem * var(--mantine-scale)) solid transparent;color:inherit" class="m_7341320d mantine-ThemeIcon-root" data-variant="transparent" data-size="xs"><svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="1.2" stroke-linecap="round" stroke-linejoin="round" class="tabler-icon tabler-icon-chevron-compact-right "><path d="M11 4l3 8l-3 8"></path></svg></div></div></span><span class="mantine-List-itemLabel"><button style="color:inherit;text-decoration:underline" class="mantine-focus-auto m_87cf2631 mantine-UnstyledButton-root" type="button" aria-label="Истории успеха">Истории успеха</button></span></div></li><li style="margin-bottom:var(--mantine-spacing-xs)" class="m_abb6bec2 mantine-List-item" data-with-icon="true"><div class="m_75cd9f71 mantine-List-itemWrapper"><span class="m_60f83e5b mantine-List-itemIcon"><div class="m_4451eb3a mantine-Center-root"><div style="--ti-size:var(--ti-size-xs);--ti-bg:transparent;--ti-color:var(--mantine-color-indigo-light-color);--ti-bd:calc(0.0625rem * var(--mantine-scale)) solid transparent;color:inherit" class="m_7341320d mantine-ThemeIcon-root" data-variant="transparent" data-size="xs"><svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="1.2" stroke-linecap="round" stroke-linejoin="round" class="tabler-icon tabler-icon-chevron-compact-right "><path d="M11 4l3 8l-3 8"></path></svg></div></div></span><span class="mantine-List-itemLabel"><button style="color:inherit;text-decoration:underline" class="mantine-focus-auto m_87cf2631 mantine-UnstyledButton-root" type="button" aria-label="Дневник студента">Дневник студента</button></span></div></li><li style="margin-bottom:var(--mantine-spacing-xs)" class="m_abb6bec2 mantine-List-item" data-with-icon="true"><div class="m_75cd9f71 mantine-List-itemWrapper"><span class="m_60f83e5b mantine-List-itemIcon"><div class="m_4451eb3a mantine-Center-root"><div style="--ti-size:var(--ti-size-xs);--ti-bg:transparent;--ti-color:var(--mantine-color-indigo-light-color);--ti-bd:calc(0.0625rem * var(--mantine-scale)) solid transparent;color:inherit" class="m_7341320d mantine-ThemeIcon-root" data-variant="transparent" data-size="xs"><svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="1.2" stroke-linecap="round" stroke-linejoin="round" class="tabler-icon tabler-icon-chevron-compact-right "><path d="M11 4l3 8l-3 8"></path></svg></div></div></span><span class="mantine-List-itemLabel"><button style="color:inherit;text-decoration:underline" class="mantine-focus-auto m_87cf2631 mantine-UnstyledButton-root" type="button" aria-label="Код">Код</button></span></div></li><li style="margin-bottom:var(--mantine-spacing-xs)" class="m_abb6bec2 mantine-List-item" data-with-icon="true"><div class="m_75cd9f71 mantine-List-itemWrapper"><span class="m_60f83e5b mantine-List-itemIcon"><div class="m_4451eb3a mantine-Center-root"><div style="--ti-size:var(--ti-size-xs);--ti-bg:transparent;--ti-color:var(--mantine-color-indigo-light-color);--ti-bd:calc(0.0625rem * var(--mantine-scale)) solid transparent;color:inherit" class="m_7341320d mantine-ThemeIcon-root" data-variant="transparent" data-size="xs"><svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="1.2" stroke-linecap="round" stroke-linejoin="round" class="tabler-icon tabler-icon-chevron-compact-right "><path d="M11 4l3 8l-3 8"></path></svg></div></div></span><span class="mantine-List-itemLabel"><button style="color:inherit;text-decoration:underline" class="mantine-focus-auto m_87cf2631 mantine-UnstyledButton-root" type="button" aria-label="Карьера">Карьера</button></span></div></li></ul></div><div style="justify-content:end;margin-top:0rem;position:sticky;top:calc(5rem * var(--mantine-scale))" class="m_8bffd616 mantine-Flex-root __m__-_R_5dlmiub_"><div tabindex="0" style="cursor:pointer"><a style="text-decoration:none" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="/courses_web_development?promo_name=program_category&promo_position=blog_post&promo_creative=card&promo_type=card"><div style="background-color:var(--mantine-color-default);border:calc(0.0625rem * var(--mantine-scale)) solid var(--mantine-color-default-border);padding-inline:var(--mantine-spacing-xl);padding-top:var(--mantine-spacing-xl);padding-bottom:var(--mantine-spacing-xs);width:100%" class="m_e615b15f mantine-Card-root m_1b7284a3 mantine-Paper-root"><div class="m_4451eb3a mantine-Center-root" data-inline="true"><p style="font-size:var(--mantine-font-size-h4)" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">Курсы по веб-разработке</p></div><img class="m_9e117634 mantine-Image-root" src="/vite/assets/development-BVihs_d5.png"/><p style="margin-bottom:var(--mantine-spacing-xs);text-align:right" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">Посмотреть →</p></div></a></div></div></div></div></div></div></div>
</main>
<footer class="bg-dark fw-light text-light px-3 py-5">
<div class="row small">
<div class="col-12 col-sm-6 col-md-3">
<div class="h5 mb-3">Хекслет</div>
<ul class="list-unstyled">
<li>
<a class="nav-link link-light py-1 ps-0" href="/pages/about">О нас</a>
</li>
<li>
<a class="nav-link link-light py-1 ps-0" href="/testimonials">Отзывы</a>
</li>
<li>
<span class="nav-link link-light py-1 ps-0 external-link" data-href="https://b2b.hexlet.io" role="button">Корпоративное обучение</span>
</li>
<li>
<a class="nav-link link-light py-1 ps-0" href="/blog">Блог</a>
</li>
<li>
<a class="nav-link link-light py-1 ps-0" href="/qna">Вопросы и ответы</a>
</li>
<li>
<a class="nav-link link-light py-1 ps-0" href="/glossary">Глоссарий</a>
</li>
<li>
<span class="nav-link link-light py-1 ps-0 external-link" data-href="https://help.hexlet.io" data-target="_blank" role="button">Справка</span>
</li>
<li>
<a class="nav-link link-light py-1 ps-0" target="_blank" rel="noopener noreferrer" href="/map">Карта сайта</a>
</li>
</ul>
</div>
<div class="col-12 col-sm-6 col-md-3">
<div class="h5 fw-normal mb-3">Направления</div>
<ul class="list-unstyled">
<li>
<a class="nav-link link-light py-1 ps-0" href="/courses_devops">DevOps
</a></li>
<li>
<a class="nav-link link-light py-1 ps-0" href="/courses_data_analytics">Аналитика
</a></li>
<li>
<a class="nav-link link-light py-1 ps-0" href="/courses_backend_development">Бэкенд
</a></li>
<li>
<a class="nav-link link-light py-1 ps-0" href="/courses_programming">Программирование
</a></li>
<li>
<a class="nav-link link-light py-1 ps-0" href="/courses_testing">Тестирование
</a></li>
<li>
<a class="nav-link link-light py-1 ps-0" href="/courses_front_end_dev">Фронтенд
</a></li>
</ul>
</div>
<div class="col-12 col-sm-6 col-md-3">
<div class="h5">Профессии</div>
<ul class="list-unstyled">
<li>
<a class="nav-link link-light py-1 ps-0" href="/programs/devops-engineer-from-scratch">DevOps-инженер с нуля</a>
</li>
<li>
<a class="nav-link link-light py-1 ps-0" href="/programs/go">Go-разработчик</a>
</li>
<li>
<a class="nav-link link-light py-1 ps-0" href="/programs/java">Java-разработчик</a>
</li>
<li>
<a class="nav-link link-light py-1 ps-0" href="/programs/python">Python-разработчик </a>
</li>
<li>
<a class="nav-link link-light py-1 ps-0" href="/programs/data-analytics">Аналитик данных</a>
</li>
<li>
<a class="nav-link link-light py-1 ps-0" href="/programs/qa-engineer">Инженер по ручному тестированию</a>
</li>
<li>
<a class="nav-link link-light py-1 ps-0" href="/programs/php">РНР-разработчик</a>
</li>
<li>
<a class="nav-link link-light py-1 ps-0" href="/programs/frontend">Фронтенд-разработчик</a>
</li>
</ul>
</div>
<div class="col-12 col-sm-6 col-md-3">
<div class="h5">Навыки</div>
<ul class="list-unstyled">
<li>
<a class="nav-link link-light py-1 ps-0" href="/programs/python-django-developer">Django</a>
</li>
<li>
<a class="nav-link link-light py-1 ps-0" href="/programs/docker">Docker</a>
</li>
<li>
<a class="nav-link link-light py-1 ps-0" href="/programs/php-laravel-developer">Laravel</a>
</li>
<li>
<a class="nav-link link-light py-1 ps-0" href="/programs/postman">Postman</a>
</li>
<li>
<a class="nav-link link-light py-1 ps-0" href="/programs/js-react-developer">React</a>
</li>
<li>
<a class="nav-link link-light py-1 ps-0" href="/programs/js-rest-api">REST API в Node.js</a>
</li>
<li>
<a class="nav-link link-light py-1 ps-0" href="/programs/spring-boot">Spring Boot</a>
</li>
<li>
<a class="nav-link link-light py-1 ps-0" href="/programs/typescript">Typescript</a>
</li>
</ul>
</div>
</div>
<hr>
<div class="row">
<div class="col-12 col-sm-4 col-md-2">
<div class="fs-4">
<ul class="list-unstyled d-flex">
<li class="me-3">
<a aria-label="Telegram" target="_blank" class="link-light" rel="noopener noreferrer nofollow" href="https://t.me/hexlet_ru"><span class="bi bi-telegram"></span>
</a></li>
<li>
<a aria-label="Youtube" target="_blank" class="link-light" rel="noopener noreferrer nofollow" href="https://www.youtube.com/user/HexletUniversity"><span class="bi bi-youtube"></span>
</a></li>
</ul>
</div>
<div class="mb-2 d-flex flex-column">
<a class="link-light text-decoration-none" rel="nofollow" href="mailto:support@hexlet.io">support@hexlet.io</a>
<a class="link-light text-decoration-none py-2" target="_blank" href="https://t.me/hexlet_help_bot">t.me/hexlet_help_bot</a>
</div>
<ul class="list-unstyled d-flex">
<li class="me-3">
<span class="link-light text-decoration-none opacity-50 x-font-size-18 external-link" rel="nofollow" data-href="https://hexlet.io/locale/switch?new_locale=en" data-target="_self" role="button"><span class="my-auto">EN</span>
</span></li>
<li class="me-3">
<span class="link-light text-decoration-none opacity-50 x-font-size-18 opacity-100 external-link" rel="nofollow" data-href="https://ru.hexlet.io/locale/switch?new_locale=ru" data-target="_self" role="button"><span class="my-auto">RU</span>
</span></li>
<li class="me-3">
<span class="link-light text-decoration-none opacity-50 x-font-size-18 external-link" rel="nofollow" data-href="https://kz.hexlet.io/locale/switch?new_locale=kz" data-target="_self" role="button"><span class="my-auto">KZ</span>
</span></li>
</ul>
</div>
<div class="col-12 col-sm-4 col-md-3">
<ul class="list-unstyled fs-4">
<li class="mb-3">
<a class="link-light text-decoration-none" href="tel:8%20800%20100%2022%2047">8 800 100 22 47</a>
<span class="d-block opacity-50 small">бесплатно по РФ</span>
</li>
<li>
<a class="link-light text-decoration-none" href="tel:%2B7%20495%20085%2021%2062">+7 495 085 21 62</a>
<span class="d-block opacity-50 small">бесплатно по Москве</span>
</li>
</ul>
</div>
<div class="col-12 col-sm-4 col-md-3">
<div class="small mb-3">Образовательные услуги оказываются на основании Л035-01298-77/01989008 от 14.03.2025</div>
<ul class="list-unstyled small">
<li>
<a class="nav-link link-light py-1 ps-0" href="/pages/legal">Правовая информация</a>
</li>
<li>
<a class="nav-link link-light py-1 ps-0" href="/pages/offer">Оферта</a>
</li>
<li>
<a class="nav-link link-light py-1 ps-0" href="/pages/license">Лицензия</a>
</li>
<li>
<a class="nav-link link-light py-1 ps-0" href="/pages/contacts">Контакты</a>
</li>
</ul>
</div>
<div class="col-12 col-sm-12 col-md-4 small">
<div class="mb-2">
<div>ООО «<a href="/" class="text-decoration-none link-light">Хекслет Рус</a>»</div>
<div>108813 г. Москва, вн.тер.г. поселение Московский,</div>
<div>г. Московский, ул. Солнечная, д. 3А, стр. 1, помещ. 20Б/3</div>
<div>ОГРН 1217300010476</div>
<div>ИНН 7325174845</div>
</div>
<hr>
<div>АНО ДПО «<a href="/" class="text-decoration-none link-light">Учебный центр «Хекслет</a>»</div>
<div>119331 г. Москва, вн. тер. г. муниципальный округ</div>
<div>Ломоносовский, пр-кт Вернадского, д. 29</div>
<div>ОГРН 1247700712390</div>
<div>ИНН 7736364948</div>
</div>
</div>
</footer>
<div id="root-assistant-offcanvas"></div>
<script src="/vite/assets/assistant-CdBlNCiQ.js" crossorigin="anonymous" type="module"></script><link rel="modulepreload" href="/vite/assets/chunk-DsPFFUou.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/init-nkZBEvfU.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/ErrorFallbackBlock-naDSYSy9.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/MarkdownBlock-DbyKWoR_.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/gon-D3e4yh1x.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/mantine-CGMYrt2Y.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/shiki-V011pkdv.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/utils-DRqSHbQE.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/routes-CCH8ilKF.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/lib-XR8Qr8kR.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/dist-GCHh59xr.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/Box-B5-OOzBf.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/notifications.store-C-3AFSMn.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/useIsomorphicEffect-HJ6VK0D3.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/lib-KSp6QbZ0.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/axios-BEvgo0ym.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/classnames-l6ipYlLR.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/dayjs.min-BkKovM-s.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/debounce-jMQ_Cf4f.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/i18next-BlSq9s7B.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/client-U9M77rxp.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/react-dom-DaLxUz_h.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/useTranslation-Bx1Cdrkz.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/compiler-runtime-6XxiPFnt.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/jsx-runtime-CwjcCKJi.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/react-CkL4ZRHB.js" as="script" crossorigin="anonymous">
<script defer src="https://static.cloudflareinsights.com/beacon.min.js/v67327c56f0bb4ef8b305cae61679db8f1769101564043" integrity="sha512-rdcWY47ByXd76cbCFzznIcEaCN71jqkWBBqlwhF1SY7KubdLKZiEGeP7AyieKZlGP9hbY/MhGrwXzJC/HulNyg==" data-cf-beacon='{"version":"2024.11.0","token":"d11015b65d11429ea6b4a2ef37dd7e0b","server_timing":{"name":{"cfCacheStatus":true,"cfEdge":true,"cfExtPri":true,"cfL4":true,"cfOrigin":true,"cfSpeedBrain":true},"location_startswith":null}}' crossorigin="anonymous"></script>
</body>
</html>