Java for Beginner
673 subscribers
541 photos
155 videos
12 files
827 links
Канал от новичков для новичков!
Изучайте Java вместе с нами!
Здесь мы обмениваемся опытом и постоянно изучаем что-то новое!

Наш YouTube канал - https://www.youtube.com/@Java_Beginner-Dev

Наш канал на RUTube - https://rutube.ru/channel/37896292/
Download Telegram
Форматирование и парсинг дат с DateTimeFormatter

1. Создание кастомного формата с помощью DateTimeFormatter.ofPattern()

Класс DateTimeFormatter в Java используется для форматирования и парсинга дат и времени. С помощью метода ofPattern() можно создать кастомный формат, используя специальные символы. Например:

DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss");


В DateTimeFormatter используется множество символов для создания кастомных форматов. Вот полный список основных вариантов:

1. Год (Year)
y или yyyy — год (4 цифры). Пример: 2023.
yy — год (2 цифры). Пример: 23.
u — год в формате ISO (аналогичен y, но поддерживает отрицательные годы). Пример: 2023.


2. Месяц (Month)
M — месяц (от 1 до 12). Пример: 1, 12.
MM — месяц (2 цифры). Пример: 01, 12.
MMM — сокращенное название месяца. Пример: Jan, Dec.
MMMM — полное название месяца. Пример: January, December.


3. День (Day)
d — день месяца (от 1 до 31). Пример: 1, 31.
dd — день месяца (2 цифры). Пример: 01, 31.
D — день года (от 1 до 366). Пример: 1, 365.
F — день недели в месяце (например, 2-й понедельник). Пример: 1, 2.


4. День недели (Day of Week)
E — сокращенное название дня недели. Пример: Mon, Sun.
EEEE — полное название дня недели. Пример: Monday, Sunday.


5. Час (Hour)
H — час в 24-часовом формате (от 0 до 23). Пример: 0, 23.
HH — час в 24-часовом формате (2 цифры). Пример: 00, 23.
h — час в 12-часовом формате (от 1 до 12). Пример: 1, 12.
hh — час в 12-часовом формате (2 цифры). Пример: 01, 12.
k — час в 24-часовом формате (от 1 до 24). Пример: 1, 24.
K — час в 12-часовом формате (от 0 до 11). Пример: 0, 11.


6. Минуты (Minutes)
m — минуты (от 0 до 59). Пример: 0, 59.
mm — минуты (2 цифры). Пример: 00, 59.


7. Секунды (Seconds)
s — секунды (от 0 до 59). Пример: 0, 59.
ss — секунды (2 цифры). Пример: 00, 59.
S — доли секунды (миллисекунды, наносекунды). Пример: 0, 999.


8. Временная зона (Time Zone)
z — название временной зоны. Пример: PST, GMT+03:00.
Z — смещение временной зоны в формате +HHMM или -HHMM. Пример: +0300, -0500.
X — смещение временной зоны в формате Z, ±HH:mm, ±HHmm или ±HH. Пример: Z, +03, +0300.
O — локальное смещение временной зоны. Пример: GMT+3.


9. Период дня (AM/PM)
a — маркер периода дня (AM/PM). Пример: AM, PM.

10. Эра (Era)
G — эра (до н.э. или н.э.). Пример: AD, BC.

Примеры использования

Полный формат даты и времени:
DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss");
LocalDateTime now = LocalDateTime.now();
System.out.println(now.format(formatter)); // Пример: 2023-10-25 14:30:00


Сокращенное название месяца и дня недели:
DateTimeFormatter formatter = DateTimeFormatter.ofPattern("dd MMM yyyy, EEE");
LocalDate date = LocalDate.now();
System.out.println(date.format(formatter)); // Пример: 25 Oct 2023, Wed


Время с AM/PM:
DateTimeFormatter formatter = DateTimeFormatter.ofPattern("hh:mm:ss a");
LocalTime time = LocalTime.now();
System.out.println(time.format(formatter)); // Пример: 02:30:00 PM


Смещение временной зоны:
DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss Z");
ZonedDateTime zonedDateTime = ZonedDateTime.now();
System.out.println(zonedDateTime.format(formatter)); // Пример: 2023-10-25 14:30:00 +0300


Нюансы использования

Регистр символов: Символы в DateTimeFormatter чувствительны к регистру. Например, MM — это месяц, а mm — минуты.
Повторение символов: Количество символов влияет на формат. Например, yyyy выводит год с 4 цифрами, а yy — с 2.
Экранирование символов: Если нужно вывести символ как текст (например, T в ISO_DATE_TIME), используйте одинарные кавычки: 'T'.


