[v8] Несколько решений для работы с периодами и датами

25.03.20

Разработка - Универсальные функции

Несколько готовых решений нетривиальных задач.

Отсчет количества рабочих дней от даты в запросе.
Появилась потребность вычислять плановую дату оплаты в запросе. Для этого необходимо к дате договора прибавить определенное количество рабочих (банковских) дней (отсрочку платежа). Перерыл весь интернет, решения не нашел. Потратив некоторое количество времени на написание своих запросов, решил выложить сюда, может это кому-то сэкономит немного времени.
В запрос передается дата и количество дней отсрочки. Возвращается конечная дата (дата +- количество дней отсрочки). Если дата выпадает на не рабочий день, берется следующий рабочий.

Назад

ВЫБРАТЬ
    КОЛИЧЕСТВО(РАЗЛИЧНЫЕ РегламентированныйПроизводственныйКалендарь.ДатаКалендаря) КАК Отсрочка,
    МИНИМУМ(РегламентированныйПроизводственныйКалендарь.ДатаКалендаря) КАК Дата
ПОМЕСТИТЬ втДаты
ИЗ
    РегистрСведений.РегламентированныйПроизводственныйКалендарь КАК РегламентированныйПроизводственныйКалендарь
        ВНУТРЕННЕЕ СОЕДИНЕНИЕ РегистрСведений.РегламентированныйПроизводственныйКалендарь КАК РегламентированныйПроизводственныйКалендарь1
        ПО РегламентированныйПроизводственныйКалендарь.ДатаКалендаря >= РегламентированныйПроизводственныйКалендарь1.ДатаКалендаря
ГДЕ
    (РегламентированныйПроизводственныйКалендарь.ВидДня = ЗНАЧЕНИЕ(Перечисление.ВидыДнейПроизводственногоКалендаря.Предпраздничный)
            ИЛИ РегламентированныйПроизводственныйКалендарь.ВидДня = ЗНАЧЕНИЕ(Перечисление.ВидыДнейПроизводственногоКалендаря.Рабочий))
    И РегламентированныйПроизводственныйКалендарь.ДатаКалендаря < &Дата
    И ДОБАВИТЬКДАТЕ(&Дата, ДЕНЬ, -&Отсрочка - 50) < РегламентированныйПроизводственныйКалендарь.ДатаКалендаря

СГРУППИРОВАТЬ ПО
    РегламентированныйПроизводственныйКалендарь1.ДатаКалендаря

ИМЕЮЩИЕ
    МИНИМУМ(РегламентированныйПроизводственныйКалендарь.ДатаКалендаря) = РегламентированныйПроизводственныйКалендарь1.ДатаКалендаря
;

////////////////////////////////////////////////////////////////////////////////
ВЫБРАТЬ
    втДаты.Дата
ИЗ
    втДаты КАК втДаты
ГДЕ
    втДаты.Отсрочка = &Отсрочка

Вперед

ВЫБРАТЬ
    КОЛИЧЕСТВО(РАЗЛИЧНЫЕ РегламентированныйПроизводственныйКалендарь.ДатаКалендаря) КАК Отсрочка,
    МАКСИМУМ(РегламентированныйПроизводственныйКалендарь.ДатаКалендаря) КАК Дата
ПОМЕСТИТЬ втДаты
ИЗ
    РегистрСведений.РегламентированныйПроизводственныйКалендарь КАК РегламентированныйПроизводственныйКалендарь
        ВНУТРЕННЕЕ СОЕДИНЕНИЕ РегистрСведений.РегламентированныйПроизводственныйКалендарь КАК РегламентированныйПроизводственныйКалендарь1
        ПО РегламентированныйПроизводственныйКалендарь.ДатаКалендаря <= РегламентированныйПроизводственныйКалендарь1.ДатаКалендаря
ГДЕ
    (РегламентированныйПроизводственныйКалендарь.ВидДня = ЗНАЧЕНИЕ(Перечисление.ВидыДнейПроизводственногоКалендаря.Предпраздничный)
            ИЛИ РегламентированныйПроизводственныйКалендарь.ВидДня = ЗНАЧЕНИЕ(Перечисление.ВидыДнейПроизводственногоКалендаря.Рабочий))
    И РегламентированныйПроизводственныйКалендарь.ДатаКалендаря > &Дата
    И ДОБАВИТЬКДАТЕ(&Дата, ДЕНЬ, &Отсрочка + 50) > РегламентированныйПроизводственныйКалендарь.ДатаКалендаря

