Универсальные функции с примерами использования

14 универсальных функций, с примерами использования — для обычного и управляемого интерфейса

Вместо предисловия

Я знаю про CloudConf, и сервисы хранения кода, но на Инфостарте сижу каждый день, поэтому мне здесь удобнее. Да и возможность поделиться какими-то своими идеями, получив отклик, тоже интересна.
Код я оформляю в соответствии с соглашениями о написании кода.

 

1. Замер времени выполнения

 

 2. Проверка значения на вхождение в некоторый интервал

 

 3. Разбиение строки на левую и правую часть

 

 4. Преобразование табличного документа в коллекцию

 

 4.1 Табличный документ в таблицу значений (перебор)

 

 4.2. Табличный документ в дерево значений

 

 5. Загрузка файлов: выбор на клиенте, обработка на сервере

 

 6. Попытка записи данных в файл с таймаутом на время записи

 

 7. Проверка интернет-соединения (ping)

 

 8. Быстрое описание типа

 

 9. Шаблон чтения Excell через ADO

 

 10. Объединение ячеек шапки табличного документа с повторяющимся текстом

 

 11. Формирование идентификатора по строке

 

 12. Формирование представления по идентификатору

 

 13. Формирование структуры для первой (единственной) записи результата запроса

36 Comments

  1. Йожкин Кот

    Замер времени можно сделать точнее:

    ТекущаяУниверсальнаяДатаВМиллисекундах(). Функция появилась в каком-то релизе 8.3

    Reply
  2. tireal

    вот держи в копилку, вместо своего замера

    докВремяНачала = ТекущаяУниверсальнаяДатаВМиллисекундах();
    //процедура замера
    докВремяКонцаВыполнения = ТекущаяУниверсальнаяДатаВМиллисекундах();
    докДельта = (докВремяКонцаВыполнения — докВремяНачала) / 1000;
    
    
    докДельтаПредставление = ПолучитьПредставлениеВремени(докДельта);
    
    Функция ПолучитьПредставлениеВремени(Время)
    Секунды = Время % 60;
    Минуты = (Время-Секунды)/60;
    Минуты = Минуты % 60;
    Часы = (Время-Секунды-Минуты)/3600;
    
    Возврат «» + Окр(Часы) + » ч. » + Окр(Минуты) + » мин. » + Окр(Секунды, 2) + » сек.»;
    
    КонецФункции
    

    Показать

    Reply
  3. unichkin

    (1) Йожкин Кот, это поддерживается только с 8.2.17

    (2) tireal, спасибо за комментарий. Но:

    — свою функцию я могу использовать в любом релизе платформы. Да, я порой работаю с 8.1 — как ни странно оно еще есть.

    — если мне когда-нибудь понадобится настолько низкоуровневое отслеживание выполнения алгоритма — я скорее воспользуюсь замером времени отладчика

    — могу предположить что назначение использования вашей функции — тестирование скорости исполнения алгоритма (т. е. она для этой цели писалась); моей — просто замер времени работы для сообщения/ помещения в лог — секунды вполне устраивают

    Останусь при своем, но буду знать и о таком варианте, еще раз спасибо.

    Reply
  4. androgin

    вместо Функция РазбитьСтроку(стр, сим)

    можно вполне использовать СтрРазделить()

    Reply
  5. unichkin

    (4) androgin, Во-первых назначение СтрРазделить() и моей РазбитьСтроку() — разное. От слова совсем. Во-вторых СтрРазделить() — это начиная с 8.3.6.1977.

    Reply
  6. DrAku1a

    (3) посмотри как сделано в ИР и универсальном отчете (от alexk-is), там есть до миллиекунд.

    Попытка
    Scr = Новый COMОбъект(«MSScriptControl.ScriptControl»);
    Исключение
    Сообщить(ОписаниеОшибки(), СтатусСообщения.Внимание);
    Возврат 0;
    КонецПопытки;
    Scr.Language = «javascript»;
    Время = Scr.Eval(«new Date().getTime()»);
    Возврат Время;

    Показать

    По поводу разделения строки — держи в копилку:

    Функция ИзвлечьСлово(Строка, Разделитель)
    П = найти(Строка+Разделитель, Разделитель);
    Ответ = Лев(Строка, П-1);
    Строка = Сред(Строка, П+СтрДлина(Разделитель));
    Возврат Ответ;
    КонецФункции
    
    Пример:
    стр = «1,2,3,4»;
    Сообщить(ИзвлечьСлово(стр, «,»)); //1
    Сообщить(стр); // 2,3,4
    Сообщить(ИзвлечьСлово(стр, «,»)); //2
    Сообщить(стр); // 3,4
    Сообщить(ИзвлечьСлово(стр, «,»)); //3
    Сообщить(стр); // 4
    Сообщить(ИзвлечьСлово(стр, «,»)); //4
    Сообщить(стр); // «» (пустая строка)
    Сообщить(ИзвлечьСлово(стр, «,»)); // «» (пустая строка)

    Показать

    Если заранее известно, что строка не содержит символов перевода строки, то можно пользоваться стандартным:

    СтрПолучитьСтроку(СтрЗаменить(<ИсходнаяСтрока>, <Разделитель>, Символы.ПС), <НомерСлова>)
    Reply
  7. h00k

    (3)

    — если мне когда-нибудь понадобится настолько низкоуровневое отслеживание выполнения алгоритма — я скорее воспользуюсь замером времени отладчика

    Смешно. Вы думаете всякие извращения с джаваскриптами и т.п. появились потому что ни кто не додумался время выполнения в отладчике смотреть?!

    Спасибо любимой 1С, они хоть от необходимости использовать внешние скрипты избавили, добавив в платформу функцию

    ТекущаяУниверсальнаяДатаВМиллисекундах()

    А для старых версий платформы остается только замерять время выполнения при помощи внешних средств, например джава-скрипта.

    Reply
  8. unichkin

    (6) Знаю я об этих фишках, спасибо. Функция «ИзвлечьСлово» — не учитывает что символа может и не быть. И режет исходную строку. Как-раз чтобы этого избежать я наваял сп_РазбитьСтроку() — ее конечно не стоит применять на большом массиве данных, но для чтения небольших листов Excell — самое оно.

    (7) h00k, почитайте внимательнее мой комментарий в (3) . И да, мне удобнее воспользоваться замером времени, чем вставлять в конфу отладочную печать. Не могу вспомнить случая, когда я не мог без этого обойтись. Может опыта еще маловато)

    Имхо, все извращения с получением настолько точного времени — связаны с необходимостью отслеживать СКОРОСТЬ, а не ВРЕМЯ выполнения алгоритма. Разница трудноуловима, но она есть. Мне не надо с точностью до миллисекунды знать сколько времени проводился массив документов — 20 мин меня вполне устроит. Но если я тестирую напр. как быстрее создавать таблицу значений — тут нужны миллисекунды. Просто для универсальности функция замера обычно реализуется наиболее детализированно — т.е. через тот-же ява-скрипт. Ну а мне как-то не нужно было. Если понадобится — возьму из (6).

    Reply
  9. mars207

    Загрузка файлов: выбор на клиенте, обработка на сервере

    На веб клиенте давно не работает. Проблема всплывающих окон в браузерах. На последних релизах платформы не тестировал, но подозреваю что не исправили.

    Reply
  10. unichkin

    (9) mars207, тестировал на 8.3.5 — у меня работает. чяднт?

    Reply
  11. androgin

    (9) mars207, все прекрасно работает

    Reply
  12. androgin

    (5) уже давно все типовые версии требуют 8.3.6.2094 как минимальную

    Reply
  13. MishaD

    Странно, у меня половина типовых на 8.2.14.540, а еще половина на 7.7.027

    Reply
  14. unichkin

    (12) androgin, далеко не все используют исключительно типовые, и далеко не все еще перешли на новые конфы с УФ и БСП. Я уже писал в (3) что и с 8.1 иногда работаю.

    Reply
  15. h00k

    (15) androgin,

    то, что вы тормозите собственное развитие и сидите в старых платформах и на старых методах — вынуждает вас писать подобные коды

    Вы можете ошибаться… собственное развитие и вынужденная работа со старыми версиями конфигураций в учетных системах заказчика/ работодателя — в большинстве случаев ни как не связаны.

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

    Reply
  16. unichkin

    (15) androgin, не понимаю сути претензий. И с чего вы решили что я где-то там «торможусь»?.. Я подписан на рассылку, просматриваю ИТС, мониторю зазеркалье (о птичках) работаю в основном на 8.3.6. И абсолютно солидарен с вами что в 1С сидят не дураки. Я только не понимаю, к чему это было сказано) Лучше перечитайте (5), попробуйте вникнуть в назначение сп_РазбитьСтроку() и понять таки что СтрРазделитьСтроку() в ее контексте в принципе не нужна. Я наверное сам дал еды троллю — лишним было писать «Во-вторых СтрРазделить() — это начиная с 8.3.6.1977».

    Reply
  17. K_A_O

    > Преобразование табличного документа в таблицу значений

    Вот это вроде с давних времен работает

    Построитель = Новый ПостроительОтчета;
    Построитель.ИсточникДанных=Новый ОписаниеИсточникаДанных(ТабДок.Область());
    Построитель.Выполнить();
    ТЗ = Построитель.Результат.Выгрузить();
    Reply
  18. sansys

    Спасибо за статью. Иногда стоит вылезти из своего болота и посмотреть как люди делают :))))) +1

    Reply
  19. ImHunter

    Про замеры времени. Написал пару процедур — засечь время и показать время замера (+ 2 вспомогательные). Недостаток — в многопотоке некорректно работает. А так, удобненько.

    // клиентский модуль
    
    Процедура ЗасечьТаймер() Экспорт
    УправляемыйИнтерфейсСервер.СохранитьЗначениеВХранилище( «ТекВремяТаймера», ТекущаяДата() );
    //ПоместитьВоВременноеХранилище( ТекущаяДата(), );
    КонецПроцедуры
    
    Процедура ПоказатьВремяТаймера( пЗаголовок = «», пЧерезСообщение = Ложь ) Экспорт
    д = УправляемыйИнтерфейсСервер.ПолучитьЗначениеИзХранилища( «ТекВремяТаймера» );
    общ = ТекущаяДата() — д;
    мин = цел( общ / 60 );
    сек = общ — мин * 60;
    час = цел( мин / 60);
    мин = мин — час * 60;
    лВрем = ?( пЧерезСообщение, «Время выполнения:», «Время:» );
    Если час <> 0 Тогда
    лВрем = лВрем + » » + час + » час»
    КонецЕсли;
    Если Мин <> 0 Тогда
    лВрем = лВрем + » » + мин + » мин»
    КонецЕсли;
    лВрем = лВрем + » » + сек + » сек»;
    Если пЧерезСообщение=Ложь Тогда
    ПоказатьОповещениеПользователя( пЗаголовок, , лВрем );
    Иначе
    Сообщить( ?( ПустаяСтрока( пЗаголовок ), «Действие выполнено», пЗаголовок ) + «. » + лВрем );
    КонецЕсли;
    КонецПроцедуры
    
    // серверный модуль УправляемыйИнтерфейсСервер
    
    Процедура СохранитьЗначениеВХранилище( пКлюч, пЗначение ) Экспорт
    ХранилищеСистемныхНастроек.Сохранить( , пКлюч, пЗначение, , ПараметрыСеанса.ТекущийПользователь );
    КонецПроцедуры
    
    Функция ПолучитьЗначениеИзХранилища( пКлюч ) Экспорт
    Возврат ХранилищеСистемныхНастроек.Загрузить( , пКлюч, , ПараметрыСеанса.ТекущийПользователь );
    КонецФункции
    
    
    // пример использования
    &НаКлиенте
    Процедура СделатьЧтоТо()
    УправляемыйИнтерфейс.ЗасечьТаймер();
    // что-то делаем
    // ………………….
    УправляемыйИнтерфейс.ПоказатьВремяТаймера(«Сделали что-то!»);
    КонецПроцедуры
    
    

    Показать

    Reply
  20. Altair777

    (18) А у меня еще дополнено

    Построитель.ДобавлениеПредставлений = ТипДобавленияПредставлений.НеДобавлять;
    Построитель.ЗаполнитьНастройки();
    Reply
  21. nSpirit2

    Я конечно все понимаю но раз вы пишите

    Код я оформляю в соответствии с соглашениями о написании кода.

    Еще раз прочитайте эту часть

    https://its.1c.ru/db/v8std#content:2149184103:hdoc

    Мне кажется вы не правильно поняли блок как не надо делать

    Reply
  22. unichkin

    (22)

    Мне кажется вы не правильно поняли блок как не надо делать

    Какой участок кода навел вас на эту мысль?

    Reply
  23. LexSeIch

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

    Reply
  24. Aphanas

    Функция ИдентификаторПоСтроке соответствует алгоритму платформы?

    Reply
  25. unichkin

    (25)

    Функция ИдентификаторПоСтроке соответствует алгоритму платформы?

    Специально не тестил, свое предназначение выполняет; вообще, должна. Возможно в регистрах символов будет различие, но т.к не имею кода платформы на руках 100% гарантии дать не могу)

    Reply
  26. SerVer1C
    Функция ЭтоGUID(Идентификатор)
    Попытка
    GUID = Новый УникальныйИдентификатор(Идентификатор);
    Исключение
    КонецПопытки;
    
    Возврат GUID <> Неопределено;
    КонецФункции
    Reply
  27. unichkin

    (27) Решать задачи через попытку там где можно ее не использовать — очень плохой тон.

    Reply
  28. SerVer1C

    (28) Эту фразу вы вычитали в «умных» книжках? Корректная работа с СУБД при записи данных идет через попытку. Вы считаете, что использовать портянки кода там, где можно обойтись одной строкой, это хороший тон? Сначала доработайте алгоритм, чтобы он не считал «zzzzzzzz-zzzz-zzzz-zzzz-zzzzzzzzzzzz» валидным ГУИДом, а потом еще проверьте его на быстродействие.

    Reply
  29. Артано

    (29) Регулярные выражения. По быстродействию предлагаю провести сравнение. Было бы инетересно

    Reply
  30. SerVer1C

    (30) Замерил с помощью внешней компоненты /public/940766 на венде х64 — получился прирост в скорости в ~16 раз. Использовал шаблон «([0-9A-Fa-f]{8})-([0-9A-Fa-f]{4})-([0-9A-Fa-f]{4})-([0-9A-Fa-f]{4})-([0-9A-Fa-f]{4,12})»

    Reply
  31. Артано

    (31) А в цифрах для каждого типа?

    Reply
  32. unichkin

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

    Reply
  33. SerVer1C

    (33) Для этого давным-давно придумали замечательную вещь: Исключения.

    Reply
  34. unichkin

    (34) не понимаю, о чем вы, если можно — подробнее? з.ы. я не уточнил — имелась в виду отладка с включенным флагом «Остановка по ошибке».

    Reply
  35. svilsa

    Очень классно, спасибо!!! Такая шпаргалка в закладках, чтобы не ломать каждый раз голову. Коротко и ясно.

    Reply
  36. unichkin

    (27) Обнаружил недавно в БСП СтроковыеФункцииКлиентСервер.ЭтоУникальныйИдентификатор:

    // Проверяет, является ли строка уникальным идентификатором.
    // В качестве уникального идентификатора предполагается строка вида
    // «XXXXXXXX-XXXX-XXXX-XXXX-XXXXXXXXXXXX», где X = [0..9,a..f].
    //
    // Параметры:
    //  Значение — Строка — проверяемая строка.
    //
    // Возвращаемое значение:
    //  Булево — Истина, если переданная строка является уникальным идентификатором.
    //
    Функция ЭтоУникальныйИдентификатор(Знач Значение) Экспорт
    
    Шаблон = «XXXXXXXX-XXXX-XXXX-XXXX-XXXXXXXXXXXX»;
    
    Если СтрДлина(Шаблон) <> СтрДлина(Значение) Тогда
    Возврат Ложь;
    КонецЕсли;
    Для Позиция = 1 По СтрДлина(Значение) Цикл
    Если КодСимвола(Шаблон, Позиция) = 88 // X
    И ((КодСимвола(Значение, Позиция) < 48 Или КодСимвола(Значение, Позиция) > 57) // 0..9
    И (КодСимвола(Значение, Позиция) < 97 Или КодСимвола(Значение, Позиция) > 102) // a..f
    И (КодСимвола(Значение, Позиция) < 65 Или КодСимвола(Значение, Позиция) > 70)) // A..F
    Или КодСимвола(Шаблон, Позиция) = 45 И КодСимвола(Значение, Позиция) <> 45 Тогда // —
    Возврат Ложь;
    КонецЕсли;
    КонецЦикла;
    
    Возврат Истина;
    
    КонецФункции
    
    

    Показать

    Reply

Leave a Comment

Ваш адрес email не будет опубликован. Обязательные поля помечены *