#Java #Training #Medium #Date #DateTimeFormatter
2. Форматирование LocalDateTime или ZonedDateTime в строку

Чтобы отформатировать объект LocalDateTime или ZonedDateTime в строку, используйте метод format():
LocalDateTime now = LocalDateTime.now();
DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss");
String formattedDateTime = now.format(formatter);
System.out.println("Formatted DateTime: " + formattedDateTime);


Для ZonedDateTime:

ZonedDateTime zonedDateTime = ZonedDateTime.now();
DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss z");
String formattedZonedDateTime = zonedDateTime.format(formatter);
System.out.println("Formatted Zoned DateTime: " + formattedZonedDateTime);


3. Парсинг строки обратно в объект


Чтобы преобразовать строку обратно в объект LocalDateTime или ZonedDateTime, используйте метод parse():
```
String dateTimeString = "2023-10-25 14:30:00";
DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss");
LocalDateTime parsedDateTime = LocalDateTime.parse(dateTimeString, formatter);
System.out.println("Parsed DateTime: " + parsedDateTime);

Для ZonedDateTime:
String zonedDateTimeString = "2023-10-25 14:30:00 Europe/Moscow";
DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss z");
ZonedDateTime parsedZonedDateTime = ZonedDateTime.parse(zonedDateTimeString, formatter);
System.out.println("Parsed Zoned DateTime: " + parsedZonedDateTime);


4. Стандартные форматы в DateTimeFormatter

Java предоставляет несколько стандартных форматов, которые можно использовать без создания кастомного DateTimeFormatter:
ISO_LOCAL_DATE — формат даты без времени (например, 2023-10-25).
ISO_DATE_TIME — полный формат даты и времени с часовым поясом (например, 2023-10-25T14:30:00+03:00).
ISO_LOCAL_DATE_TIME — формат даты и времени без часового пояса (например, 2023-10-25T14:30:00).

Пример использования стандартных форматов:
LocalDateTime now = LocalDateTime.now();
String isoLocalDateTime = now.format(DateTimeFormatter.ISO_LOCAL_DATE_TIME);
System.out.println("ISO Local Date Time: " + isoLocalDateTime);

ZonedDateTime zonedNow = ZonedDateTime.now();
String isoDateTime = zonedNow.format(DateTimeFormatter.ISO_DATE_TIME);
System.out.println("ISO Date Time: " + isoDateTime);


Плюсы и минусы DateTimeFormatter

Плюсы:
Гибкость: можно создавать любые кастомные форматы.
Потокобезопасность: DateTimeFormatter является immutable и thread-safe.
Поддержка стандартных форматов.


Минусы:
Сложность: для сложных форматов требуется знание множества символов.
Ошибки парсинга: если строка не соответствует формату, будет выброшено исключение DateTimeParseException.


#Java #Training #Medium #Date #DateTimeFormatter
Что выведет код?

import java.time.OffsetDateTime;
import java.time.ZoneOffset;

public class Task210125_1 {
public static void main(String[] args) {
OffsetDateTime dateTime = OffsetDateTime.of(2023, 10, 1, 12, 0, 0, 0, ZoneOffset.ofHours(2));
OffsetDateTime convertedDateTime = dateTime.withOffsetSameInstant(ZoneOffset.ofHours(-3));
System.out.println(convertedDateTime.getHour());
}
}


#Tasks
А еще постоянно хочу кофе и спать... 🤪😂

https://t.me/Java_for_beginner_dev

#Mems
Please open Telegram to view this post
VIEW IN TELEGRAM
ZoneOffset, OffsetDateTime и конвертация времени между часовыми поясами

1. Что такое ZoneOffset?

ZoneOffset — это класс, представляющий фиксированное смещение от UTC (Coordinated Universal Time). Он выражается в виде количества часов, минут и секунд, на которые время отличается от UTC.

Например:
+03:00 — смещение на 3 часа вперед от UTC.
-05:00 — смещение на 5 часов назад от UTC.


ZoneOffset используется для работы с временем, которое не зависит от правил временных зон (например, летнего времени). Это особенно полезно, когда вам нужно работать с фиксированным смещением.

Пример создания ZoneOffset:
ZoneOffset offset = ZoneOffset.ofHours(3); // +03:00
System.out.println("ZoneOffset: " + offset);

ZoneOffset offsetWithMinutes = ZoneOffset.ofHoursMinutes(5, 30); // +05:30
System.out.println("ZoneOffset with minutes: " + offsetWithMinutes);


2. Что такое OffsetDateTime?

OffsetDateTime — это класс, представляющий дату и время с фиксированным смещением от UTC. Он объединяет LocalDateTime и ZoneOffset. В отличие от ZonedDateTime, OffsetDateTime не учитывает правила временных зон (например, летнее время).

Пример создания OffsetDateTime:

ZoneOffset offset = ZoneOffset.ofHours(3);
OffsetDateTime offsetDateTime = OffsetDateTime.now(offset);
System.out.println("OffsetDateTime: " + offsetDateTime);


Когда использовать OffsetDateTime?
Когда вам нужно работать с временем, которое имеет фиксированное смещение от UTC.
Когда вам не нужно учитывать правила временных зон (например, летнее время).


3. Конвертация времени между часовыми поясами с помощью withZoneSameInstant()

Метод withZoneSameInstant() позволяет конвертировать время из одного часового пояса в другой, сохраняя момент времени (instant). Это полезно, когда вам нужно отобразить время в другом регионе.

Пример:
ZonedDateTime moscowTime = ZonedDateTime.now(ZoneId.of("Europe/Moscow"));
ZonedDateTime newYorkTime = moscowTime.withZoneSameInstant(ZoneId.of("America/New_York"));
System.out.println("Moscow Time: " + moscowTime);
System.out.println("New York Time: " + newYorkTime);


4. Получение текущего времени в конкретной временной зоне

Чтобы получить текущее время в конкретной временной зоне, используйте ZonedDateTime.now() с указанием ZoneId:
ZonedDateTime moscowTime = ZonedDateTime.now(ZoneId.of("Europe/Moscow"));
ZonedDateTime newYorkTime = ZonedDateTime.now(ZoneId.of("America/New_York"));
System.out.println("Moscow Time: " + moscowTime);
System.out.println("New York Time: " + newYorkTime);


5. Daylight Saving Time (летнее время)

Летнее время — это практика перевода часов на час вперед весной и обратно осенью. Java Time API автоматически учитывает переход на летнее время. Например:
ZonedDateTime beforeDST = ZonedDateTime.of(2023, 3, 26, 1, 59, 0, 0, ZoneId.of("Europe/Moscow"));
ZonedDateTime afterDST = beforeDST.plusMinutes(1);
System.out.println("Before DST: " + beforeDST);
System.out.println("After DST: " + afterDST);


#Java #Training #Medium #Date #ZonedDateTime #OffsetDateTime #withZoneSameInstant
6. Проверка, действует ли летнее время

Чтобы проверить, действует ли летнее время для конкретной временной зоны в определенный момент, используйте метод isDaylightSavings():
ZoneId zone = ZoneId.of("Europe/Moscow");
ZonedDateTime dateTime = ZonedDateTime.now(zone);
boolean isDST = zone.getRules().isDaylightSavings(dateTime.toInstant());
System.out.println("Is DST active? " + isDST);


Плюсы и минусы работы с ZoneOffset и OffsetDateTime

Плюсы:
Простота: ZoneOffset и OffsetDateTime работают с фиксированным смещением, что упрощает их использование.
Потокобезопасность: Оба класса являются immutable и thread-safe.
Универсальность: OffsetDateTime подходит для случаев, когда не нужно учитывать правила временных зон.


Минусы:
Ограниченность: ZoneOffset не учитывает правила временных зон, такие как летнее время.
Ошибки: Неправильное использование может привести к некорректным результатам, особенно при работе с регионами, где применяется летнее время.


Пример использования ZoneOffset и OffsetDateTime

// Создание ZoneOffset
ZoneOffset offset = ZoneOffset.ofHours(3); // +03:00

// Создание OffsetDateTime
OffsetDateTime offsetDateTime = OffsetDateTime.now(offset);
System.out.println("OffsetDateTime: " + offsetDateTime);

// Конвертация в другой часовой пояс
ZonedDateTime zonedDateTime = offsetDateTime.atZoneSameInstant(ZoneId.of("America/New_York"));
System.out.println("ZonedDateTime in New York: " + zonedDateTime);


#Java #Training #Medium #Date #ZonedDateTime #OffsetDateTime #withZoneSameInstant
Please open Telegram to view this post
VIEW IN TELEGRAM
Please open Telegram to view this post
VIEW IN TELEGRAM
Класс Instant и TemporalAdjusters

Класс Instant и временные метки

Instant — это класс из Java Time API (пакет java.time), который представляет собой точку на временной шкале. Он хранит количество секунд и наносекунд, прошедших с эпохи Unix (1 января 1970 года, 00:00:00 UTC). Это делает его аналогом временных меток (timestamp), которые часто используются в системах для фиксации моментов времени.

Как работает Instant под капотом?


Внутри Instant хранит два поля:
long seconds — количество секунд с эпохи Unix.
int nanos — количество наносекунд (от 0 до 999,999,999), чтобы обеспечить точность до наносекунд.
Методы getEpochSecond() и getNano() позволяют получить эти значения.


Пример использования Instant:
import java.time.Instant;

public class InstantExample {
public static void main(String[] args) {
// Получаем текущий момент времени
Instant now = Instant.now();
System.out.println("Текущий момент: " + now);

// Получаем количество секунд с эпохи Unix
long epochSecond = now.getEpochSecond();
System.out.println("Секунд с эпохи Unix: " + epochSecond);

// Получаем наносекунды
int nano = now.getNano();
System.out.println("Наносекунды: " + nano);
}
}


Плюсы Instant:
Высокая точность (до наносекунд).
Независимость от временных зон (всегда в UTC).
Удобен для хранения и передачи временных меток.


Минусы Instant:
Не подходит для работы с человеко-читаемыми датами и временем (например, "2023-10-15 14:30").
Требует конвертации для отображения в локальных временных зонах.


Использование TemporalAdjusters

TemporalAdjusters — это утилитный класс, который предоставляет готовые методы для выполнения сложных операций с датами. Например, он позволяет находить "следующий понедельник", "последний день месяца" и т.д.

Как работает TemporalAdjusters?

Каждый метод TemporalAdjusters возвращает объект TemporalAdjuster, который можно передать в метод with() классов LocalDate, LocalDateTime и других.

Пример использования TemporalAdjusters:
import java.time.DayOfWeek;
import java.time.LocalDate;
import java.time.temporal.TemporalAdjusters;

public class TemporalAdjustersExample {
public static void main(String[] args) {
// Текущая дата
LocalDate today = LocalDate.now();
System.out.println("Сегодня: " + today);

// Следующий понедельник
LocalDate nextMonday = today.with(TemporalAdjusters.next(DayOfWeek.MONDAY));
System.out.println("Следующий понедельник: " + nextMonday);

// Последний день месяца
LocalDate lastDayOfMonth = today.with(TemporalAdjusters.lastDayOfMonth());
System.out.println("Последний день месяца: " + lastDayOfMonth);
}
}


Плюсы TemporalAdjusters:
Упрощает сложные операции с датами.
Читаемый и понятный код.
Гибкость: можно создавать собственные TemporalAdjuster.


Минусы TemporalAdjusters:
Ограниченный набор стандартных методов (например, нет встроенной поддержки для "первого вторника месяца").
Требует понимания временных типов (LocalDate, LocalDateTime и т.д.).


#Java #Training #Medium #Date #Instant #TemporalAdjusters
Что выведет код?

public class Task220125_1 {
public static void main(String[] args) {
try {
System.out.println("A");
throw new RuntimeException("B");
} catch (RuntimeException e) {
System.
out.println(e.getMessage());
} finally {
System.out.println("C");
}
}
}


#Tasks
Сложный выбор... 🧐😵‍💫

https://t.me/Java_for_beginner_dev

#Mems
Please open Telegram to view this post
VIEW IN TELEGRAM
Работа с временными зонами

Почему важно хранить даты в UTC?

Универсальность: UTC — это стандарт, который не зависит от временных зон.
Согласованность: данные, хранящиеся в UTC, можно легко конвертировать в любую временную зону.
Избежание путаницы: исключаются ошибки, связанные с переходом на летнее время или разными временными зонами.


Пример конвертации из UTC в локальное время:
import java.time.Instant;
import java.time.ZoneId;
import java.time.ZonedDateTime;

public class UTCToLocalTime {
public static void main(String[] args) {
// Текущий момент в UTC
Instant now = Instant.now();
System.out.println("UTC: " + now);

// Конвертация в локальное время (например, для Москвы)
ZonedDateTime localTime = now.atZone(ZoneId.of("Europe/Moscow"));
System.out.println("Московское время: " + localTime);
}
}


Обработка временных зон в веб-приложениях

Как принимать время от пользователя и конвертировать его в UTC?
Принимайте время от пользователя с указанием его временной зоны.
Используйте ZonedDateTime для конвертации в UTC.

Пример:
import java.time.LocalDateTime;
import java.time.ZoneId;
import java.time.ZonedDateTime;

public class UserTimeToUTC {
public static void main(String[] args) {
// Время, введенное пользователем (например, 2023-10-15T14:30)
LocalDateTime userTime = LocalDateTime.parse("2023-10-15T14:30");

// Временная зона пользователя (например, Нью-Йорк)
ZoneId userZone = ZoneId.of("America/New_York");
ZonedDateTime userZonedTime = ZonedDateTime.of(userTime, userZone);

// Конвертация в UTC
ZonedDateTime utcTime = userZonedTime.withZoneSameInstant(ZoneId.of("UTC"));
System.out.println("UTC время: " + utcTime);
}
}


Проблемы с временными зонами

Неоднозначность времени при переходе на летнее время:
Например, в некоторых зонах время 2:30 может быть дважды (при переходе на зимнее время).

Решение:
Используйте ZonedDateTime и методы, которые учитывают такие случаи.

Разные форматы временных зон:
Например, "UTC+3" и "Europe/Moscow" могут означать одно и то же, но лучше использовать стандартные идентификаторы (например, "Europe/Moscow").

Ошибки при конвертации:
Убедитесь, что вы всегда знаете, в какой временной зоне находятся данные.

#Java #Training #Medium #Date #Best_Practicies
Please open Telegram to view this post
VIEW IN TELEGRAM
Please open Telegram to view this post
VIEW IN TELEGRAM
Основные аннотации для работы с датами в Spring

В Spring работа с датами часто связана с аннотациями, которые упрощают обработку временных данных в приложениях.

1. @DateTimeFormat

Эта аннотация используется для форматирования дат и времени в полях объектов или параметрах методов. Она позволяет указать, как строковое представление даты должно быть преобразовано в объект LocalDate, LocalDateTime и другие типы.

Пример использования:

public class Event {
@DateTimeFormat(pattern = "yyyy-MM-dd")
private LocalDate eventDate;

// Геттеры и сеттеры
}


Здесь поле eventDate будет автоматически преобразовано из строки формата yyyy-MM-dd в объект LocalDate.

Как это работает под капотом:
Spring использует DateTimeFormatAnnotationFormatterFactory для обработки аннотации. Этот фабричный класс создает форматтер, который преобразует строку в объект даты и наоборот.

Плюсы:
Упрощает преобразование строк в даты.
Поддерживает различные форматы дат.


Минусы:
Если формат строки не совпадает с указанным в pattern, возникает исключение.

2. @JsonFormat

Эта аннотация используется для управления сериализацией и десериализацией дат в JSON. Она часто применяется в REST API для корректного отображения дат.

Пример использования:
public class Event {
@JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss", timezone = "UTC")
private LocalDateTime eventDateTime;

// Геттеры и сеттеры
}


Здесь eventDateTime будет сериализован в JSON в формате yyyy-MM-dd HH:mm:ss и десериализован обратно в объект LocalDateTime.

Как это работает под капотом:
Jackson (библиотека для работы с JSON) использует эту аннотацию для настройки ObjectMapper. Форматтер преобразует дату в строку и обратно.

Плюсы:
Удобно для работы с REST API.
Поддерживает временные зоны.


Минусы:
Требует настройки формата для каждого поля.

3. @Temporal

Эта аннотация используется в JPA для указания типа временного значения (DATE, TIME, TIMESTAMP). Она помогает базе данных правильно интерпретировать даты.

Пример использования:
@Entity
public class Event {
@Temporal(TemporalType.TIMESTAMP)
private Date eventDate;

// Геттеры и сеттеры
}


Здесь eventDate будет сохранен в базе данных как TIMESTAMP.


Как это работает под капотом:
Hibernate (реализация JPA) использует эту аннотацию для определения типа SQL-столбца в базе данных.

Плюсы:
Упрощает работу с датами в базе данных.
Поддерживает различные типы временных данных.


Минусы:
Работает только с устаревшим типом java.util.Date.

#Java #Training #Spring #Date #DateTimeFormat #JsonFormat #Temporal
Что выведет код?

public class Task230125 {
public static void main(String[] args) {
int i = 0;
do {
System.out.print(i + " ");
i++;
if (i == 3) break;
} while (i < 5);
System.out.print(i);
}
}


#Tasks
Варианты ответа:
Anonymous Quiz
10%
0 1 2 3 4 5
10%
0 1 2 3 4
27%
0 1 2 2
53%
0 1 2 3
И ведь не откажешься 😂🥺

https://t.me/Java_for_beginner_dev

#Mems
Please open Telegram to view this post
VIEW IN TELEGRAM