СГРУППИРОВАТЬ ПО
    РегламентированныйПроизводственныйКалендарь1.ДатаКалендаря

ИМЕЮЩИЕ
    МАКСИМУМ(РегламентированныйПроизводственныйКалендарь.ДатаКалендаря) = РегламентированныйПроизводственныйКалендарь1.ДатаКалендаря
;

////////////////////////////////////////////////////////////////////////////////
ВЫБРАТЬ
    втДаты.Дата
ИЗ
    втДаты КАК втДаты
ГДЕ
    втДаты.Отсрочка = &Отсрочка

Разбиение произвольного периода на равные подпериоды (не в запросе).
Для тяжелых алгоритмов иногда требуется разбивать произвольный период на n-ное количество равных подпериодов, чтобы обрабатывать их асинхронно. Вот пример небольшой функции, которая этим занимается. Результат работы представлен на скриншоте.

КоличествоПодпериодов = 10;

Подпериоды = Новый ТаблицаЗначений;
Подпериоды.Колонки.Добавить("НачалоПериода");
Подпериоды.Колонки.Добавить("КонецПериода");

Шаг  = Цел((КонецПериода - НачалоПериода)/КоличествоПодпериодов);
День = НачалоПериода;
Пока День < КонецДня(КонецПериода) Цикл
	мНачалоПериода = ?(День = НачалоПериода, День, День + 1);
		
	День = КонецДня(День + Шаг);
		
	НоваяСтрока = Подпериоды.Добавить();
	НоваяСтрока.НачалоПериода = мНачалоПериода; 
	НоваяСтрока.КонецПериода  = ?(День >= КонецДня(КонецПериода), КонецДня(КонецПериода), День);  
КонецЦикла;

Вступайте в нашу телеграмм-группу Инфостарт

Вы можете заказать платную адаптацию этой статьи под ваши задачи на «Бирже заказов».

  • 0% комиссии — оплата напрямую исполнителю;
  • Исполнители любого масштаба — от отдельных специалистов до команд под проект;
  • Прямой обмен контактами между заказчиком и исполнителем;
  • Безопасная сделка — при необходимости;
  • Рейтинги, кейсы и прозрачная система откликов.

См. также

Загрузка и выгрузка в Excel Универсальные функции Программист 1С:Предприятие 8 Россия Бесплатно (free)

Описанный ниже подход позволяет в три шага заполнять формулы в Excel файлы, вне зависимости от ОС сервера (MS Windows Server или Linux). Подход подразумевает отказ от работы с COM-объектом в пользу работы через "объектную модель документа" (DOM).

30.10.2025    4825    Abysswalker    11    

47

Универсальные функции Работа с интерфейсом Программист 1С:Предприятие 8 Бесплатно (free)

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

14.05.2025    9024    DeerCven    15    

63

Универсальные функции Программист 1С:Предприятие 8 1C:Бухгалтерия Бесплатно (free)

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

21.05.2024    57913    dimanich70    85    

175

Универсальные функции Программист 1С:Предприятие 8 1C:Бухгалтерия Абонемент ($m)

Задача: вставить картинку из буфера обмена на форму средствами платформы 1С.

1 стартмани

18.03.2024    8138    7    John_d    13    

59

Универсальные функции Программист Стажер 1С:Предприятие 8 1C:Бухгалтерия Бесплатно (free)

Пришлось помучиться с GUID-ами немного, решил поделиться опытом, мало ли кому пригодится.

12.02.2024    72661    atdonya    31    

73

Универсальные функции Программист 1С:Предприятие 8 Бесплатно (free)

На заключительных этапах, когда идет отладка или доработка интерфейса, необходимо много раз переоткрыть внешний объект. Вот один из способов автоматизации этого.

30.11.2023    10112    ke.92@mail.ru    17    

