Форматы даты по умолчанию:
По умолчанию Oracle выводит даты в формате DD-MON-YY, где YY - две последние цифры года:
select sysdate from dual;
02.07.14
При вставке в таблицу значений типа date, по умолчанию можно использовать литерал в формате
DD-MON-YYYY
(две цифры номера дня, три буквы месяца и четыре цифры года)
Например:
insert into t1 (d) values ('28-APR-1971');
или использовать ключевое слово DATE для передачи в базу литерала типа data в формате ANSI
YYYY-MM-DD
(четыре цифры года, две цифры месяца, две цифры номера дня)
Например:
insert into t1 (d) values ( DATE '1971-04-28');
Конвертация даты в строку:
TO_CHAR(дата, [формат])
примеры использования:
select to_char(sysdate) from dual;
02.07.14
select to_char(sysdate, 'DD') from dual; -- день
02
select to_char(sysdate, 'MONTH') from dual; --месяц
ИЮЛЬ
select to_char(sysdate, 'YYYY') from dual; -- год
2014
select to_char(sysdate, 'HH24:MI:SS') from dual; -- часы, минуты, секунды
16:58:31
select to_char(sysdate, 'DD MONTH YYYY HH24:MI:SS') from dual; -- комбинация параметров формата
02 ИЮЛЬ 2014 17:00:51
select to_char(sysdate, 'CC') from dual; -- двузначное столетие (век)
21
select to_char(sysdate - 1000000, 'SCC') from dual; -- двузначное столетие (век), со знаком минус до нашей эры
-08
select to_char(sysdate, 'Q') from dual; -- однозначный квартал года
3
Немного о стандарте ISO.
В стандарте ISO, год, относящийся к номеру недели ISO, может отличаться от календарного года.
Например:
1 января 1988 года попадает на 53-ю неделю ISO для 1987 года.
Неделя всегда начинается с понедельника и заканчивается воскресеньем.
Как связан год с номером недели по стандарту ISO:
Если 1 января падает на пятницу, субботу или воскресенье, то неделя, включающая 1 января,
считается последней неделей предыдущего года, потому что большинство дней этой недели
принадлежат предыдущему году.
Если 1 января падает на понедельник, вторник, среду или четверг, то эта неделя считается
первой неделей нового года, потому что большинство дней этой недели принадлежат новому году.
Например:
1 января 1991 падает на вторник, поэтому неделя с понедельника, 31 декабря 1990 по воскресенье, 6 января 1991 считается неделей 1.
Чтобы получить номер недели ISO, используйте маску формата 'IW' для номера недели и одну из масок вида 'IY' для года.
Пример:
select to_char( DATE '1991-01-01', 'YYYY WW') from dual; -- в обычном календарном формате
1991 01
select to_char( DATE '1991-01-01', 'IYYY IW') from dual; -- в формате по ISO
1991 01
в данном случае результаты совпадают.
Попробуем с другой датой:
select to_char( DATE '1988-01-01', 'YYYY WW') from dual; -- в обычном календарном формате
1988 01
select to_char( DATE '1988-01-01', 'IYYY WW') from dual; -- год в формате ISO
1987 01
select to_char( DATE '1988-01-01', 'IYYY IW') from dual; -- год и номер недели в формате ISO
1987 53
Как видим результаты разные.
При вставке в таблицу даты, рекомендуется указывать все четыре цифры года.
Если указать только две последние цифры года, то две первые цифры (столетие)
Oracle будет интерпретировать в зависимости от того, какой формат был использован при вводе.
Если использовать формат YY, то в качестве столетия будет использовано текущее столетие,
которое в настоящее время установлено на сервере.
select
to_char(to_date('28-04-14', 'DD-MM-YY'), 'DD-MM-YYYY'),
to_char(to_date('28-04-77', 'DD-MM-YY'), 'DD-MM-YYYY')
from dual;
28-04-2014 28-04-2077
Неважно какой год мы указали, столетие всегда будет текущее (т.е. 20)
Если использовать формат YYYY но при этом указать только две последние цифры года
то в качестве столетия Oracle подставит нули (т.е. 00)
select
to_char(to_date('28-04-14', 'DD-MM-YYYY'), 'DD-MM-YYYY'),
to_char(to_date('28-04-77', 'DD-MM-YYYY'), 'DD-MM-YYYY')
from dual;
28-04-0014 28-04-0077
Если использовать формат RR и указать только две последние цифры года, то две первые цифры (столетие)
Oracle будет вычислять по следующим правилам:
Если указанный год находится в интервале от 00 до 49 и текущий год тоже попадает в этот интервал,
то столетие будет текущим, но если при этом текуший год будет находится в интервале от 50 до 99,
то столетие при этом будет увеличено на 1 (текущее столетие + 1).
Если указанный год находится в интервале от 50 до 99 и текущий год тоже попадает в этот интервал,
то столетие будет текущим, но если при этом текуший год будет находится в интервале от 00 до 49,
то столетие при этом будет уменьшено на 1 (текущее столетие - 1).
select
to_char(to_date('28-04-14', 'DD-MM-RR'), 'DD-MM-YYYY'),
to_char(to_date('28-04-77', 'DD-MM-RR'), 'DD-MM-YYYY')
from dual;
28-04-2014 28-04-1977
Вобщем запомнить легко, если указанный год, больше текущего диапазона, значит столетие уменьшаем
и наоборот если указанный год, меньше текущего диапазона, значит столетие увеличиваем.
Интересно, а что будет если использовать формат RRRR, но при этом указать только две последние цифры года:
select
to_char(to_date('28-04-14', 'DD-MM-RRRR'), 'DD-MM-YYYY'),
to_char(to_date('28-04-77', 'DD-MM-RRRR'), 'DD-MM-YYYY')
from dual;
28-04-2014 28-04-1977
В качестве столетия Oracle не подставил нули, вывод аналогичен формату RR.
Для выделения первой цифры столетия в формате года можно использовать запятую:
select to_char(sysdate, 'Y,YYY') from dual; -- год с разделителем
2 014
Допустимые форматы года:
select to_char(sysdate, 'YYYY IYYY RRRR SYYYY Y,YYY YYY IYY YY IY RR Y I') from dual; -- год в различных форматах
2014 2014 2014 2014 2 014 014 014 14 14 14 4 4
А также год прописью:
select to_char(sysdate, 'YEAR') from dual; -- в верхнем регистре
TWENTY FOURTEEN
или
select to_char(sysdate, 'Year') from dual; -- каждое слово с большой буквы
Twenty Fourteen
Форматы месяца:
select to_char(sysdate, 'MM') from dual; -- двузначный номер месяца
10
select to_char(sysdate, 'MONTH') from dual; -- полное название в верхнем регистре
ОКТЯБРЬ
select to_char(sysdate, 'Month') from dual; -- полное название с большой буквы
Октябрь
select to_char(sysdate, 'MON') from dual; -- три первые буквы в верхнем регистре
ОКТ
select to_char(sysdate, 'Mon') from dual; -- три первые буквы с большой буквы
Окт
select to_char(sysdate, 'RM') from dual; -- римскими цифрами
X
Форматы недели:
select to_char(sysdate, 'WW') from dual; -- двузначный номер недели года
42
select to_char(sysdate, 'IW') from dual; -- двузначный номер недели года по ISO
42
select to_char(sysdate, 'W') from dual; -- однозначный номер недели месяца
3
Форматы дня:
select to_char(sysdate, 'DDD') from dual; -- трехзначный номер дня года
291
select to_char(sysdate, 'DD') from dual; -- двузначный номер дня месяца
18
select to_char(sysdate, 'D') from dual; -- однозначный номер дня недели
6
select to_char(sysdate, 'DAY') from dual; -- полное название дня в верхнем регистре
СУББОТА
select to_char(sysdate, 'Day') from dual; -- полное название дня с заглавной буквы
Суббота
select to_char(sysdate, 'DY') from dual; -- первые две буквы названия в верхнем регистре
СБ
select to_char(sysdate, 'Dy') from dual; -- первые две буквы названия с заглавной буквы
Сб
select to_char(sysdate, 'J') from dual; -- Юлианский день - число дней, прошедшее с 1 января 4713 г. до нашей эры
2456949
Формат часов:
select to_char(sysdate, 'HH24') from dual; -- двузначный номер часа в 24 часовом формате
04
select to_char(sysdate, 'HH24 PM') from dual; -- с суффиксом
04 AM
select to_char(sysdate, 'HH') from dual; -- двузначный номер часа в 12 часовом формате
04
select to_char(sysdate, 'HH PM') from dual; -- с суффиксом
04 AM
select to_char(sysdate, 'HH A.M.') from dual; -- с суффиксом
04 AM
Форматы минут:
select to_char(sysdate, 'MI') from dual; -- двузначное количество минут
19
Форматы секунд:
select to_char(sysdate, 'SS') from dual; -- двузначное количество секунд
18
Существует тип TIMESTAMP, который может хранить дробную часть секунд.
Необязательную точность представления секунд можно определить параметром FF[1..9]
Значение этого параметра по умолчанию равно 6 (справа от десятичной точки секунд можно поместить до 6 цифр)
При попытке поместить большее количество цифр в дробную часть секунд, значение дробной части будет округлено.
SELECT TO_CHAR(SYSTIMESTAMP, 'YYYY-MM-DD HH24:MI.SS.FF') FROM dual; -- шесть цифр после десятичной точки (по умолчанию)
2014-10-18 08:55.42.050000
SELECT TO_CHAR(SYSTIMESTAMP, 'YYYY-MM-DD HH24:MI.SS.FF3') FROM dual; -- три цифры после десятичной точки
2014-10-18 08:56.23.606
SELECT TO_CHAR(SYSTIMESTAMP, 'YYYY-MM-DD HH24:MI.SS.FF9') FROM dual; -- девять цифр после десятичной точки
2014-10-18 08:56.55.526000000
select to_char(sysdate, 'SSSSS') from dual; -- число секунд отсчитываемое от полуночи
32498
В отчетах statspack применяются следующие обозначения долей секунд:
second (s)
centisecond (cs) - 100th of a second
millisecond (ms) - 1,000th of a second
microsecond (us) - 1,000,000th of a second
Разделители:
Символы, позволяющие разделять аспекты дат и времени.
- / , . ; : или любой текст в кавычках "текст"
SELECT TO_CHAR(SYSDATE, 'YYYY-MM-DD HH24:MI.SS') FROM dual;
2014-10-18 14:30.43
SELECT TO_CHAR(SYSDATE, 'YYYY/MM/DD;HH24 "часов" MI "минут" SS "секунд"') FROM dual;
2014/10/18;14 часов 31 минут 18 секунд
Суффиксы:
AM или PM (A.M. или P.M.)
12-часовой формат исчисления времени предполагает разбиение 24 часов, составляющих сутки,
на два 12-часовых интервала, обозначаемых a.m. (лат. ante meridiem дословно — «до полудня»)
и p.m. (лат. post meridiem дословно — «после полудня»).
00:00 (полночь) 12:00 a.m.* (полночь)
12:00 (полдень) 12:00 p.m.* (полдень)
Проблемы в обозначениях полудня и полуночи:
Несмотря на наличие международного стандарта ISO 8601, 12 часов ночи и 12 часов дня обозначается в разных
странах по-разному. Это связано с тем, что в латинских словосочетаниях лат. ante meridiem и
лат. post meridiem слово meridiem означает буквально «середина дня» или «полдень»,
и нет однозначности между обозначением полудня как «12 a.m.» («12 ante meridiem»,
или «12 часов до середины дня») или как «12 p.m.» («12 post meridiem», или «12 часов после середины дня»).
С другой стороны, полночь также можно логично назвать «12 p.m.» (12 post meridiem,
12 часов после предыдущей середины дня) или «12 a.m.» (12 ante meridiem, 12 часов до следующей середины дня).
National Maritime Museum в Гринвиче рекомендует обозначать эти временные моменты как «12 дня» и «12 ночи».
То же советует и The American Heritage Dictionary of the English Language. Многие руководства по стилю,
принятые в США, предлагают «полночь» заменять на «11:59 p.m.», если мы хотим обозначить конец дня,
и «12:01 a.m.», если мы хотим обозначить начало следующего дня.
SELECT TO_CHAR(SYSDATE, 'YYYY-MM-DD HH24:MI.SS AM') FROM dual;
2014-10-18 14:53.58 PM
AD или BC (A.D. или B.C.)
AD - нашей эры
BC - до нашей эры
SELECT TO_CHAR(SYSDATE, 'YYYY-MM-DD HH24:MI.SS BC') FROM dual;
2014-10-18 15:00.25 Н.З.
TH - суффикс для чисел
SELECT TO_CHAR(SYSDATE, 'DDTH') FROM dual;
18TH
SELECT TO_CHAR(SYSDATE, 'ddTH') FROM dual;
18th
SELECT TO_CHAR(SYSDATE, 'mmTH') FROM dual;
10th
SELECT TO_CHAR(SYSDATE, 'YYYYTH') FROM dual;
2014TH
SELECT TO_CHAR(SYSDATE, 'yyyyTH-MMTH-DDTH HH24TH:miTH.SSTH BC') FROM dual;
2014th-10TH-18TH 17TH:56th.52ND Н.З.
SP - числовые значения записываются словами
SELECT TO_CHAR(SYSDATE, 'DDSP') FROM dual;
EIGHTEEN
SELECT TO_CHAR(SYSDATE, 'ddSP') FROM dual;
eighteen
SELECT TO_CHAR(SYSDATE, 'mmTHSP') FROM dual;
tenth
SELECT TO_CHAR(SYSDATE, 'mmSP') FROM dual;
ten
SELECT TO_CHAR(SYSDATE, 'YYYYTHSP') FROM dual;
TWO THOUSAND FOURTEENTH
SELECT TO_CHAR(SYSDATE, 'YYYYSP') FROM dual;
TWO THOUSAND FOURTEEN
EE - Полное название эпохи для японского календаря, календаря КНР и буддийского календаря.
E - Сокращенное название эпохи
select TO_DATE('H19-01-01' , 'EYY-MM-DD' , 'NLS_CALENDAR=''JAPANESE IMPERIAL''') e_date
from dual;
01.01.07
select TO_DATE('平成19-01-01' , 'EEYY-MM-DD' , 'NLS_CALENDAR=''JAPANESE IMPERIAL''') ee_date
from dual;
01.01.07
Часовые пояса:
В Oracle с версии 9i появилась возможность использовать различные часовые пояса.
Часовой пояс - это смещение от времени по Гринвичу(GMT).
Но теперь оно называется Всемирное скоординированное время(UTC).
Часовой пояс определяется либо как смещение относительно UTC, либо по имени региона (названию часового пояса).
Получить названия часовых поясов можно так:
select * from v$timezone_names;
Africa/Abidjan LMT
Africa/Abidjan GMT
Africa/Accra LMT
Africa/Accra GMT
Africa/Accra GHST
Africa/Addis_Ababa LMT
Africa/Addis_Ababa ADMT
Africa/Addis_Ababa EAT
Africa/Algiers LMT
Africa/Algiers PMT
Africa/Algiers WET
........
При определении смещения используется формат HH:MI с префиксом в виде знака + или -
+/- HH:MI
Посмотрим какое смещение относительно UTC установлено в нашей БД:
select dbtimezone from dual;
+04:00
(меняется параметром time_zone в spfile.ora)
Часовой пояс сеанса можно определить так:
select sessiontimezone from dual;
Europe/Moscow
Его легко можно поменять на время сеанса:
alter session set time_zone = 'PST';
select sessiontimezone from dual;
PST
Стандартное Тихоокеанское время PST отстает от UTC на восемь часов.
Восточное стандартное время EST отстает от UTC на пять часов.
Текущую дату для сеанса в локальном часовом поясе можно определить так:
select current_date from dual;
18.10.14
select to_char(current_date, 'YYYY-MM-DD HH24:MI.SS' ) from dual;
2014-10-18 11:03.56
sysdate() - возвращает значение даты и времени, установленных в ОС компьютера, на котором размещена БД.
current_date() - возвращает значение даты и времени для часового пояса вашего сеанса.
Для любого часового пояса можно найти величину смещения с помощью функции tz_offset().
select tz_offset('PST') from dual;
-07:00
select tz_offset('Europe/Moscow') from dual;
+04:00
TZH - время в часах часового пояса
TZM - минуты часового пояса
TZR - регион часового пояса
TZD - часовой пояс с информацией о переходе на летнее время
Tип TIMESTAMP, в отличие от типа DATE, может хранить информацию о часовых поясах.
select to_char(SYSTIMESTAMP, 'TZH:TZM') from dual;
+04:00
select to_char(SYSTIMESTAMP, 'TZR') from dual;
+04:00
select to_char(SYSTIMESTAMP, 'TZD') from dual;
null
select to_char(SYSTIMESTAMP, 'HH:MI:SS.FFTZH:TZM') from dual;
10:51:57.733000+04:00
select to_char(SYSTIMESTAMP, 'YYYY-MM-DD HH:MI:SS TZH:TZM') from dual;
2014-10-18 10:52:19 +04:00
select to_char(SYSTIMESTAMP, 'YYYY-MM-DD HH:MI:SS.FF AM TZH:TZM TZR TZD') from dual;
2014-10-18 10:52:31.802000 PM +04:00 +04:00
Чтобы конвертировать дату-время из одного часового пояса к другому,
можно воспользоваться функцией NEW_TIME().
select to_char( new_time( to_date( '28-04-1971 10:30' , 'DD-MM-YYYY HH24:MI'), 'PST' , 'EST'), 'DD-MM-YYYY HH24:MI')
from dual;
28-04-1971 13:30
Конвертация строки в тип дата-время.
Функцию TO_DATE(x [, формат])
можно использовать для конвертирования строки x в тип дата-время.
Если строка формата опущена, то дата должна быть представлена в формате по умолчанию:
DD-MON-YYYY или DD-MON-YY
(Вообще формат даты по умолчанию определяет параметр БД NLS_DATE_FORMAT)
alter session set NLS_DATE_LANGUAGE = 'AMERICAN' ;
alter session set NLS_DATE_FORMAT = 'SYYYY-MM-DD' ;
alter session set NLS_TIMESTAMP_FORMAT = 'SYYYY-MM-DD HH24:MI:SS' ;
alter session set NLS_TIMESTAMP_TZ_FORMAT = 'SYYYY-MM-DD HH24:MI:SS TZH:TZM' ;
Например:
alter session set NLS_DATE_LANGUAGE = 'AMERICAN';
alter session set NLS_DATE_FORMAT = 'DD-MON-RRRR';
select to_date('28-APR-1971'), to_date('28-APR-71') from dual;
28-АПР-1971 28-АПР-1971
Можно и явно задать формат
select to_date('April 28, 1971' , 'MONTH DD, YYYY') from dual;
28-АПР-1971
select to_date('28-APR-1971 18:30:55' , 'DD-MON-YYYY HH24:MI:SS') from dual;
28-АПР-1971
Совместное использование to_date() и to_char()
select to_char(to_date('28-APR-1971 18:30:55' , 'DD-MON-YYYY HH24:MI:SS') , 'HH24:MI:SS') from dual;
18:30:55
Формат даты по умолчанию, можно использовать и при вставке строк в таблицу:
alter session set NLS_DATE_FORMAT = 'DD-MON-YYYY';
insert into t1 ( id, bday ) values (1, '28-APR-1971' );
NLS - параметры:
National language_support (До Oracle9i)
Globalisation support (Начиная с Oracle9i)
NLS_LANG = язык_территория.кодировка
Кодировка устанавливается только в переменных окружения!
Язык - RUSSIAN, AMERICAN
1) Язык вывода сообщений об ошибках
2) на каком языке выводить названия месяцев и дней недели
(Если явно не задан параметр NLS_DATE_LANGUAGE)
SELECT * FROM v$nls_valid_values
WHERE parameter = 'LANGUAGE'
ORDER BY value
Территория:
CIS - СНГ
1. первый день недели
2. символ национальной валюты
(Если явно не задан параметр NLS_CURRENCY)
3. Десятичный и групповой разделители чисел
SELECT * FROM v$nls_valid_values
WHERE parameter = 'TERRITORY'
ORDER BY value
Кодировка:
SELECT * FROM v$nls_valid_values
WHERE parameter = 'CHARACTERSET'
-- Русский язык, Кириллица
AND (value LIKE 'CL%'
OR
value LIKE 'RU%')
ORDER BY value
WE8ISO8859P1 - Западная Европа
NLS_LANG = AMERICAN_CIS.CL8MSWIN1251
NLS_LANG = AMERICAN_AMERICA.RU8PC866
NLS_LANG = RUSSIAN_CIS.CL8ISO8859P1
Какие есть параметры NLS?
SELECT * FROM nls_session_parameters
PARAMETER VALUE
================ ==========
NLS_LANGUAGE=AMERICAN
NLS_TERRITORY=CIS
-- Символ нац. валюты
NLS_CURRENCY='р.'
-- Символ нац. валюты по стандарту ISO
NLS_ISO_CURRENCY='CIS'
-- Десятичный разделитель и разделитель групп
NLS_NUMERIC_CHARACTERS=', '
-- Календарь
NLS_CALENDAR=GREGORIAN
-- Формат ввода и вывода даты по-умолчанию
NLS_DATE_FORMAT='DD.MM.RR'
-- Язык для вывода названий месяцев и дней недели
NLS_DATE_LANGUAGE='AMERICAN'
-- Тип Сортировки
NLS_SORT=BINARY
-- ??? (нет описания)
NLS_TIME_FORMAT='HH24:MI:SSXFF'
-- Формат ввода и вывода даты типа TIMESTAMP по-умолчанию
NLS_TIMESTAMP_FORMAT='DD.MM.RR HH24:MI:SSXFF'
-- ??? (нет описания)
NLS_TIME_TZ_FORMAT='HH24:MI:SSXFF TZR'
-- Формат ввода и вывода даты типа TIMESTAMP с временнОй зоной по-умолчанию
NLS_TIMESTAMP_TZ_FORMAT='DD.MM.RR HH24:MI:SSXFF TZR'
-- Замещает символ нац. валюты, установленный по умолчанию параметром NLS_TERRITORY
NLS_DUAL_CURRENCY='р.'
-- Как сравнивать строки BINARY или ASCII (по правилам нац. алфавита)
NLS_COMP=BINARY
-- CHAR по умолчанию в байтах или в символах
NLS_LENGTH_SEMANTICS=BYTE
-- NLS_NCHAR_CONV_EXCP determines whether an error is reported when there is
-- data loss during an implicit OR explicit CHARACTER TYPE conversion.
-- The DEFAULT value results IN no error being reported.
NLS_NCHAR_CONV_EXCP=FALSE
Как можно устанавливать значения параметров NLS?
1. В системном реестре Windows
2. Установить переменные окружения
Для Windows (в bat-файле)
SET NLS_DATE_LANGUAGE=RUSSIAN
SET NLS_LANG=AMERICAN_CIS.CL8MSWIN1251
sqlplus ....
3. ALTER SESSION SET
NLS_DATE_LANGUAGE=RUSSIAN
NLS_DATE_FORMAT='DD.MM.YYYY';
SELECT TO_CHAR(SYSDATE, 'Month day')
FROM dual
Посмотреть nls-параметры сессии, базы данных и инстанса можно так:
select * from
(select 'SESSION' SCOPE,s.* from nls_session_parameters s
union
select 'DATABASE' SCOPE,d.* from nls_database_parameters d
union
select 'INSTANCE' SCOPE,i.* from nls_instance_parameters i
) a
pivot (LISTAGG(VALUE) WITHIN GROUP (ORDER BY SCOPE)
FOR SCOPE
in ('SESSION' as "SESSION",'DATABASE' as "DATABASE",'INSTANCE' as "INSTANCE"));
Функции для работы с типом data.
ADD_MONTHS(data, n)
Позволяет добавить к дате целое количество месяцев (или отнять, если n отрицательное)
SELECT ADD_MONTHS('28.04.1971' , 13) FROM DUAL; -- Добавить 13 месяцев
28.05.72
SELECT ADD_MONTHS('28.04.1971' , -12) FROM DUAL; -- Отнять 12 месяцев
28.04.70
В функцию ADD_MONTHS() можно передать и дату и время:
SELECT ADD_MONTHS(TO_DATE('28.04.1971 15:23:44', 'DD.MM.YYYY HH24:MI:SS') , 5) FROM DUAL;-- Добавить 5 месяцев
28.09.71
SELECT TO_CHAR(ADD_MONTHS(TO_DATE('28.04.1971 15:23:44', 'DD.MM.YYYY HH24:MI:SS') , 5), 'DD.MM.YYYY HH24:MI:SS') FROM DUAL;-- Добавить 5 месяцев
28.09.1971 15:23:44
SELECT
TO_CHAR(ADD_MONTHS(TO_DATE('31.10.2001 21:08:50', 'fxDD.MM.YYYY HH24:MI:SS')
, 4) -- Добавить 4 месяца
, 'DD.MM.YYYY HH24:MI:SS')
FROM DUAL;
28.02.2002 21:08:50 - это последний день месяца
SELECT
TO_CHAR(ADD_MONTHS(TO_DATE('28.02.2001 21:08:50', 'fxDD.MM.YYYY HH24:MI:SS')
, -4) -- Отнять 4 месяца
, 'DD.MM.YYYY HH24:MI:SS')
FROM DUAL;
31.10.2000 21:08:50 - это последний день месяца
По умолчанию Oracle выводит даты в формате DD-MON-YY, где YY - две последние цифры года:
select sysdate from dual;
02.07.14
При вставке в таблицу значений типа date, по умолчанию можно использовать литерал в формате
DD-MON-YYYY
(две цифры номера дня, три буквы месяца и четыре цифры года)
Например:
insert into t1 (d) values ('28-APR-1971');
или использовать ключевое слово DATE для передачи в базу литерала типа data в формате ANSI
YYYY-MM-DD
(четыре цифры года, две цифры месяца, две цифры номера дня)
Например:
insert into t1 (d) values ( DATE '1971-04-28');
Конвертация даты в строку:
TO_CHAR(дата, [формат])
примеры использования:
select to_char(sysdate) from dual;
02.07.14
select to_char(sysdate, 'DD') from dual; -- день
02
select to_char(sysdate, 'MONTH') from dual; --месяц
ИЮЛЬ
select to_char(sysdate, 'YYYY') from dual; -- год
2014
select to_char(sysdate, 'HH24:MI:SS') from dual; -- часы, минуты, секунды
16:58:31
select to_char(sysdate, 'DD MONTH YYYY HH24:MI:SS') from dual; -- комбинация параметров формата
02 ИЮЛЬ 2014 17:00:51
select to_char(sysdate, 'CC') from dual; -- двузначное столетие (век)
21
select to_char(sysdate - 1000000, 'SCC') from dual; -- двузначное столетие (век), со знаком минус до нашей эры
-08
select to_char(sysdate, 'Q') from dual; -- однозначный квартал года
3
Немного о стандарте ISO.
В стандарте ISO, год, относящийся к номеру недели ISO, может отличаться от календарного года.
Например:
1 января 1988 года попадает на 53-ю неделю ISO для 1987 года.
Неделя всегда начинается с понедельника и заканчивается воскресеньем.
Как связан год с номером недели по стандарту ISO:
Если 1 января падает на пятницу, субботу или воскресенье, то неделя, включающая 1 января,
считается последней неделей предыдущего года, потому что большинство дней этой недели
принадлежат предыдущему году.
Если 1 января падает на понедельник, вторник, среду или четверг, то эта неделя считается
первой неделей нового года, потому что большинство дней этой недели принадлежат новому году.
Например:
1 января 1991 падает на вторник, поэтому неделя с понедельника, 31 декабря 1990 по воскресенье, 6 января 1991 считается неделей 1.
Чтобы получить номер недели ISO, используйте маску формата 'IW' для номера недели и одну из масок вида 'IY' для года.
Пример:
select to_char( DATE '1991-01-01', 'YYYY WW') from dual; -- в обычном календарном формате
1991 01
select to_char( DATE '1991-01-01', 'IYYY IW') from dual; -- в формате по ISO
1991 01
в данном случае результаты совпадают.
Попробуем с другой датой:
select to_char( DATE '1988-01-01', 'YYYY WW') from dual; -- в обычном календарном формате
1988 01
select to_char( DATE '1988-01-01', 'IYYY WW') from dual; -- год в формате ISO
1987 01
select to_char( DATE '1988-01-01', 'IYYY IW') from dual; -- год и номер недели в формате ISO
1987 53
Как видим результаты разные.
При вставке в таблицу даты, рекомендуется указывать все четыре цифры года.
Если указать только две последние цифры года, то две первые цифры (столетие)
Oracle будет интерпретировать в зависимости от того, какой формат был использован при вводе.
Если использовать формат YY, то в качестве столетия будет использовано текущее столетие,
которое в настоящее время установлено на сервере.
select
to_char(to_date('28-04-14', 'DD-MM-YY'), 'DD-MM-YYYY'),
to_char(to_date('28-04-77', 'DD-MM-YY'), 'DD-MM-YYYY')
from dual;
28-04-2014 28-04-2077
Неважно какой год мы указали, столетие всегда будет текущее (т.е. 20)
Если использовать формат YYYY но при этом указать только две последние цифры года
то в качестве столетия Oracle подставит нули (т.е. 00)
select
to_char(to_date('28-04-14', 'DD-MM-YYYY'), 'DD-MM-YYYY'),
to_char(to_date('28-04-77', 'DD-MM-YYYY'), 'DD-MM-YYYY')
from dual;
28-04-0014 28-04-0077
Если использовать формат RR и указать только две последние цифры года, то две первые цифры (столетие)
Oracle будет вычислять по следующим правилам:
Если указанный год находится в интервале от 00 до 49 и текущий год тоже попадает в этот интервал,
то столетие будет текущим, но если при этом текуший год будет находится в интервале от 50 до 99,
то столетие при этом будет увеличено на 1 (текущее столетие + 1).
Если указанный год находится в интервале от 50 до 99 и текущий год тоже попадает в этот интервал,
то столетие будет текущим, но если при этом текуший год будет находится в интервале от 00 до 49,
то столетие при этом будет уменьшено на 1 (текущее столетие - 1).
select
to_char(to_date('28-04-14', 'DD-MM-RR'), 'DD-MM-YYYY'),
to_char(to_date('28-04-77', 'DD-MM-RR'), 'DD-MM-YYYY')
from dual;
28-04-2014 28-04-1977
Вобщем запомнить легко, если указанный год, больше текущего диапазона, значит столетие уменьшаем
и наоборот если указанный год, меньше текущего диапазона, значит столетие увеличиваем.
Интересно, а что будет если использовать формат RRRR, но при этом указать только две последние цифры года:
select
to_char(to_date('28-04-14', 'DD-MM-RRRR'), 'DD-MM-YYYY'),
to_char(to_date('28-04-77', 'DD-MM-RRRR'), 'DD-MM-YYYY')
from dual;
28-04-2014 28-04-1977
В качестве столетия Oracle не подставил нули, вывод аналогичен формату RR.
Для выделения первой цифры столетия в формате года можно использовать запятую:
select to_char(sysdate, 'Y,YYY') from dual; -- год с разделителем
2 014
Допустимые форматы года:
select to_char(sysdate, 'YYYY IYYY RRRR SYYYY Y,YYY YYY IYY YY IY RR Y I') from dual; -- год в различных форматах
2014 2014 2014 2014 2 014 014 014 14 14 14 4 4
А также год прописью:
select to_char(sysdate, 'YEAR') from dual; -- в верхнем регистре
TWENTY FOURTEEN
или
select to_char(sysdate, 'Year') from dual; -- каждое слово с большой буквы
Twenty Fourteen
Форматы месяца:
select to_char(sysdate, 'MM') from dual; -- двузначный номер месяца
10
select to_char(sysdate, 'MONTH') from dual; -- полное название в верхнем регистре
ОКТЯБРЬ
select to_char(sysdate, 'Month') from dual; -- полное название с большой буквы
Октябрь
select to_char(sysdate, 'MON') from dual; -- три первые буквы в верхнем регистре
ОКТ
select to_char(sysdate, 'Mon') from dual; -- три первые буквы с большой буквы
Окт
select to_char(sysdate, 'RM') from dual; -- римскими цифрами
X
Форматы недели:
select to_char(sysdate, 'WW') from dual; -- двузначный номер недели года
42
select to_char(sysdate, 'IW') from dual; -- двузначный номер недели года по ISO
42
select to_char(sysdate, 'W') from dual; -- однозначный номер недели месяца
3
Форматы дня:
select to_char(sysdate, 'DDD') from dual; -- трехзначный номер дня года
291
select to_char(sysdate, 'DD') from dual; -- двузначный номер дня месяца
18
select to_char(sysdate, 'D') from dual; -- однозначный номер дня недели
6
select to_char(sysdate, 'DAY') from dual; -- полное название дня в верхнем регистре
СУББОТА
select to_char(sysdate, 'Day') from dual; -- полное название дня с заглавной буквы
Суббота
select to_char(sysdate, 'DY') from dual; -- первые две буквы названия в верхнем регистре
СБ
select to_char(sysdate, 'Dy') from dual; -- первые две буквы названия с заглавной буквы
Сб
select to_char(sysdate, 'J') from dual; -- Юлианский день - число дней, прошедшее с 1 января 4713 г. до нашей эры
2456949
Формат часов:
select to_char(sysdate, 'HH24') from dual; -- двузначный номер часа в 24 часовом формате
04
select to_char(sysdate, 'HH24 PM') from dual; -- с суффиксом
04 AM
select to_char(sysdate, 'HH') from dual; -- двузначный номер часа в 12 часовом формате
04
select to_char(sysdate, 'HH PM') from dual; -- с суффиксом
04 AM
select to_char(sysdate, 'HH A.M.') from dual; -- с суффиксом
04 AM
Форматы минут:
select to_char(sysdate, 'MI') from dual; -- двузначное количество минут
19
Форматы секунд:
select to_char(sysdate, 'SS') from dual; -- двузначное количество секунд
18
Существует тип TIMESTAMP, который может хранить дробную часть секунд.
Необязательную точность представления секунд можно определить параметром FF[1..9]
Значение этого параметра по умолчанию равно 6 (справа от десятичной точки секунд можно поместить до 6 цифр)
При попытке поместить большее количество цифр в дробную часть секунд, значение дробной части будет округлено.
SELECT TO_CHAR(SYSTIMESTAMP, 'YYYY-MM-DD HH24:MI.SS.FF') FROM dual; -- шесть цифр после десятичной точки (по умолчанию)
2014-10-18 08:55.42.050000
SELECT TO_CHAR(SYSTIMESTAMP, 'YYYY-MM-DD HH24:MI.SS.FF3') FROM dual; -- три цифры после десятичной точки
2014-10-18 08:56.23.606
SELECT TO_CHAR(SYSTIMESTAMP, 'YYYY-MM-DD HH24:MI.SS.FF9') FROM dual; -- девять цифр после десятичной точки
2014-10-18 08:56.55.526000000
select to_char(sysdate, 'SSSSS') from dual; -- число секунд отсчитываемое от полуночи
32498
В отчетах statspack применяются следующие обозначения долей секунд:
second (s)
centisecond (cs) - 100th of a second
millisecond (ms) - 1,000th of a second
microsecond (us) - 1,000,000th of a second
Разделители:
Символы, позволяющие разделять аспекты дат и времени.
- / , . ; : или любой текст в кавычках "текст"
SELECT TO_CHAR(SYSDATE, 'YYYY-MM-DD HH24:MI.SS') FROM dual;
2014-10-18 14:30.43
SELECT TO_CHAR(SYSDATE, 'YYYY/MM/DD;HH24 "часов" MI "минут" SS "секунд"') FROM dual;
2014/10/18;14 часов 31 минут 18 секунд
Суффиксы:
AM или PM (A.M. или P.M.)
12-часовой формат исчисления времени предполагает разбиение 24 часов, составляющих сутки,
на два 12-часовых интервала, обозначаемых a.m. (лат. ante meridiem дословно — «до полудня»)
и p.m. (лат. post meridiem дословно — «после полудня»).
00:00 (полночь) 12:00 a.m.* (полночь)
12:00 (полдень) 12:00 p.m.* (полдень)
Проблемы в обозначениях полудня и полуночи:
Несмотря на наличие международного стандарта ISO 8601, 12 часов ночи и 12 часов дня обозначается в разных
странах по-разному. Это связано с тем, что в латинских словосочетаниях лат. ante meridiem и
лат. post meridiem слово meridiem означает буквально «середина дня» или «полдень»,
и нет однозначности между обозначением полудня как «12 a.m.» («12 ante meridiem»,
или «12 часов до середины дня») или как «12 p.m.» («12 post meridiem», или «12 часов после середины дня»).
С другой стороны, полночь также можно логично назвать «12 p.m.» (12 post meridiem,
12 часов после предыдущей середины дня) или «12 a.m.» (12 ante meridiem, 12 часов до следующей середины дня).
National Maritime Museum в Гринвиче рекомендует обозначать эти временные моменты как «12 дня» и «12 ночи».
То же советует и The American Heritage Dictionary of the English Language. Многие руководства по стилю,
принятые в США, предлагают «полночь» заменять на «11:59 p.m.», если мы хотим обозначить конец дня,
и «12:01 a.m.», если мы хотим обозначить начало следующего дня.
SELECT TO_CHAR(SYSDATE, 'YYYY-MM-DD HH24:MI.SS AM') FROM dual;
2014-10-18 14:53.58 PM
AD или BC (A.D. или B.C.)
AD - нашей эры
BC - до нашей эры
SELECT TO_CHAR(SYSDATE, 'YYYY-MM-DD HH24:MI.SS BC') FROM dual;
2014-10-18 15:00.25 Н.З.
TH - суффикс для чисел
SELECT TO_CHAR(SYSDATE, 'DDTH') FROM dual;
18TH
SELECT TO_CHAR(SYSDATE, 'ddTH') FROM dual;
18th
SELECT TO_CHAR(SYSDATE, 'mmTH') FROM dual;
10th
SELECT TO_CHAR(SYSDATE, 'YYYYTH') FROM dual;
2014TH
SELECT TO_CHAR(SYSDATE, 'yyyyTH-MMTH-DDTH HH24TH:miTH.SSTH BC') FROM dual;
2014th-10TH-18TH 17TH:56th.52ND Н.З.
SP - числовые значения записываются словами
SELECT TO_CHAR(SYSDATE, 'DDSP') FROM dual;
EIGHTEEN
SELECT TO_CHAR(SYSDATE, 'ddSP') FROM dual;
eighteen
SELECT TO_CHAR(SYSDATE, 'mmTHSP') FROM dual;
tenth
SELECT TO_CHAR(SYSDATE, 'mmSP') FROM dual;
ten
SELECT TO_CHAR(SYSDATE, 'YYYYTHSP') FROM dual;
TWO THOUSAND FOURTEENTH
SELECT TO_CHAR(SYSDATE, 'YYYYSP') FROM dual;
TWO THOUSAND FOURTEEN
EE - Полное название эпохи для японского календаря, календаря КНР и буддийского календаря.
E - Сокращенное название эпохи
select TO_DATE('H19-01-01' , 'EYY-MM-DD' , 'NLS_CALENDAR=''JAPANESE IMPERIAL''') e_date
from dual;
01.01.07
select TO_DATE('平成19-01-01' , 'EEYY-MM-DD' , 'NLS_CALENDAR=''JAPANESE IMPERIAL''') ee_date
from dual;
01.01.07
Часовые пояса:
В Oracle с версии 9i появилась возможность использовать различные часовые пояса.
Часовой пояс - это смещение от времени по Гринвичу(GMT).
Но теперь оно называется Всемирное скоординированное время(UTC).
Часовой пояс определяется либо как смещение относительно UTC, либо по имени региона (названию часового пояса).
Получить названия часовых поясов можно так:
select * from v$timezone_names;
Africa/Abidjan LMT
Africa/Abidjan GMT
Africa/Accra LMT
Africa/Accra GMT
Africa/Accra GHST
Africa/Addis_Ababa LMT
Africa/Addis_Ababa ADMT
Africa/Addis_Ababa EAT
Africa/Algiers LMT
Africa/Algiers PMT
Africa/Algiers WET
........
При определении смещения используется формат HH:MI с префиксом в виде знака + или -
+/- HH:MI
Посмотрим какое смещение относительно UTC установлено в нашей БД:
select dbtimezone from dual;
+04:00
(меняется параметром time_zone в spfile.ora)
Часовой пояс сеанса можно определить так:
select sessiontimezone from dual;
Europe/Moscow
Его легко можно поменять на время сеанса:
alter session set time_zone = 'PST';
select sessiontimezone from dual;
PST
Стандартное Тихоокеанское время PST отстает от UTC на восемь часов.
Восточное стандартное время EST отстает от UTC на пять часов.
Текущую дату для сеанса в локальном часовом поясе можно определить так:
select current_date from dual;
18.10.14
select to_char(current_date, 'YYYY-MM-DD HH24:MI.SS' ) from dual;
2014-10-18 11:03.56
sysdate() - возвращает значение даты и времени, установленных в ОС компьютера, на котором размещена БД.
current_date() - возвращает значение даты и времени для часового пояса вашего сеанса.
Для любого часового пояса можно найти величину смещения с помощью функции tz_offset().
select tz_offset('PST') from dual;
-07:00
select tz_offset('Europe/Moscow') from dual;
+04:00
TZH - время в часах часового пояса
TZM - минуты часового пояса
TZR - регион часового пояса
TZD - часовой пояс с информацией о переходе на летнее время
Tип TIMESTAMP, в отличие от типа DATE, может хранить информацию о часовых поясах.
select to_char(SYSTIMESTAMP, 'TZH:TZM') from dual;
+04:00
select to_char(SYSTIMESTAMP, 'TZR') from dual;
+04:00
select to_char(SYSTIMESTAMP, 'TZD') from dual;
null
select to_char(SYSTIMESTAMP, 'HH:MI:SS.FFTZH:TZM') from dual;
10:51:57.733000+04:00
select to_char(SYSTIMESTAMP, 'YYYY-MM-DD HH:MI:SS TZH:TZM') from dual;
2014-10-18 10:52:19 +04:00
select to_char(SYSTIMESTAMP, 'YYYY-MM-DD HH:MI:SS.FF AM TZH:TZM TZR TZD') from dual;
2014-10-18 10:52:31.802000 PM +04:00 +04:00
Чтобы конвертировать дату-время из одного часового пояса к другому,
можно воспользоваться функцией NEW_TIME().
select to_char( new_time( to_date( '28-04-1971 10:30' , 'DD-MM-YYYY HH24:MI'), 'PST' , 'EST'), 'DD-MM-YYYY HH24:MI')
from dual;
28-04-1971 13:30
Конвертация строки в тип дата-время.
Функцию TO_DATE(x [, формат])
можно использовать для конвертирования строки x в тип дата-время.
Если строка формата опущена, то дата должна быть представлена в формате по умолчанию:
DD-MON-YYYY или DD-MON-YY
(Вообще формат даты по умолчанию определяет параметр БД NLS_DATE_FORMAT)
alter session set NLS_DATE_LANGUAGE = 'AMERICAN' ;
alter session set NLS_DATE_FORMAT = 'SYYYY-MM-DD' ;
alter session set NLS_TIMESTAMP_FORMAT = 'SYYYY-MM-DD HH24:MI:SS' ;
alter session set NLS_TIMESTAMP_TZ_FORMAT = 'SYYYY-MM-DD HH24:MI:SS TZH:TZM' ;
Например:
alter session set NLS_DATE_LANGUAGE = 'AMERICAN';
alter session set NLS_DATE_FORMAT = 'DD-MON-RRRR';
select to_date('28-APR-1971'), to_date('28-APR-71') from dual;
28-АПР-1971 28-АПР-1971
Можно и явно задать формат
select to_date('April 28, 1971' , 'MONTH DD, YYYY') from dual;
28-АПР-1971
select to_date('28-APR-1971 18:30:55' , 'DD-MON-YYYY HH24:MI:SS') from dual;
28-АПР-1971
Совместное использование to_date() и to_char()
select to_char(to_date('28-APR-1971 18:30:55' , 'DD-MON-YYYY HH24:MI:SS') , 'HH24:MI:SS') from dual;
18:30:55
Формат даты по умолчанию, можно использовать и при вставке строк в таблицу:
alter session set NLS_DATE_FORMAT = 'DD-MON-YYYY';
insert into t1 ( id, bday ) values (1, '28-APR-1971' );
NLS - параметры:
National language_support (До Oracle9i)
Globalisation support (Начиная с Oracle9i)
NLS_LANG = язык_территория.кодировка
Кодировка устанавливается только в переменных окружения!
Язык - RUSSIAN, AMERICAN
1) Язык вывода сообщений об ошибках
2) на каком языке выводить названия месяцев и дней недели
(Если явно не задан параметр NLS_DATE_LANGUAGE)
SELECT * FROM v$nls_valid_values
WHERE parameter = 'LANGUAGE'
ORDER BY value
Территория:
CIS - СНГ
1. первый день недели
2. символ национальной валюты
(Если явно не задан параметр NLS_CURRENCY)
3. Десятичный и групповой разделители чисел
SELECT * FROM v$nls_valid_values
WHERE parameter = 'TERRITORY'
ORDER BY value
Кодировка:
SELECT * FROM v$nls_valid_values
WHERE parameter = 'CHARACTERSET'
-- Русский язык, Кириллица
AND (value LIKE 'CL%'
OR
value LIKE 'RU%')
ORDER BY value
WE8ISO8859P1 - Западная Европа
NLS_LANG = AMERICAN_CIS.CL8MSWIN1251
NLS_LANG = AMERICAN_AMERICA.RU8PC866
NLS_LANG = RUSSIAN_CIS.CL8ISO8859P1
Какие есть параметры NLS?
SELECT * FROM nls_session_parameters
PARAMETER VALUE
================ ==========
NLS_LANGUAGE=AMERICAN
NLS_TERRITORY=CIS
-- Символ нац. валюты
NLS_CURRENCY='р.'
-- Символ нац. валюты по стандарту ISO
NLS_ISO_CURRENCY='CIS'
-- Десятичный разделитель и разделитель групп
NLS_NUMERIC_CHARACTERS=', '
-- Календарь
NLS_CALENDAR=GREGORIAN
-- Формат ввода и вывода даты по-умолчанию
NLS_DATE_FORMAT='DD.MM.RR'
-- Язык для вывода названий месяцев и дней недели
NLS_DATE_LANGUAGE='AMERICAN'
-- Тип Сортировки
NLS_SORT=BINARY
-- ??? (нет описания)
NLS_TIME_FORMAT='HH24:MI:SSXFF'
-- Формат ввода и вывода даты типа TIMESTAMP по-умолчанию
NLS_TIMESTAMP_FORMAT='DD.MM.RR HH24:MI:SSXFF'
-- ??? (нет описания)
NLS_TIME_TZ_FORMAT='HH24:MI:SSXFF TZR'
-- Формат ввода и вывода даты типа TIMESTAMP с временнОй зоной по-умолчанию
NLS_TIMESTAMP_TZ_FORMAT='DD.MM.RR HH24:MI:SSXFF TZR'
-- Замещает символ нац. валюты, установленный по умолчанию параметром NLS_TERRITORY
NLS_DUAL_CURRENCY='р.'
-- Как сравнивать строки BINARY или ASCII (по правилам нац. алфавита)
NLS_COMP=BINARY
-- CHAR по умолчанию в байтах или в символах
NLS_LENGTH_SEMANTICS=BYTE
-- NLS_NCHAR_CONV_EXCP determines whether an error is reported when there is
-- data loss during an implicit OR explicit CHARACTER TYPE conversion.
-- The DEFAULT value results IN no error being reported.
NLS_NCHAR_CONV_EXCP=FALSE
Как можно устанавливать значения параметров NLS?
1. В системном реестре Windows
2. Установить переменные окружения
Для Windows (в bat-файле)
SET NLS_DATE_LANGUAGE=RUSSIAN
SET NLS_LANG=AMERICAN_CIS.CL8MSWIN1251
sqlplus ....
3. ALTER SESSION SET
NLS_DATE_LANGUAGE=RUSSIAN
NLS_DATE_FORMAT='DD.MM.YYYY';
SELECT TO_CHAR(SYSDATE, 'Month day')
FROM dual
Посмотреть nls-параметры сессии, базы данных и инстанса можно так:
select * from
(select 'SESSION' SCOPE,s.* from nls_session_parameters s
union
select 'DATABASE' SCOPE,d.* from nls_database_parameters d
union
select 'INSTANCE' SCOPE,i.* from nls_instance_parameters i
) a
pivot (LISTAGG(VALUE) WITHIN GROUP (ORDER BY SCOPE)
FOR SCOPE
in ('SESSION' as "SESSION",'DATABASE' as "DATABASE",'INSTANCE' as "INSTANCE"));
Функции для работы с типом data.
ADD_MONTHS(data, n)
Позволяет добавить к дате целое количество месяцев (или отнять, если n отрицательное)
SELECT ADD_MONTHS('28.04.1971' , 13) FROM DUAL; -- Добавить 13 месяцев
28.05.72
SELECT ADD_MONTHS('28.04.1971' , -12) FROM DUAL; -- Отнять 12 месяцев
28.04.70
В функцию ADD_MONTHS() можно передать и дату и время:
SELECT ADD_MONTHS(TO_DATE('28.04.1971 15:23:44', 'DD.MM.YYYY HH24:MI:SS') , 5) FROM DUAL;-- Добавить 5 месяцев
28.09.71
SELECT TO_CHAR(ADD_MONTHS(TO_DATE('28.04.1971 15:23:44', 'DD.MM.YYYY HH24:MI:SS') , 5), 'DD.MM.YYYY HH24:MI:SS') FROM DUAL;-- Добавить 5 месяцев
28.09.1971 15:23:44
SELECT
TO_CHAR(ADD_MONTHS(TO_DATE('31.10.2001 21:08:50', 'fxDD.MM.YYYY HH24:MI:SS')
, 4) -- Добавить 4 месяца
, 'DD.MM.YYYY HH24:MI:SS')
FROM DUAL;
28.02.2002 21:08:50 - это последний день месяца
SELECT
TO_CHAR(ADD_MONTHS(TO_DATE('28.02.2001 21:08:50', 'fxDD.MM.YYYY HH24:MI:SS')
, -4) -- Отнять 4 месяца
, 'DD.MM.YYYY HH24:MI:SS')
FROM DUAL;
31.10.2000 21:08:50 - это последний день месяца