Опыт оптимизации одной простенькой пользовательской задачи

24.04.24

Разработка - Механизмы типовых конфигураций

Опыт оптимизации одной простенькой пользовательской задачи с использованием запросов, универсального отчета и общих модулей с повторным использованием возвращаемых значений.

Исходные данные: конфигурация ERP ред. 2.5.16.101

Пользователи просят отчет по фактическим трудозатратам, указанным в Этапах производства за период, с группировкой по подразделениям, направлениям деятельности (включена функциональная опция), исполнителям.

Реализуем с помощью универсального отчета по регистру накопления ТрудозатратыКОформлению.

Настройки отчета:

 

 

 

Все довольны и счастливы.

Через какое-то время пользователи просят добавить в отчет должность исполнителя.

Варианты реализации:

  1. Используем уже существующую экспортную функцию в расширении в серверном модуле:
    Функция ПолучитьДолжностьСотрудника(Сотрудник,ДатаСрезаИнформации = "23991231") Экспорт   
    	УстановитьПривилегированныйРежим(Истина);
    	ТолькоРазрешенные   = Ложь;
    	ДатаСрезаИнформации = ДатаСрезаИнформации;
    	КадровыеДанные      = "Должность";
    	Должность = Неопределено;
    	МассивСотрудников = Новый Массив;
    	МассивСотрудников.Добавить(Сотрудник);
    	
    	КадровыеДанныеСотрудников = КадровыйУчет.КадровыеДанныеСотрудников(
    	ТолькоРазрешенные, МассивСотрудников, КадровыеДанные, ДатаСрезаИнформации);
    	
    	Для Каждого Строка Из КадровыеДанныеСотрудников Цикл
    		Должность       = Строка.Должность;
    	КонецЦикла;
    КадровыеДанныеСотрудников = ЦС_ЭкспортныеФункцииСерверПовтИсп.ДолжностьСотрудника(Сотрудник,ДатаСрезаИнформации);
    	
    	СтрокаСотрудника = КадровыеДанныеСотрудников.Найти(Сотрудник, "Сотрудник");
    	Для Каждого Строка Из СтрокаСотрудника Цикл
    		Должность       = СтрокаСотрудника.Должность;
    	КонецЦикла;
    	
    	Возврат Должность;
    КонецФункции
    

    Функция обращается к типовой (КадровыйУчет.КадровыеДанныеСотрудников(

                   ТолькоРазрешенные, МассивСотрудников, КадровыеДанные, ДатаСрезаИнформации)) для получения кадровых данных.

    Но – здесь нужен сотрудник, а в регистре накопления ТрудозатратыКОформлению Исполнитель – физЛицо.

    Меняем запрос на:

    ВЫБРАТЬ РАЗРЕШЕННЫЕ
    	ТрудозатратыКОформлениюОстаткиИОбороты.ПартияПроизводства КАК ПартияПроизводства,
    	ТрудозатратыКОформлениюОстаткиИОбороты.Распоряжение КАК Распоряжение,
    	ТрудозатратыКОформлениюОстаткиИОбороты.ВидРабот КАК ВидРабот,
    	ТрудозатратыКОформлениюОстаткиИОбороты.Исполнитель КАК Исполнитель,
    	ТрудозатратыКОформлениюОстаткиИОбороты.Подразделение КАК Подразделение,
    	ТрудозатратыКОформлениюОстаткиИОбороты.ПериодДень КАК ПериодДень,
    	ТрудозатратыКОформлениюОстаткиИОбороты.КоличествоПриход КАК КоличествоПриход,
    	ISNULL(ОсновныеСотрудникиФизическихЛиц.Сотрудник, СправочникСотрудники.ПустаяСсылка) КАК Сотрудник,
    ИЗ
    	РегистрНакопления.ТрудозатратыКОформлению.ОстаткиИОбороты({&НачалоПериода}, {&КонецПериода}, Авто, Движения, ) КАК ТрудозатратыКОформлениюОстаткиИОбороты
    		ЛЕВОЕ СОЕДИНЕНИЕ РегистрСведений.ОсновныеСотрудникиФизическихЛиц КАК ОсновныеСотрудникиФизическихЛиц
    		ПО ТрудозатратыКОформлениюОстаткиИОбороты.Исполнитель = ОсновныеСотрудникиФизическихЛиц.ФизическоеЛицо
    ГДЕ
    	ОсновныеСотрудникиФизическихЛиц.ДатаНачала = &ПустаяДата
    

    Соединяем нашу таблицу с регистром сведений ОсновныеСотрудникиФизическихЛиц, устанавливаем в Параметрах значение Пустой даты.

    Добавляем в схему СКД универсального отчета вычисляемое поле, в выражении пишем

    НАЗВАНИЕ_ВАШЕГО_ОБЩЕГО_МОДУЛЯ.ПолучитьДолжностьСотрудника(Сотрудник, ТекущаяДата()) – вместо текущей даты можно ставить и дату начала периода и дату окончания периода – в зависимости от ваших желаний (надо понимать, что должность может меняться, и с пользователями это надо обсудить – на какой период получать должность они хотят).

    Выводим поле должность в отчет.

    Отчет формируется, но – и это вполне ожидаемо – долго.

    На наших данных за 10 дней периода отчет формируется примерно 60 секунд.

    Пользователи довольны, но несчастливы (отчет все-таки формируется).

    Мы недовольны, но счастливы (ибо все это суета).

  2. Изучаем типовую функцию КадровыйУчет.КадровыеДанныеСотрудников(ТолькоРазрешенные, МассивСотрудников, КадровыеДанные, ДатаСрезаИнформации), откуда она берет данные, выясняем, что из регистра сведений КадроваяИсторияСотрудниковИнтервальный. Добавляем в запрос соединение с этим регистром:
    ВЫБРАТЬ РАЗРЕШЕННЫЕ
    	ТрудозатратыКОформлениюОстаткиИОбороты.ПартияПроизводства КАК ПартияПроизводства,
    	ТрудозатратыКОформлениюОстаткиИОбороты.Распоряжение КАК Распоряжение,
    	ТрудозатратыКОформлениюОстаткиИОбороты.ВидРабот КАК ВидРабот,
    	ТрудозатратыКОформлениюОстаткиИОбороты.Исполнитель КАК Исполнитель,
    	ТрудозатратыКОформлениюОстаткиИОбороты.Подразделение КАК Подразделение,
    	ТрудозатратыКОформлениюОстаткиИОбороты.ПериодДень КАК ПериодДень,
    	ТрудозатратыКОформлениюОстаткиИОбороты.КоличествоПриход КАК КоличествоПриход,
    	ОсновныеСотрудникиФизическихЛиц.Сотрудник КАК Сотрудник,
    	ОсновныеСотрудникиФизическихЛиц.ДатаНачала КАК ДатаНачала,
    	ОсновныеСотрудникиФизическихЛиц.ДатаОкончания КАК ДатаОкончания
    ПОМЕСТИТЬ втТрудозатратыСотрудников
    ИЗ
    	РегистрНакопления.ТрудозатратыКОформлению.ОстаткиИОбороты({&НачалоПериода}, {&КонецПериода}, Авто, Движения, ) КАК ТрудозатратыКОформлениюОстаткиИОбороты
    		ЛЕВОЕ СОЕДИНЕНИЕ РегистрСведений.ОсновныеСотрудникиФизическихЛиц КАК ОсновныеСотрудникиФизическихЛиц
    		ПО ТрудозатратыКОформлениюОстаткиИОбороты.Исполнитель = ОсновныеСотрудникиФизическихЛиц.ФизическоеЛицо
    ГДЕ
    	ОсновныеСотрудникиФизическихЛиц.ДатаНачала = &ПустаяДата
    ;
    
    ////////////////////////////////////////////////////////////////////////////////
    ВЫБРАТЬ
    	втТрудозатратыСотрудников.ПартияПроизводства КАК ПартияПроизводства,
    	втТрудозатратыСотрудников.Распоряжение КАК Распоряжение,
    	втТрудозатратыСотрудников.ВидРабот КАК ВидРабот,
    	втТрудозатратыСотрудников.Исполнитель КАК Исполнитель,
    	ЕстьNULL(КадроваяИсторияСотрудниковИнтервальный.Должность, Справочник.Должности.ПустаяСсылка КАК Должность,
    	втТрудозатратыСотрудников.Подразделение КАК Подразделение,
    	втТрудозатратыСотрудников.ПериодДень КАК ПериодДень,
    	втТрудозатратыСотрудников.КоличествоПриход КАК КоличествоПриход,
    	втТрудозатратыСотрудников.Сотрудник КАК Сотрудник,
    	втТрудозатратыСотрудников.ДатаНачала КАК ДатаНачала,
    	втТрудозатратыСотрудников.ДатаОкончания КАК ДатаОкончания
    ИЗ
    	втТрудозатратыСотрудников КАК втТрудозатратыСотрудников
    		ЛЕВОЕ СОЕДИНЕНИЕ РегистрСведений.КадроваяИсторияСотрудниковИнтервальный КАК КадроваяИсторияСотрудниковИнтервальный
    		ПО втТрудозатратыСотрудников.Сотрудник = КадроваяИсторияСотрудниковИнтервальный.Сотрудник
    			И втТрудозатратыСотрудников.Исполнитель = КадроваяИсторияСотрудниковИнтервальный.ФизическоеЛицо
    			И (КадроваяИсторияСотрудниковИнтервальный.ДатаОкончания = ДАТАВРЕМЯ(3999, 12, 31, 23, 59, 59))
    

    Убираем вычисляемое поле Должность, вместо него появляется поле Должность, выводим в структуру отчета.

    Отчет формируется и выводится на тех же данных за 2 секунды.

    Все довольны и счастливы.

  3. Попытка реабилитировать и (для себя) понять рамки и возможности применения вычисляемых выражений с обращением к экспортным функциям. Создаем еще один общий модуль в расширении с настройками:

 

В этом модуле размещаем функцию, в которой получаем должность для ВСЕХ сотрудников организации.

Функция ДолжностьСотрудника(Сотрудник,ДатаСрезаИнформации = "23991231") Экспорт   
	УстановитьПривилегированныйРежим(Истина);
	ТолькоРазрешенные   = Ложь;
	ДатаСрезаИнформации = ДатаСрезаИнформации;
	КадровыеДанные      = "Должность";
	Должность = Неопределено;
	
	МассивСотрудников = Новый Массив;
	
	Запрос = Новый Запрос;
	Запрос.Текст = "Выбрать Сотрудники.Ссылка КАК Сотрудник ИЗ Справочник.Сотрудники КАК Сотрудники ГДЕ НЕ Сотрудники.ПометкаУдаления";
	МассивСотрудников = Запрос.Выполнить().Выгрузить();
	КадровыеДанныеСотрудников = КадровыйУчет.КадровыеДанныеСотрудников(
	ТолькоРазрешенные, МассивСотрудников, КадровыеДанные, ДатаСрезаИнформации);
	Возврат КадровыеДанныеСотрудников;
КонецФункции

А в модуле из п.1 (выше): 

Функция ДолжностьСотрудника(Сотрудник,ДатаСрезаИнформации = "23991231") Экспорт   
	КадровыеДанныеСотрудников = ЦС_ЭкспортныеФункцииСерверПовтИсп.ДолжностьСотрудника(Сотрудник,ДатаСрезаИнформации);
	
	СтрокаСотрудника = КадровыеДанныеСотрудников.Найти(Сотрудник, "Сотрудник");
	Должность       = СтрокаСотрудника.Должность;
	
	Возврат Должность;
КонецФункции;

                Здесь мы обращаемся к функции из модуля с повторным использованием возвращаемых значений для получения таблицы КадровыеДанныеСотрудников, из которой берем только ту, которая в данный момент нужна. В настройках отчета ничего менять не нужно по сравнению с п.1.

Отчет формируется и выводится на тех же данных за 6-8 секунд.

 

На этом пока наше исследование закончено. Если у кого есть предложения по альтернативным видам реализации этой задачи – нам будет интересно их узнать.

Спасибо за внимание.

оптимизация отчет СКД общий модуль универсальный отчет

См. также

СКД Механизмы типовых конфигураций Запросы Программист Платформа 1С v8.3 1С:Зарплата и кадры государственного учреждения 3 1С:Зарплата и Управление Персоналом 3.x Россия Бесплатно (free)

Работая с типовыми отчетами в конфигурациях «Зарплата и управление персоналом, редакция 3», «Зарплата и кадры государственного учреждения, редакция 3» и подобных, в схемах компоновки данных можно встретить конструкции запросов, которые обращаются к некоторым виртуальным таблицам.

20.08.2024    1738    PROSTO-1C    0    

18

Механизмы типовых конфигураций Программист Платформа 1С v8.3 1С:Комплексная автоматизация 2.х Россия Бесплатно (free)

Эта ошибка была обнаружена мной в типовой конфигурации 1С:Комплексная автоматизация 2 (2.5.16.115), БСП версия 3.1.9.302. Возникает она после того, как вы добавляете в расширение бизнес-процесс или задачу, выполняете обновление идентификаторов метаданных расширений, но ошибка при записи любого элемента справочника "Профили групп доступа" всё равно остаётся.

01.07.2024    1747    Vidz    0    

11

Механизмы типовых конфигураций Программист Платформа 1С v8.3 Конфигурации 1cv8 Россия Бесплатно (free)

Очень часто в написании кода требуется обращаться к предопределённым значениям. Если идёт обращение к типовым предопределённым значениям, то проблем не возникает.

24.06.2024    1106    olja-ljaaa    0    

3

Механизмы типовых конфигураций Программист Платформа 1С v8.3 Конфигурации 1cv8 Бесплатно (free)

Статистическая выборка сценариев и точек изменения отчетов на общей форме ФормаОтчета в типовых конфигурациях. Примеры кода.

03.06.2024    4894    Serg2000mr    32    

108

Механизмы типовых конфигураций Ценообразование, анализ цен Программист Пользователь Платформа 1С v8.3 1С:ERP Управление предприятием 2 1С:Управление торговлей 11 Россия Абонемент ($m)

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

1 стартмани

11.04.2024    1058    tango    5    

3
Оставьте свое сообщение