68
Комментарии
Подписаться на ответы Инфостарт бот Сортировка: Древо развёрнутое
Свернуть все
1. danjer74 4 25.03.20 12:19 Сейчас в теме
Автор молодец. Может пригодиться на сдаче спеца по платформе.
1C_Casual; +1 Ответить
2. Dach 391 27.03.20 00:31 Сейчас в теме
Можно немного улучшить:


ВЫБРАТЬ
	МИНИМУМ(РегламентированныйПроизводственныйКалендарь.ДатаКалендаря) КАК Дата,
	"Назад" КАК Направление
ИЗ
	РегистрСведений.РегламентированныйПроизводственныйКалендарь КАК РегламентированныйПроизводственныйКалендарь
		ВНУТРЕННЕЕ СОЕДИНЕНИЕ РегистрСведений.РегламентированныйПроизводственныйКалендарь КАК РегламентированныйПроизводственныйКалендарь1
		ПО РегламентированныйПроизводственныйКалендарь.ДатаКалендаря >= РегламентированныйПроизводственныйКалендарь1.ДатаКалендаря
			И (РегламентированныйПроизводственныйКалендарь.ВидДня В (ЗНАЧЕНИЕ(Перечисление.ВидыДнейПроизводственногоКалендаря.Предпраздничный), ЗНАЧЕНИЕ(Перечисление.ВидыДнейПроизводственногоКалендаря.Рабочий)))
			И (РегламентированныйПроизводственныйКалендарь.ДатаКалендаря < &Дата)
			И (ДОБАВИТЬКДАТЕ(&Дата, ДЕНЬ, -&Отсрочка - 50) < РегламентированныйПроизводственныйКалендарь.ДатаКалендаря)

СГРУППИРОВАТЬ ПО
	РегламентированныйПроизводственныйКалендарь1.ДатаКалендаря

ИМЕЮЩИЕ
	МИНИМУМ(РегламентированныйПроизводственныйКалендарь.ДатаКалендаря) = РегламентированныйПроизводственныйКалендарь1.ДатаКалендаря И
	КОЛИЧЕСТВО(РАЗЛИЧНЫЕ РегламентированныйПроизводственныйКалендарь.ДатаКалендаря) = &Отсрочка

ОБЪЕДИНИТЬ ВСЕ

ВЫБРАТЬ
	МАКСИМУМ(РегламентированныйПроизводственныйКалендарь.ДатаКалендаря),
	"Вперед"
ИЗ
	РегистрСведений.РегламентированныйПроизводственныйКалендарь КАК РегламентированныйПроизводственныйКалендарь
		ВНУТРЕННЕЕ СОЕДИНЕНИЕ РегистрСведений.РегламентированныйПроизводственныйКалендарь КАК РегламентированныйПроизводственныйКалендарь1
		ПО РегламентированныйПроизводственныйКалендарь.ДатаКалендаря <= РегламентированныйПроизводственныйКалендарь1.ДатаКалендаря
			И (РегламентированныйПроизводственныйКалендарь.ВидДня В (ЗНАЧЕНИЕ(Перечисление.ВидыДнейПроизводственногоКалендаря.Предпраздничный), ЗНАЧЕНИЕ(Перечисление.ВидыДнейПроизводственногоКалендаря.Рабочий)))
			И (РегламентированныйПроизводственныйКалендарь.ДатаКалендаря > &Дата)
			И (ДОБАВИТЬКДАТЕ(&Дата, ДЕНЬ, &Отсрочка + 50) > РегламентированныйПроизводственныйКалендарь.ДатаКалендаря)

СГРУППИРОВАТЬ ПО
	РегламентированныйПроизводственныйКалендарь1.ДатаКалендаря

ИМЕЮЩИЕ
	МАКСИМУМ(РегламентированныйПроизводственныйКалендарь.ДатаКалендаря) = РегламентированныйПроизводственныйКалендарь1.ДатаКалендаря И
	КОЛИЧЕСТВО(РАЗЛИЧНЫЕ РегламентированныйПроизводственныйКалендарь.ДатаКалендаря) = &Отсрочка

Показать
3. Eremkin 27.03.20 15:11 Сейчас в теме
Добрый день. А рассматривали в качестве решения БСП-ую функцию ДатыБлижайшихРабочихДней модуля КалендарныеГрафики? Или она не подошла для вашего случая?
Для отправки сообщения требуется регистрация/авторизация