ВЫРАЗИТЬ - неоднозначный оператор.

Функцию ВЫРАЗИТЬ в языке запросов 1С 8 многие интерпретируют как преобразователь типов, но она предназначена совершенно не для этих целей. Подробности под катом...

Итак, многие ошибочно полагают, что смогут преобразовать поле с типом Строка в поле с типом Число или ссылку в строку. На самом деле оператор ВЫРАЗИТЬ может преобразовать:

    • настройки примитивного типа;

    • поле составного типа в поле одиночного типа;

Рассмотрим эти ситуации более подробно...

Преобразование настроек примитивного типа

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

ВЫБРАТЬ

ВЫРАЗИТЬ(Приходная.Комментарий КАК СТРОКА(300)) КАК Комментарий,

ВЫРАЗИТЬ("" КАК Строка(10)) КАК КИ_СтатьяУправленческогоУчета

КОЛИЧЕСТВО(Приходная.Ссылка) КАК Ссылка

ИЗ

Документ.Приходная КАК Приходная

СГРУППИРОВАТЬ ПО

ВЫРАЗИТЬ(Приходная.Комментарий КАК СТРОКА(300))

Еще ситуация когда в запросе используются расчеты, на выходе мы можем получить число с большим количество знаков после запятой (1100,001568794) . Чтобы не обрабатывать это число полсе выполнения запроса его сразу можно обрезать до нужной длины, но важно понимать что число именно обрезается, а не округляется. Пример:

ВЫБРАТЬ

Продажи.Товар,

ВЫРАЗИТЬ(Продажи.Количество * Продажи.Цена КАК ЧИСЛО(15, 2)) КАК Сумма

ИЗ

РегистрНакопления.Продажи КАК Продажи

Преобразование составного типа к одиночному

Регистратор у регистров часто имеет составной тип, чтобы преобразовать его к одиночному типу используйте конструкцию ВЫРАЗИТЬ правда если на этапе выборки вы попытаетесь преобразовать документ реализацию в документ поступление, то запрос обязательно вылетит с ошибкой, поэтому перед преобразованием следует проверить тип ссылки. Вот такая вот белиберда))) Зачем все это нужно спросите вы. Отвечаю, это один из моментов неявной оптимизации запроса в ущерб краткости написания. Рассмотрим приминение этого момента на примере.

Допустим, вы задались целью получить номер каждого регистратора у РН Продажи. пишем запрос:

ВЫБРАТЬ РАЗЛИЧНЫЕ

Продажи.Регистратор.Номер

ИЗ

РегистрНакопления.Продажи КАК Продажи

Собственно, ничего не может быть проще. Вот только 1С на этапе выполнения преобразует этот запрос без каких либо соединений в запрос с таким количество левых соединений сколько у нас возможных регистраторов. Т.е. если пишут в этот регистр 20 документов, то получим SQL запрос с 20 левыми соединениями. Почему так происходит? Потому что встроенный 1С оптимизатор не совсем хорошо обрабатывает поля получаемые через точку, в данном случае это реквизит Номер. Вот такие пироги, если мы часто захотим получать номер документа, то разумнее всего включить его в реквизиты регистра или же использовать оператор ВЫРАЗИТЬ, но в ущерб краткости:

ВЫБРАТЬ РАЗЛИЧНЫЕ

Продажи.Регистратор.Номер,

ВЫБОР

КОГДА Продажи.Регистратор ССЫЛКА Документ.Расходная

ТОГДА ВЫРАЗИТЬ(Продажи.Регистратор КАК Документ.Расходная)

ИНАЧЕ ВЫБОР

КОГДА Продажи.Регистратор ССЫЛКА Документ.Реализация

ТОГДА ВЫРАЗИТЬ(Продажи.Регистратор КАК Документ.Реализация)

КОНЕЦ

...

КОНЕЦ КАК Номер

ИЗ

РегистрНакопления.Продажи КАК Продажи

Теперь в левом соединении будет участвовать одна конкретная таблица.

В общем случае, стоит внимательно обращаться к данным через точку, т.к. 1С в этом случае использует левое соединение в запросе SQL, что может существенно отразиться на производительности. Это один из моментов оптимизации.