Автор статьи - Печеный Василий.
Данная статья не является полным описанием и не претендует на оригинальность. Цель - облегчить понимание работы 1С, а также способствовать написанию прямых запросов к таблицам 1С. Статья написанная автором на основе личного опыта, а также той немногочисленной информации, которую можно раздобыть в Internet в свободном (и не очень) доступе. Возможно некоторые поля или таблицы могут измениться с выходом последующих релизов, но пока (до 25 релиза включительно) таких изменений не было замечено.
ID может иметь 3 представления (уровня) в зависимости от длины (количества значащих символов):
В некоторых случаях, при указании неопределенного типа объекта (длина ID кода 23 символа) создается дополнительное поле с символом «T» вначале (например, если в справочнике задан реквизит SP235 как неопределенный, то в таблице справочника будет создано еще одно поле TSP235). Рассмотрим поподробнее значения этого поля. Это поле по умолчанию заполняется пустой строкой (3 пробела).
Примечание
Поскольку для неопределенных реквизитов (полей) длина поля всегда равна 23 символам, то соответственно в такое поле можно записать значение максимум с 22 символьным значением (1 отводится под определение типа, в случае базовых типов, таких как число, строка, дата).
Краткое описание: таблица предназначена для хранения уникального идентификатора сессии первого присоединившегося к базе данных пользователя в режиме предприятия (поэтому в таблице всегда только одна запись). Судя по всему, именно по наличию записи в этой таблицы и происходит верификация процедур при первом запуске в SQL.
Название поля | Описание |
---|---|
CONNECTUUID | GUID (уникальный идентификатор) первого сеанса, соединившегося с 1С в режиме предприятия. Тип - Строка(36). |
Краткое описание: таблица предназначена для хранения значений констант и периодических реквизитов справочников.
Название поля | Описание |
---|---|
ROW_ID | Порядковый номер записи в таблице. Тип - Число(int). |
OBJID | ID объекта периодического реквизита (справочника) или периодической константы, для непериодических констант это поле равно ‘ 0 ’. Тип - Строка(9). |
ID | ID (идентификатор) константы или периодического реквизита справочника. Тип представлен в виде десятичного числа (_StrToID()). Тип - Число(int). |
DATE | Дата установки периодического реквизита. Для базы в формате SQL и для непериодического реквизита (или константы) поле заполняется значением '1753-01-01 00:00:00.000'. Тип - Дата (datetime для SQL). |
VALUE | Значение константы или периодического реквизита. Для неопределенных типов по умолчанию заполняется «U». Тип- Строка(255). |
DOCID | ID (идентификатор) документа (поле, связанное с полем IDDOC таблицы журналов (_1SJOURN)). Для значений установленных вручную или непериодических заполняется ' 0 '. |
TIME | Время установки значения в десятичном формате ((Часы*3600+Минуты*60+секунды)*10000). Тип - Число(int). |
ACTNO | Номер движения документа (включая каждое движения по регистрам и запись периодических реквизитов за исключением проводок). В случае непериодического значения заполняется нулем. Тип - Число(int). |
LINENO_ | Номер строки документа (заполняется при вызове метода ПривязыватьСтроку(), если привязка не выполнена или непериодическое значение - заполняется нулем). Тип - Число(tinyint). |
TVALUE | Заполняется только для неопределенных реквизитов, для типов данных 1С (когда длина ID равна 23 символам). Тип - Строка(3). |
Краткое описание: таблица предназначена для хранения ссылок подчиненных документов и граф отборов реквизитов документов, которые не являются общими реквизитами. Для хранения ссылок документов в подчиненном документе должен быть реквизит (поле) в котором будет ссылка на документ-родитель. Без такой ссылки документ не считается подчиненным.
Название поля | Описание |
---|---|
ROW_ID | Порядковый номер записи в таблице. Номер проставляется автоматически при вводе новой строки. При изменении даты документа, введенного на основании, это поле «переприсваивается». Тип - Число(int). |
MDID | Номер графы отбора подчиненного документа. Тип - Число(int). |
PARENTVAL | Полный идентификатор документа родителя. Тип – Строка(23). |
CHILD_DATE_TIME_IDDOC | Дата + Время (в 36-ричном формате) + ИД подчиненного документа. Тип – Строка(23). |
CHILDID | ИД подчиненного документа. Тип – Строка(9). |
FLAGS | ??? Тип – Число(1). |
Краткое описание: таблица предназначена для хранения списка всех документов. Именно эта таблица является узким местом при работе большого числа пользователей (так как она блокируется полностью каждым пользователем в момент проведения документов).
Название поля | Описание |
---|---|
ROW_ID | Порядковый номер записи в таблице. Номер присваивается в порядке ввода документов. При изменении даты или времени – не изменяется. Тип - Число(4). |
IDJOURNAL | ID журнала документов, которому принадлежит документ (из метаданных). Тип - Число(4). |
IDDOC | ID документа в 36-ричной системе счисления. Тип - Строка(9). |
IDDOCDEF | ID вида документа. Номер документа из метаданных (номер, а не 36-ричное представление). Тип - Число(4). |
APPCODE | Число, отражающее принадлежность документа к компонентам. Возможные
значения:
|
DATE_TIME_IDDOC | Составное поле, обозначающее позицию документа во времени + ID
документа.
|
DNPREFIX | Префикс номера документа. Тип - Строка(18). Для документов, у которых код числовой, это поле равно десятичному ID вида документа. Если нумерация в пределах периода - то также хранится и период в виде ГГГГММДД (например 2006 для нумерации в пределах года). |
DOCNO | Номер документа. Длина определяется максимальным номером всех документов. Тип - Строка (максимальное значение - 20). |
CLOSED | Флаг проведения документа. Для каждого набора компонент свое значение (аналогично полю APPCODE). Значение последнего байта отвечает за флаг проведения документа. Тип - Число(1). |
ISMARK | Флаг пометки на удаление. 0 - не помечен, 1 - помечен. Тип - Число(1,0) (бинарное значение). |
ACTCNT | Счетчик действий (движения) для документа (один документ может вызывать несколько движений регистров). Фактически хранит информацию о количестве движений по всем регистрам + записи периодических реквизитов (без учета проводок по бух. учету). Тип – Число(4). |
VERSTAMP | Количество изменений записи таблицы. Изменением считается любое действие "Изменить (открыть)" + действия при изменении структуры. Тип - Число(4). |
RFxxx | Флаг наличия движений по регистру "xxx". Тип - Число(1,0). |
SPyyy | Значение общих реквизитов документов, в которых установлено свойство «отбор». Если свойство «отбор» не установлено, то общие реквизиты хранятся в таблицах документов (DHxxx). Тип - Число, Строка, Дата. |
TSPyyy | Дополнение к общим реквизитам, заполняется только для неопределенных типов значений реквизитов отбора (используется совместно с полем «SPyyy»). Тип - Строка(3). |
DSzzz | Флаг принадлежности последовательности. ZZZ - десятичный ID последовательности. Принимает 3 значения: 0 - документ не принадлежит последовательности, 1 - документ находится на границе или за границей последовательности (или находился на ней), 2 - документ находится перед границей последовательности (устанавливается в момент проведения). |
Краткое описание: таблица предназначена для хранения информации о зарезервированных номерах документов, т.е. тех. номерах документов, которые вводятся в данный момент в систему, но еще не сохранены (не записаны в таблицу журналов _1SJOURN)
Название поля | Описание |
---|---|
DNPREFIX | Идентификатор вида документа или справочника. Для документа также включает время (всегда 0 0). Тип – Строка(28). |
DOCNO | Зарезервированный номер. Тип – Строка(10) (определяется максимальным номером в системе, максимальное значение - 20). |
Краткое описание: таблица предназначена для хранения информации о последовательностях.
Название поля | Описание |
---|---|
ID | Идентификатор последовательности (числовое представление). Тип – Число(4). |
DATE_TIME_DOCID | Дата+Время+идентификатор документа, на котором установлена последовательность. Это поле аналогично полю DATE_TIME_IDDOC таблицы _1SJOURN. Тип – Строка(23). |
Краткое описание: таблица предназначена для хранения информации о датах точки актуальности и рассчитанного периода бухгалтерских итогов, также в таблице хранится информация о параметрах УРБД.
Название поля | Описание |
---|---|
CURDATE | Дата точки актуальности. Тип - Дата (для SQL DateTime). |
CURTIME | Время точки актуальности. Время хранится в десятичном виде: (Часы*3600+Минуты*60+секунды)*10000. Преобразовав значение этого поля в 36-ричное значение, получим вторую подгруппу значений поля DATE_TIME_IDDOC (сред(DATE_TIME_IDDOC ,9,6)) таблицы 1SJOURN. Тип - Число(4). |
EVENTIDTA | ID документа, на котором установлена ТА. Тип - Строка(9). |
DBSIGN | Код базы УРИБ (ТекущаяИБКод()). Тип - Строка(3). |
DBSETUUID | GUID базы УРИБ. Тип - Строка(36). |
SNAPSHPER | Периодичность итогов регистров остатков (устанавливается в меню Операции - Управление оперативными тогами - Периодичность сохранения остатков). F – 5 дней. C – Декада (10 дней). T – 15 дней. M – месяц. Тип - Строка(1). |
ACCDATE | Дата актуальности бухгалтерских итогов. Тип - Дата (для SQL DateTime). |
FLAGS | ??? Тип - Число(4). |
Периодичность итогов регистров оборотов: «D» - День, «W» - Неделя, «C» - Декада, «M» - Месяц, «Q» - Квартал, «Y» - Год.
Краткое описание: таблица предназначена для хранения информации о последнем ID коде документов и справочников.
Название поля | Описание |
---|---|
TYPEID | Идентификатор вида справочника, или же 0 для всех документов (у всех документов сквозная нумерация ID). Тип – Число(4). |
MAXID | Максимальный используемый идентификатор (ID). |
Краткое описание: таблица предназначена для хранения информации о подключенных пользователях. В таблице всегда одна строка. Именно по наличию строки в этой таблице 1С судит об аварийном завершении программы и предлагает переиндексировать ИБ в случае dbf-версии.
Название поля | Описание |
---|---|
USRSCNT | Количество подключенных пользователей к 1С в режиме 1С предприятия. |
NETCHGCN | Счетчик действий пользователей, которые привели к изменению в базе данных (записи в таблицы). Счетчик учитывает количество записей в таблицы (т.е. в случае проведения документа с несколькими движениями учитывается каждое движение). |
Распределённая база содержит две строки - по центральной базе и самой себя, в центральной базе данные по всем распределенным базам.
Краткое описание: таблица содержит записи ИД объектов, которые должны быть выгружены при очередном обмене данных.
Название поля | Описание |
---|---|
DBSIGN | Код ИБ куда должна произойти загрузка, char(3) |
TYPEID | Тип объекта, int |
OBJID | ИД объекта, char(9) |
DELETED | флаг удаления объекта из базы, char(1), (символ D или пусто) |
DWNLDID | идентификатор обмена. при выгрузке 1С ставит уникальное значение, после обмена 1С должна получить это значение от распределённой базы (от DBSIGN), т.о. считается, что выгрузка прошла успешно и записи из таблицы удаляются, иначе записи хранятся и попадают в следующую выгрузку, где им заново присваивается новый идентификатор, char(9) |
Данная часть предназначена для тех, кто пытается разобраться в структуре хранения данных в системе 1С версии 7.7. А также покажет, как можно получать данные напрямую из таблиц 1С, минуя программу 1С. Для понимания того, о чем идет речь в статье, необходимо понимать принципы работы 1С версии 7.7 и иметь начальные навыки работы с SQL Server Enterprise Manager и SQL Server Query Analyzer.
Что являют собой справочники? В понимании 1С - это объекты для хранения условно постоянной информации (константы - для хранения постоянной или очень редко изменяющейся информации).
С токи зрения же теории баз данных, справочники являются типичными таблицами (по одной на каждый справочник). А вот с константами - гораздо сложнее. Точнее с первого взгляда проще, но это только с первого взгляда. Так, все константы хранятся в одной таблице _1SCONST, но также в этой таблице хранятся и значения всех периодических реквизитов справочников.
Все таблицы справочников имеют почти одинаковые имена, эти таблицы именуются первыми двумя символами SC, далее следует десятичное представление справочника. Это десятичное представление - сквозная нумерация всех объектов внутри конфигурации (включая реквизиты и т.д.). В наших примерах таблица справочника будет иметь имя таблицы SC19, т.е. этот справочник был создан 19-ым по номеру в конфигурации среди всех объектов.
Название поля | Описание |
---|---|
ROW_ID | Порядковый номер записи в таблице. Тип - Число(int). |
ID | ID элемента, тип «строка», по этому полю осуществляется связь с таблицами, где в качестве реквизита выбирается справочник, а также с таблицей констант (для периодических реквизитов). Нумерация сквозная, именно этот код должен быть уникальным в пределах таблицы. Тип - Char(9). |
CODE | Номер элемента (Код) справочника. Тип - Char(n), где n - длина номера справочника. Если длина кода = 0, это поле отсутствует. |
DESCR | Наименование элемента. Тип - Char(n), где n - длина номера справочника. Если длина кода = 0, это поле отсутствует. |
ISMARK | Флаг пометки на удаление элемента. Тип - bit. 0 - не помечен, 1 - помечен. |
VERSTAMP | Количество изменений записи таблицы. Изменением считается любое действие "Изменить (открыть)" + действия при изменении структуры. Тип - Integer. |
SPххx | Реквизит справочника(типы: Numeric, DateTime, Char(n), n=1:999). |
TSPххx | Дополнение к реквизиту, заполняется только для неопределенных типов значений (используется совместно с полем «SPххx»). Тип - Char(3). |
PARENTID | ID элемента, являющегося родителем (группой) для текущей записи (элемента). Поле связано с полем ID или же если родителя нет - заполнено пустым ID ' 0 '. Тип - Сhar(9). Это поле появляется в таблице справочника лишь тогда, когда справочник имеет больше 1 уровня. |
ISFOLDER | Флаг того, что запись является элементом или группой. Для групп это поле равно 1, для элементов - 2. Тип - tinyint (0-255). |
PARENTEXT | ID элемента, являющегося владельцем (этот элемент подчинен владельцу с этим ID). Тип - Сhar(9). Это поле появляется в таблице справочника лишь тогда, когда справочнику установлено значение "Подчинен" одному из справочников системы. |
Особое внимание надо уделить полям «TSP». Это поле создается лишь тогда, когда реквизит (измерение, ресурс) имеет неопределенный тип (длина ID кода 23 символа). Опытным путем было установлено, что это поле по умолчанию заполняется пустой строкой (3 пробела).
Описание таблицы _1SCONST находится выше.
Необходимо только учесть, что значения периодических реквизитов элементов справочников хранятся в таблице _1SCONST с заполненным полем OBJID, которое равно полю ID таблицы справочника. Для констант же значение поля OBJID всегда равно ' 0 '. Также необходимо учесть, что выбрать все периодические значения всех реквизитов одного элемента (одной записи) невозможно. Дело в том, что в поле OBJID хранится краткий (строка 9 символов) ID, а такое значение ID не подразумевает определения вида справочника. Соответственно чтобы получить значение конкретного периодического реквизита надо знать десятичное значение реквизита (_StrToID). Т.е., выбрав только с условием по OBJID получим периодические реквизиты всех справочников, с таким ID, а не только одного. Но, поскольку нумерация всех ID объектов конфигурации (включая и все реквизиты) сквозная, то не может быть в двух разных справочниках реквизитов с одинаковым ID реквизита. Соответственно, необходимым условием для получения значений периодических реквизитов является как условие по полю OBJID (ID элемента справочника), так и по полю ID (десятичное значение ID реквизита справочника).
Получение всех записей справочника (кроме периодических реквизитов). SC19 - имя таблицы справочника.
SELECT * FROM SC19
Получение записей только групп справочника (кроме периодических реквизитов). ISFOLDER - флаг того, что это элемент или группа элементов.
SELECT * FROM SC19 WHERE ISFOLDER = 1
Получение записей только элементов (не групп) справочника (кроме периодических реквизитов).
SELECT * FROM SC19 WHERE ISFOLDER = 0
Получение записей только непомеченных элементов (не групп) справочника (кроме периодических реквизитов). ISMARK - флаг пометки элемента (или группы) на удаление.
SELECT * FROM SC19 WHERE ISFOLDER = 0 AND ISMARK = 0
Получение записей элементов с периодическим реквизитом, значение которого получается на максимальную дату (возможно и будущую). В данном примере условие TabConst.ID = 101 необходимо для отбора периодических значений только по реквизиту с десятичным ID кодом равным 101.
SELECT TabSpr.*, ISNULL(TabConst.VALUE,'') FROM SC19 As TabSpr LEFT OUTER JOIN _1SCONST As TabConst ON ((TabSpr.ID = TabConst.OBJID) AND (TabConst.ID = 101) AND (TabConst.DATE = (SELECT MAX(TabConstl.DATE) FROM _1SCONST AS TabConstl WHERE TabConstl.OBJID = TabConst.OBJID AND TabConstl.ID = TabConst.ID)) )
В этом примере могут возвращаться и значения NULL для тех реквизитов, для которых не были установлены периодические реквизиты:
SELECT TabSpr.*, (SELECT (TabConst.VALUE) FROM _1SCONST As TabConst WHERE (TabSpr.ID = TabConst.OBJID) AND (TabConst.ID = 101) AND (TabConst.DATE = (SELECT MAX(TabConstl.DATE) FROM _1SCONST AS TabConstl WHERE TabConstl.OBJID = TabConst.OBJID AND TabConstl.ID = TabConst.ID) ) ) FROM SC19 As TabSpr
Применение конструкции UNION. В первом запросе получаем список только тех записей, для которых есть установленные периодические реквизиты, а во втором - всех остальных:
SELECT TabSpr.*, ISNULL(TabConst.VALUE,'') FROM SC19 As TabSpr LEFT OUTER JOIN _1SCONST As TabConst ON ((TabSpr.ID = TabConst.OBJID) AND (TabConst.ID = 101)) WHERE (TabConst.DATE = (SELECT MAX(TabConstl.DATE) FROM _1SCONST AS TabConstl WHERE (TabConstl.OBJID = TabConst.OBJID) AND (TabConstl.ID = TabConst.ID) ) ) UNION ALL SELECT TabSpr.*, '' FROM SC19 As TabSpr WHERE NOT EXISTS( SELECT * FROM _1SCONST As TabConst WHERE (TabSpr.ID = TabConst.OBJID) AND (TabConst.ID = 101) )
Получение записей элементов с периодическим реквизитом, на конкретную дату.
Периодический реквизит получается на 11.03.2006. В данном примере условие TabConst.ID = 101 необходимо для отбора периодических значений только по реквизиту с десятичным ID кодом, равным 101.
SELECT TabSpr.*, ISNULL(TabConst.VALUE,'') FROM SC19 As TabSpr LEFT OUTER JOIN _1SCONST As TabConst ON ((TabSpr.ID = TabConst.OBJID) AND (TabConst.ID = 101) AND (TabConst.DATE = (SELECT MAX(TabConstl.DATE) FROM _1SCONST AS TabConstl WHERE TabConstl.OBJID = TabConst.OBJID AND TabConstl.ID = TabConst.ID AND TabConstl.DATE <= Convert(DateTime,'20060311',112) ) ) )
Рассмотрим теперь пример получения значений непериодических реквизитов справочников, которые являются документами или элементами справочника. В системе 1С получение значений таких реквизитов осуществляется обращением к реквизитам и их атрибутам. Например, получение значение реквизита "Менеджер" элемента справочника "Контрагенты" (наименование менеджера). В 1С это легко реализуется, например, если СпрКонтрагенты является объектом "Справочник.Контрагенты" и спозиционирован на конкретном элементе (например "НайтиПоНаименованию("Иванов А. А.")") - то получение менеджера этого элемента осуществляется так: "СпрКонтрагенты.Менеджер.Наименование". Но в самой таблице справочника "Контрагенты" (допустим это таблица "SC191") в поле, отвечающем за реквизит "Менеджер" (например "SP10494") будут значение ID элемента справочника "Менеджеры", а не наименование менеджера (и это вполне нормально). Само же наименование менеджера хранится в другой таблице, это таблица "Сотрудники" (SC258).
Итак, для этого примера надо выполнить запрос, выполняющий 2 действия: 1 - позиционирование на элемент с наименованием "Иванов А. А.", 2 - получение наименования менеджера, для спозиционированного элемента справочника Контрагенты.
Первая часть запроса будет выглядеть так:
SELECT TabSpr.DESCR As Наименование FROM SC191 As TabSpr WHERE TabSpr.DESCR = 'Иванов А. А.'
Но приведенный выше код содержит ошибку. Дело в том, что поле DESCR определено как Char, и его длина строго задана и равна длине наименования для справочника "Контрагенты". Например, если длина наименования 50 символов, то предыдущий запрос надо было написать так:
SELECT TabSpr.DESCR As Наименование FROM SC191 As TabSpr WHERE TabSpr.DESCR = 'Иванов А. А. '
Как видно, в условии происходит сравнение на полную строку, включая недостающие пробелы. Для того, чтобы не заполнять строку поиска недостающими пробелами, можно воспользоваться функцией усечения пробелов справа, или же определить переменную как Char(50) и поиск вести по значению этой переменной.
Для первого случая пример запроса будет такой:
SELECT TabSpr.DESCR As Наименование FROM SC191 As TabSpr WHERE RTRIM(TabSpr.DESCR) = 'Иванов А. А.'
Для второго случая пример запроса будет такой:
DECLARE @NAIM CHAR(50) SET @NAIM = 'Иванов А. А.' SELECT TabSpr.DESCR As Наименование FROM SC191 As TabSpr WHERE TabSpr.DESCR = @NAIM
Вторая часть запроса, получение имени менеджера. Необходимо сделать выборку из 2 таблиц. Это можно сделать как минимум двумя способами. В первом примере выборка из 2 таблиц с условием:
DECLARE @NAIM CHAR(50) SET @NAIM = 'Иванов А. А.' SELECT TabSpr.DESCR As Наименование, TabManag.DESCR As Менеджер FROM SC191 As TabSpr, SC258 As TabManag WHERE TabSpr.DESCR = @NAIM AND TabSpr.SP10494 = TabManag.ID
Во втором примере выборка из 2 таблиц с объединением. Этот код аналогичен предыдущему, за исключением того, что происходит объединение с условием, а не выборка из таблиц с условием:
DECLARE @NAIM CHAR(50) SET @NAIM = 'Иванов А. А.' SELECT TabSpr.DESCR As Наименование, TabManag.DESCR As Менеджер FROM SC191 As TabSpr INNER JOIN SC258 As TabManag ON TabSpr.SP10494 = TabManag.ID WHERE TabSpr.DESCR = @NAIM
Но эти оба примера не совсем корректные. Дело в том, что запросы работают по полном объединении (или полном условии), т.е. в выборку попадут лишь те записи из таблицы Контрагентов, для которых есть записи в таблице Сотрудников (т.е. поле "Менеджер" справочника Контрагенты заполнено). Если же для элемента с наименованием 'Иванов А. А.' поле Менеджер пустое (в таблице находиться или NULL или значение ' 0 ', именно так 1С хранит значения невыбранных реквизитов), то в выборку не попадет строка таблицы, так как не выполняется условие на вхождение в таблицу Сотрудников (в таблице SC258 в столбце ID нет ни одной записи, для которой есть значение ' 0 ').
Первый пример будет выглядеть так:
DECLARE @NAIM CHAR(50) SET @NAIM = 'Иванов А. А.' SELECT TabSpr.DESCR As Наименование, TabManag.DESCR As Менеджер FROM SC191 As TabSpr, SC258 As TabManag WHERE TabSpr.DESCR = @NAIM AND TabSpr.SP10494 *= TabManag.ID
Второй пример выборки со связыванием 2 таблиц:
DECLARE @NAIM CHAR(50) SET @NAIM = 'Иванов А. А.' SELECT TabSpr.DESCR As Наименование, TabManag.DESCR As Менеджер FROM SC191 As TabSpr LEFT OUTER JOIN SC258 As TabManag ON TabSpr.SP10494 = TabManag.ID WHERE TabSpr.DESCR = @NAIM
Рассмотрим пример получения значений записей элементов справочников с их родителями (группами).
Для примера возьмем тот же справочник "Контрагенты" (допустим это таблица "SC191"). Самый простой пример - получение родителя для каждого элемента (не для родителей).
SELECT TabSpr.DESCR As Наименование, TabSprGr1.DESCR As Родитель FROM SC191 As TabSpr LEFT OUTER JOIN SC191 As TabSprGr1 ON TabSpr.PARENTID = TabSprGr1.ID WHERE TabSpr.ISFOLDER = 2
В этом коде условие TabSpr.ISFOLDER = 2 необходимо для отбора только элементов (не групп). Левое внешнее связывание применяется для того чтобы вывести все элементы, а не только те, у которых есть выбранные родители (в случае с INNER JOIN).
Рассмотрим теперь более сложный пример. Получим всех родителей для элементов (т.е. включая и родителей родителей). Для получения родителей надо знать максимальное количество родителей. Это число легко узнать с помощью команды "Метаданные.Справочник(х).КоличествоУровней". Приведем пример для значения количества уровней, равного 4.
SELECT TabSprGr3.DESCR As Родитель3, TabSprGr2.DESCR As Родитель2, TabSprGr1.DESCR As Родитель, TabSpr.DESCR As Наименование FROM SC191 As TabSpr LEFT OUTER JOIN SC191 As TabSprGr1 ON TabSpr.PARENTID = TabSprGr1.ID LEFT OUTER JOIN SC191 As TabSprGr2 ON TabSprGr1.PARENTID = TabSprGr2.ID LEFT OUTER JOIN SC191 As TabSprGr3 ON TabSprGr2.PARENTID = TabSprGr3.ID WHERE TabSpr.ISFOLDER = 2
В этом коде условие TabSpr.ISFOLDER = 2 необходимо для отбора только элементов (не групп). Количество внешних связываний равно количеству уровней справочника - 1 (в приведенном примере 3).
В результате выполнения данного запроса получаются данные по всем возможным родителям, даже если нет родителей у элемента 1 или 2, то все равно будет 3 колонки с родителями, но в таком случае в качестве родителя будет значение NULL. Для того чтобы в первой колонке всегда был родитель (если есть у элемента родитель), то необходимо выполнить вот такой запрос:
SELECT CASE WHEN TabSprGr3.DESCR IS NOT NULL THEN TabSprGr3.DESCR ELSE CASE WHEN TabSprGr2.DESCR IS NOT NULL THEN TabSprGr2.DESCR ELSE TabSprGr1.DESCR END END As Родитель3, CASE WHEN (TabSprGr2.DESCR IS NOT NULL) AND (TabSprGr3.DESCR IS NOT NULL) THEN TabSprGr2.DESCR ELSE TabSprGr1.DESCR END As Родитель2, CASE WHEN (TabSprGr2.DESCR IS NOT NULL) AND (TabSprGr3.DESCR IS NOT NULL) THEN TabSprGr1.DESCR ELSE NULL END As Родитель, TabSpr.DESCR As Наименование FROM SC191 As TabSpr LEFT OUTER JOIN SC191 As TabSprGr1 ON TabSpr.PARENTID = TabSprGr1.ID LEFT OUTER JOIN SC191 As TabSprGr2 ON TabSprGr1.PARENTID = TabSprGr2.ID LEFT OUTER JOIN SC191 As TabSprGr3 ON TabSprGr2.PARENTID = TabSprGr3.ID WHERE TabSpr.ISFOLDER = 2
В этом коде условие TabSpr.ISFOLDER = 2 необходимо для отбора только элементов (не групп).
Вместо проверки на NULL можно воспользоваться функцией COALESCE, которая ищет слева направо в переданных параметрах значения не равные NULL, но тогда усложняется процесс получения различных значений групп на каждом уровне, поэтому рекомендуется это делать уже на клиентском приложении (после выполнения запроса).
Следующий этап - выборка элементов из подчиненного справочника. Для примера возьмем справочник "Контрагенты" (SC191) и справочник "ДенежныеСчета" (SC146), подчиненный справочнику Контрагенты.
SELECT TabSpr.DESCR As Наименование, TabRS.SP143 As НомерСчета FROM SC191 As TabSpr LEFT OUTER JOIN SC146 As TabRS ON TabSpr.ID = TabRS.PARENTEXT WHERE TabSpr.ISFOLDER = 2
В этом коде условие TabSpr.ISFOLDER = 2 необходимо для отбора только элементов (не групп). Ведь в 1С подчиненные элементы справочника существуют только у элементов. В результат выборки попадут и помеченные на удаление подчиненные элементы. Если надо получить только непомеченные подчиненные элементы (расчетные счета), тогда текст запроса должен быть таким:
SELECT TabSpr.DESCR As Наименование, TabRS.SP143 As НомерСчета FROM SC191 As TabSpr LEFT OUTER JOIN SC146 As TabRS ON TabSpr.ID = TabRS.PARENTEXT AND TabRS.ISMARK = 0 WHERE TabSpr.ISFOLDER = 2
В этом коде условие TabRS.ISMARK = 0 необходимо включать как раз в условие связывания, так как если включить условие в раздел WHERE - то те записи справочника Контрагенты, у которых есть запись в подчиненном справочнике и этот элемент помечен на удаление, вообще не попадут в результат запроса.
Данная часть предназначена для тех, кто пытается разобраться в структуре хранения данных в системе 1С версии 7.7. А также покажет, как можно получать данные напрямую из таблиц 1С, минуя программу 1С. Для понимания того, о чем идет речь в статье, необходимо понимать принципы работы 1С версии 7.7 и иметь начальные навыки работы с SQL Server Enterprise Manager и SQL Server Query Analyzer.
Следует учесть, что в статье рассматриваются только примеры для SQL формата базы данных. Для DBF формата есть некоторые особенности, и не все запросы буду идентичны как для DBF, так и для SQL формата баз. В любом случае для выполнения запросов к DBF необходимы или ODBC или OLEDB драйвера. Можно использовать любой драйвер, работающий с DBASE2 форматом DBF. Опыт показывает, что наиболее сопоставимы по тексту запросов драйвера Visual FoxPro (ведь MS SQL Server и MS Visual FoxPro принадлежат одной фирме, и есть вероятность, что тексты запросов будут унифицироваться в будущем). Я советую использовать драйвер OLE DB Visual FoxPro 9.0, так как в нем меньше ограничений и он поддерживает больше функций и методов по сравнению с ODBC Visual FoxPro 6.0. Скачать последний драйвер (OLE DB Visual FoxPro 9.0) можно по этой ссылке: http://www.microsoft.com/downloads/details.aspx?FamilyId=E1A87D8F-2D58-491F-A0FA-95A3289C5FD4&displaylang=en.
Для того, чтобы получить ID имен таблиц, достаточно просмотреть файл КаталогоИБ()+"1cv7.dds" ("1cv7.dd" для DBF формата базы).
Для получения ID объектов 1С можно воспользоваться компонентой 1С++ (http://www.1cpp.ru/) позволяющей получать ID объектов 1С прямо из 1С (например, преобразование значения ТекущийДокумент() в его ID (строка(9), или в длинную строку ИД (строка(13)), включая вид документа, или в самый длинный ID длиной 23 символа (строка(23)), включая ID типа, вида и самого объекта).
Также получение ID объектов рассмотрено в этой статье: http://www.sinor.ru/~my1c/knowhow/get_id.html
Как формируется ID код, можно почитать выше.
Для начала рассмотрим диаграмму связей таблиц подсистемы оперативного учета (диаграмма не полная, но отражает тот минимум, который необходим для получения выборок по регистрам или документам).
DH - таблицы документов (реквизитов шапки). Создаются при первом добавлении реквизита шапки в документ.
DT – таблицы документов (реквизитов табличной части). Создаются при первом добавлении реквизита табличной части в документ.
_1SJOURN – таблица документов (общих и системных реквизитов). Именно это и есть таблица полного журнала документов. Все остальные журналы формируются по полному журналу с дополнительным отбором по типам документов.
_1SCRDOC – Таблица подчиненных документов. В этой таблице хранятся ссылки документов-родителей на подчиненные документы.
В системе 1С 7.7 документы всегда хранятся максимум в 2 таблицах (т.е. табличная часть может быть только одна, и все данные табличной части хранятся в этой таблице) (за исключением общих реквизитов, которые хранятся в одной общей таблице).
Данные документов хранятся в 2 таблицах: DHххх и DTххх, где ххх – десятичный идентификатор вида документа (в файле 1cv7.MD). Непосредственные номера таблиц документов можно посмотреть в файле 1cv7.dds (для dbf версии - 1cv7.dd).
В таблице DHххх – хранятся данные реквизитов шапки (за исключением общих реквизитов). Т.е. на каждый реквизит отведено минимум одно поле таблицы (может быть и 2 в случае неопределенного реквизита). Также в этой таблице хранятся данные по некоторым реквизитам табличной части (рассмотрим позже). В таблице DTххх – хранятся данные реквизитов табличной части (каждый реквизит минимум одно поле таблицы плюс системное поле LINENO_ - номер строки документа). Отдельно следует рассмотреть реквизиты табличной части типа Число (в понимании 1С), по которым установлен флаг «Итог по колонке» в свойствах реквизита. Итог по этим реквизитам хранится в таблице реквизитов шапки (DHххх), причем название поля совпадает для таблицы шапки (DH) и для таблицы табличной части (DT).
В таблице реквизитов шапки хранятся все реквизиты за исключением общих реквизитов и атрибутов документов (также и системных, таких как время, флаг проведения и удаления). Они все хранятся в одной общей таблице журналов _1SJOURN (1SJOURN для dbf). Описание полей таблицы _1SJOURN можно посмотреть выше.
Таблица журналов _1SJOURN, таблица реквизитов шапки DHххх и таблица реквизитов табличной части DTххх «связаны» по полю IDDOC. Связь эта «мнимая» (т.е. значения полей как бы равны для соответствующих строк), но это не стандартные связи SQL.
Поэтому необходимо учесть, что 1С не создает связей для ограничения целостности данных (связи между полями) средствами MS SQL. Также особое внимание надо уделить тому, что все поля создаются как NOT NULL, т.е. в них нельзя записывать значения типа NULL (видимо поэтому и нет связей). И если значение пустое (в понимании SQL должно быть NULL), то 1С заполняет это поле таким образом:
Получение всех документов одного вида со всеми реквизитами шапки (кроме общих):
SELECT * FROM DH14
Получение всех документов одного вида со всеми реквизитами шапки (включая все общие и системные реквизиты):
SELECT TabJ.*, Tab1.* FROM DH14 As Tab1 INNER JOIN _1SJOURN As TabJ ON (Tab1.IDDOC = TabJ.IDDOC)
Получение всех документов одного вида со всеми реквизитами шапки (включая все общие и системные реквизиты), а также всех реквизитов табличной части:
SELECT TabJ.*, Tab1.*, Tab2.* FROM DH14 As Tab1 INNER JOIN _1SJOURN As TabJ ON (Tab1.IDDOC = TabJ.IDDOC) INNER JOIN DT14 As Tab2 ON (Tab1.IDDOC = Tab2.IDDOC)
Получение всех документов одного вида со всеми реквизитами шапки (включая все общие и системные реквизиты) а также всех реквизитов табличной части с фильтром по дате документа (документов за один день).
В этом примере необходимо учесть, что дата документа храниться в SQL и DBF формате базы по-разному. Но, в любом случае, дата документа храниться в таблице журналов _1SJOURN, рассмотрим пример для SQL базы:
SELECT TabJ.*, Tab1.*, Tab2.* FROM DH14 As Tab1 INNER JOIN _1SJOURN As TabJ ON (Tab1.IDDOC = TabJ.IDDOC) INNER JOIN DT14 As Tab2 ON (Tab1.IDDOC = Tab2.IDDOC) WHERE LEFT(TabJ.DATE_TIME_IDDOC,8) = '20050219'
Получение списка документов родителей по подчиненному документу (получение только общих реквизитов с отборами и системных реквизитов).
SELECT TabJ.* FROM _1SCRDOC As TabRod INNER JOIN _1SJOURN As TabJ ON (SUBSTRING(TabRod.PARENTVAL,7,9) = TabJ.IDDOC) WHERE (TabRod.MDID = 0) -- только документы, без граф отбора AND (TabRod.CHILDID = ' 6C3RK ')
где ' 6C3RK ' – ID документа родителя
Получение списка подчиненных документов (получение только общих реквизитов с отборами и системных реквизитов).
SELECT TabJ.* FROM _1SCRDOC As TabRod INNER JOIN _1SJOURN As TabJ ON (TabRod.CHILDID = TabJ.IDDOC) WHERE (TabRod.MDID = 0) -- только документы, без граф отбора AND (TabRod.PARENTVAL = @ИдДокРодителя) ORDER BY TabRod.CHILD_DATE_TIME_IDDOC
где @ИдДокРодителя – ID документа родителя (включая тип и вид документа, т.е. 23 символа), получить такой ID можно с помощью метода ЗначениеВСамуюДлиннуюСтрокуБД(ТекДок) класса MetaDataWork компоненты 1С++.
Получение списка только проведенных документов. Для отбора только проведенных документов можно воспользоваться значением поля CLOSED таблицы _1SJOURN. Так как первый бит его отвечает за то, проведен документ или нет, соответственно, сделав унарное умножение с 1, получим результат в виде 1 - проведен, 0 - непроведен.
SELECT TabJ.*, Tab1.* FROM DH14 As Tab1 INNER JOIN _1SJOURN As TabJ ON (Tab1.IDDOC = TabJ.IDDOC) WHERE TabJ.CLOSED&1 = 1
Получение списка документов, принадлежащих разным компонентам (у которых включен флажок принадлежности к типам учета). Для отбора документов по типам учета применяется унарное умножение поля APPCODE с требуемым значением. Список значений для отбора по типам учета:
Соответственно, если надо выбрать документы, принадлежащие оперативному и бухгалтерскому учету, надо наложить бинарную маску (бинарное умножение) на 000001 и 000100, или же вместе 000101, или же в десятичной системе 5.
SELECT TabJ.*, Tab1.* FROM DH14 As Tab1 INNER JOIN _1SJOURN As TabJ ON (Tab1.IDDOC = TabJ.IDDOC) WHERE TabJ.APPCODE&5 = 5
Получение списка документов по графе отбора.
Все графы отбора хранятся в 1С в таблице _1SCRDOC. Также в этой таблице хранятся и ссылки на подчиненные документы. Разница между подчиненными документами и графами отбора в том, что поле MDID для подчиненных документов равно 0, а для граф отбора - десятичному идентификатору графы отбора. Допустим, надо получить список документов с отбором по графе "Контрагент". Получим десятичный ИД этой графы отбора с помощью метода ИДОбъекта класса MetaDataWorks компоненты 1С++. К примеру так: ИДГрафы = глMDW.ИДОбъекта(Метаданные.ГрафаОтбора("Контрагент")), где глMDW = СоздатьОбъект("MetaDataWork"). Отбирать будем по столбцу PARENTVAL, в котором хранится полный идентификатор (23 символа) элемента справочника. Получить полный ИД необходимого элемента можно тем же путем, что и идентификатор для графы отбора, но немного другим методом. Например, отбираем по клиенту, значение которого хранится в переменной ВыбКлиент, тогда полный его идентификатор получаем как: IDКлиента = глMDW.ЗначениеВСамуюДлиннуюСтрокуБД(ВыбКлиент).
SELECT TabJ.IDDOC AS Документ, TabJ.IDDOCDEF AS Документ_вид, TabJ.DOCNO AS НомерДок FROM dbo._1SCRDOC AS TabGraf (NOLOCK) INNER JOIN _1SJOURN AS TabJ (NOLOCK) ON (TabJ.IDDOC = TabGraf.CHILDID) WHERE MDID = 14730 -- ИД графы отбора AND PARENTVAL = 'B1 4U 2 ' --23 ID элемента справочника ORDER BY TabJ.DATE_TIME_IDDOC
где соответственно ИДГрафы = 14730, а IDКлиента = 'B1 4U 2 '.
Физически регистры остатков состоят из двух таблиц: таблица остатков RGххх и таблица движений RAххх. В таблице движений хранятся все движения документов по регистрам. Список документов, которые сделали движения по регистру, можно получить, выбрав записи из таблицы журналов _1SJOURN с условием равенства поля RFxxx 1 (или не равно 0), например (в случае, если идентификатор регистра 16, т.е. таблица движений = RG16, а таблица остатков = RA16), для получения всех документов, сделавших движения по регистру, можно сделать так:
SELECT * FROM _1SJOURN WHERE (RF16 = 0x1) AND (CLOSED&1=1) AND (APPCODE&1=1)
где CLOSED&1=1 – ограничение на выбор записей только проведенных документов. Фактически при корректных записях в базе, это условие лишнее, но при каких-то «глюках» таким условием можно «отловить» непроведенные документы с существующими движениями (к сожалению, такое может встречаться).
А вот условие APPCODE&1=1 означает отобрать документы, принадлежащие оперативному учету. Подробнее о полях таблицы _1SJOURN см. выше.
Рассмотрим более подробно, как хранятся данные в таблицах RG и RA.
В таблицу RA записываются все движения документов, с учетом флага прихода или расхода. Т.е. в таблице RA хранятся сведения о всех движениях документов, которые сделали движения по этому регистру. Соответственно методы "ДвижениеПриходВыполнить()/ДвижениеПриход()" записывают одну строку в таблицу RA с флагом приход (DEBKRED = 0), а методы "ДвижениеРасходВыполнить()/ДвижениеРасход()" также добавляют одну запись в таблицу RA, но уже с флагом расхода (DEBKRED = 1). Соответственно, если в регистре отражается приход и расход по складу, и самое первое движение по регистру есть первый приход на склад (т.е. это первое поступление на склад, до этого на складе ничего не было), то для получения остатка на складе нам надо сложить все приходы и вычесть все расходы. Хорошо, когда таких приходов немного, и с начала заполнения таблиц прошло пару месяцев. Теперь представим, что система работает пару лет. В таком случае для получения остатка на складе нам надо выполнить операцию сложения всех приходов и вычитания всех расходов с начала работы. А если еще учесть, что остатки нам надо считать каждый раз при формировании нового движения расхода (внесения новой записи в таблицу) (для контроля остатка или для расчета себестоимости), то налицо лишние операции сканирования всех записей таблицы RA. Для того чтобы не делать таких пересчетов, 1С разработали регистры остатков таким образом, что они состоят из двух таблиц. 1 - RA (хранятся все движения с флагом прихода или расхода), 2 - RG для хранения промежуточных итогов (как бы заблаговременно подсчитанные итоги по таблице RA за какой-то период).
Рассмотрим получение остатков и оборотов по регистру остатков.
Период хранения остатков - месяц. Исходная таблица остатков (RG):
PERIOD | SP20 | SP22 | SP21 |
---|---|---|---|
2005-02-01 00:00:00.000 | ' AA ' | ' 1A ' | 35.00 |
2005-03-01 00:00:00.000 | ' AA ' | ' 1A ' | 20.00 |
Исходная таблица движений (RA):
IDDOC | LINENO_ | ACTNO | DEBKRED | IDOCDEF | DATE_TIME_IDDOC | SP20 | SP22 | SP21 |
---|---|---|---|---|---|---|---|---|
1 | 0 | 1 | 0 | 12 | '200502157579C0 1 ' | ' AA ' | ' 1A ' | 10.00 |
2 | 0 | 1 | 0 | 12 | '20050215759EHS 2 ' | ' AA ' | ' 1A ' | 10.00 |
6 | 0 | 1 | 0 | 23 | '200502157QOSK0 6 ' | ' AA ' | ' 1A ' | 15.00 |
7 | 0 | 1 | 1 | 23 | '200503013KLMO0 7 ' | ' AA ' | ' 1A ' | 15.00 |
Данное описание строится на регистрах остатков, регистры оборотов будут рассмотрены позже.
При описании таблицы остатков (RG) необходимо отметить важную особенность.
Особенность заключается в том, что записи в таблице сортируются по периодам остатков (колонка «PERIOD» и это поле является кластерным индексом (в состав индекса также всегда входит и 1 измерение регистра), по умолчанию периодичность регистров остатков - месяц, именно этот случай и рассматривается). В поле «PERIOD» хранится ДАТА НАЧАЛА периода (МЕСЯЦА). Исключением является период, совпадающий с точкой актуальности, в этом периоде все записи относятся не к концу месяца, а к времени или документу ТА (рассчитаны по этот документ, т.е. с учетом его движений, если документ последовательности проведен). Т.е. другими словами, в этой таблице всегда записи или на конец месяца, или на ТА.
Краткое описание: таблица предназначена для хранения итогов по периодам в разрезе по измерениям и по всем ресурсам регистра.
Название поля | Описание |
---|---|
PERIOD | Период остатков. Всегда равен началу периода (месяц для нашего примера). Тип - DateTime (для dbf - Date). |
SPххx | Измерения, ресурсы или атрибуты регистра (типы: Numeric, DateTime, Char(n), n=1:999). |
TSPyyy | Дополнение к измерению или реквизиту, заполняется только для неопределенных типов значений (используется совместно с полем «SPyyy»). Тип - Char(3). |
Краткое описание: таблица предназначена для хранения движений по регистрам (включая все измерения, ресурсы и реквизиты регистра).
Название поля | Описание |
---|---|
IDDOC | ID документа, тип «строка», по этому полю осуществляется связь с таблицей документов («1SJOURN» или «_1SJOURN» для SQL). Тип - Char(9). |
LINENO_ | Номер строки документа (то, что устанавливается методом «ПривязатьСтроку()»). Тип - SmallInt. |
ACTNO | Порядковый номер движения (это не номер строки, а именно движения). Тип - Integer. |
DEBKRED | Флаг прихода или расхода. 0 - приход, 1 - расход. Тип – bit. |
IDOCDEF | Поле присутствует лишь тогда, когда в свойствах регистра установлен флажок «Быстрая обработка движений». ID вида документа (из метаданных, причем номер, а не 36-ричное представление). Тип - Integer. |
DATE_TIME_IDDOC | Поле присутствует лишь тогда, когда в свойствах регистра установлен флажок «Быстрая обработка движений». Значения этого поля идентичны значениям поля «DATE_TIME_IDDOC» таблицы «_1SJOURN» (журналов). Тип - Char(23). |
SPххx | Измерения, ресурсы или атрибуты регистра (типы: Numeric, DateTime, Char(n), n=1:999). |
TSPyyy | Дополнение к измерению или реквизиту, заполняется только для неопределенных типов значений (используется совместно с полем «SPyyy»). Тип - Char(3). |
Особое внимание надо уделить полям «TSP». Это поле создается лишь тогда, когда реквизит (измерение, ресурс) имеет неопределенный тип (длина ID кода 23 символа). Опытным путем было установлено, что это поле по умолчанию заполняется пустой строкой (3 пробела).
Получение итогов на ТА или на конец периода (месяца в нашем случае).
Поскольку итоги в таблице RG хранятся на ТА или на конец месяца (для нашего примера), то для получения итогов не обходимо лишь указать условие по периоду, на который нам надо получить итоги.
SELECT TabRegOst.SP20 As Товар, Sum(TabRegOst.SP21) As КвоКонОст FROM RG13 As TabRegOst WHERE TabRegOst.PERIOD = @PERIODR GROUP BY TabRegOst.SP20 ORDER BY TabRegOst.SP20
где @PERIODR – переменная типа DateTime, условие по которой служит для получения результата запроса на тот период, который нам необходим.
Например, если надо получить остаток на ТА, при этом ТА = «19.02.2006», то переменной @PERIODR необходимо присвоить значение начала месяца «01.02.2006».
SET @PERIODR = CONVERT(DateTime, '20060201',112)
Если необходимо рассчитать остаток на конец января, то переменной необходимо присвоить значение «01.01.2006»:
SET @PERIODR = CONVERT(DateTime, '20060101',112)
Нетрудно догадаться, что если необходимо получить остаток на конец января, то надо в качестве условия поставить начало февраля (ведь остаток на конец января есть остаток на начало февраля. Именно так и делает 1С при расчете остатка на конец периода).
Спросите, почему присвоение переменной @PERIODR происходит с помощью функции CONVERT, да и еще дата в таком странном формате? Это все дело привычки (дело в том, что в таблице журналов дата документа в поле DATE_TIME_IDDOC как раз храниться в виде строки ГГГГММДД).
Получение сумм прихода и расхода за один месяц (оборотов).
Поскольку обороты хранятся в таблице RA, запрос выполняется только по этой таблице с условием на дату документа, сделавшего движение.
SELECT TabReg.SP20 As Товар, Sum(TabReg.SP21*((DEBKRED+1)%2)) As КвоПриход, Sum(TabReg.SP21*DEBKRED) As КвоРасход, FROM RA13 As TabReg INNER JOIN _1SJOURN As TabJ ON (TabReg.IDDOC = TabJ.IDDOC) WHERE TabJ.DATE_TIME_IDDOC >= @PERIODN AND TabJ.DATE_TIME_IDDOC <= @PERIODK GROUP BY TabReg.SP20 ORDER BY TabReg.SP20
где @PERIODN, @PERIODK – переменные типа Char(8), условия по которых служат для получения результата запроса на тот период, который нам необходим.
В нашем случае @PERIODN = «20060201», @PERIODK = «20060228».
Код для установки переменных:
SET @PERIODR = '20060201' SET @PERIODR = '20060228'
Получение итогов и остатков на произвольную дату.
Поскольку итоги в нашем примере хранятся только на конец месяца или ТА, то необходимо выполнить 2 запроса, первый – для получения итога на начало периода (из примера 1), второй – для получения оборотов (из примера 2).
SELECT TMP.Товар As Товар, SUM(TMP.НачОст) As НачОст, SUM(TMP.Приход) As Приход, SUM(TMP.Расход) As Расход, SUM(TMP.НачОст + TMP.Приход - TMP.Расход) As КонОст FROM ( SELECT TabRegOst.SP20 As Товар, TabRegOst.SP21 As НачОст, 0 As Приход, 0 As Расход FROM RG13 As TabRegOst WHERE TabRegOst.PERIOD = @PERIODR UNION ALL SELECT TabReg.SP20 As Товар, 0 As НачОст, (TabReg.SP21*((DEBKRED+1)%2)) As Приход, (TabReg.SP21*DEBKRED) As Расход FROM RA13 As TabReg INNER JOIN _1SJOURN As TabJ ON (TabReg.IDDOC = TabJ.IDDOC) WHERE TabJ.DATE_TIME_IDDOC >= @PERIODN AND TabJ.DATE_TIME_IDDOC <= @PERIODK ) As TMP GROUP BY TMP.Товар ORDER BY TMP.Товар
где @PERIODR – переменная типа DateTime, условие по которой служит для получения результата запроса на период, меньший на 1 от рассчитываемого. В нашем случае для получения остатка на 20.02.2006 этой переменной необходимо присвоить значения начала января, т.е. «01.01.2006» (так как в случае условия на 01.02.2006 получим остаток или на конец февраля, или на точку актуальности):
SET @PERIODR = CONVERT(DateTime, '20060101',112)
@PERIODN, @PERIODK – переменные типа Char(8), условия по которым служат для получения результата в запросе за определенный период. Для нашего примера @PERIODN = «20060201», @PERIODK = «20060220»:
SET @PERIODR = '20060201' SET @PERIODR = '20060220'
Следует отметить, что условие:
AND TabJ.DATE_TIME_IDDOC <= @PERIODK
предполагает получение остатка на начало дня, это связано с тем, что в поле DATE_TIME_IDDOC хранится не только дата в текстовом виде, но и время, и идентификатор документа. Так, чтобы получить строку представления начала дня, необходимо выполнить функцию СформироватьПозициюДокумента(<Дата>,<Час>,<Мин>,<Сек>,<ФлагКонцаСекунды>), которая и сформирует строку даты и времени в текстовом представлении. Например, для даты 14.03.2006 функция вернет значение '#20060314 0 0 ' (в нашем случае символ # надо убрать). Как видим, первый 0 - это 36 ричное представление времени, а вот второй 0 - это ID документа (невыбранного документа). Но, поскольку сравнению в условию поддается каждый символ, то любая пустая строка будет меньше любой заполненной строки (в нашем случае пробелами и 0). Поэтому, чтобы получить остаток на конец дня, надо или отсечь лишние символы в поле DATE_TIME_IDDOC, или же сравнивать с завтрашней датой. Я предпочитаю второй вариант, так как в таком случае нагрузка на сервер меньше (не требуется выполнять обрезание значений функцией Left), но приведу в качестве примера оба решения:
AND LEFT(TabJ.DATE_TIME_IDDOC,8) <= @PERIODK
или условие предыдущее, но значение переменной устанавливается на дату @PERIODK = «20060221»:
SET @PERIODR = '20060221'
Получение итогов и остатков на документ.
Задача аналогична задаче из предыдущего примера, за исключением того, что условие периода не по дату, а по позицию документа.
SELECT TMP.Товар As Товар, SUM(TMP.НачОст) As НачОст, SUM(TMP.Приход) As Приход, SUM(TMP.Расход) As Расход, SUM(TMP.НачОст + TMP.Приход - TMP.Расход) As КонОст FROM ( SELECT TabRegOst.SP20 As Товар, TabRegOst.SP21 As НачОст, 0 As Приход, 0 As Расход FROM RG13 As TabRegOst WHERE TabRegOst.PERIOD = @PERIODR UNION ALL SELECT TabReg.SP20 As Товар, 0 As НачОст, (TabReg.SP21*((DEBKRED+1)%2)) As Приход, (TabReg.SP21*DEBKRED) As Расход FROM RA13 As TabReg INNER JOIN _1SJOURN As TabJ ON (TabReg.IDDOC = TabJ.IDDOC) WHERE TabJ.DATE_TIME_IDDOC >= @PERIODN AND TabJ.DATE_TIME_IDDOC < @POZDOK ) As TMP GROUP BY TMP.Товар ORDER BY TMP.Товар
где @PERIODR – переменная типа DateTime, условие по которой служит для получения результата запроса на период, меньший на 1 от рассчитываемого. В нашем случае для получения остатка на 20.02.2006 этой переменной необходимо присвоить значения начала Января, т.е. «01.01.2006» (так как в случае условия на 01.02.2006 получим остаток или на конец февраля, или на точку актуальности):
SET @PERIODR = CONVERT(DateTime, '20060101',112)
где @PERIODN – переменная типа Char(8), условие по которой служит для получения результата в запросе на определенный период. Для нашего примера @PERIODN = «20060201».
SET @PERIODR = '20060201'
где @POZDOK – переменная типа char(23), позиция документа (получаемая методом ПолучитьПозицию()).
Следует отметить, что поле DATE_TIME_IDDOC может быть в таблице движений регистра (в случае установленного флага «Быстрая обработка движения» при конфигурировании настроек регистра), в этом случае связь с таблицей журналов (_1SJOURN) необязательна, и условие:
AND TabJ.DATE_TIME_IDDOC < @POZDOK
Может выглядеть вот так:
AND TabReg.DATE_TIME_IDDOC < @POZDOK
И строка:
INNER JOIN _1SJOURN As TabJ ON (TabReg.IDDOC = TabJ.IDDOC)
не нужна.
Т.е. Запрос будет выглядеть так:
SELECT TMP.Товар As Товар, SUM(TMP.НачОст) As НачОст, SUM(TMP.Приход) As Приход, SUM(TMP.Расход) As Расход, SUM(TMP.НачОст + TMP.Приход - TMP.Расход) As КонОст FROM ( SELECT TabRegOst.SP20 As Товар, TabRegOst.SP21 As НачОст, 0 As Приход, 0 As Расход FROM RG13 As TabRegOst WHERE TabRegOst.PERIOD = @PERIODR UNION ALL SELECT TabReg.SP20 As Товар, 0 As НачОст, (TabReg.SP21*((DEBKRED+1)%2)) As Приход, (TabReg.SP21*DEBKRED) As Расход FROM RA13 As TabReg WHERE TabReg.DATE_TIME_IDDOC >= @PERIODN AND TabReg.DATE_TIME_IDDOC < @POZDOK ) As TMP GROUP BY TMP.Товар ORDER BY TMP.Товар
Получение остатков и оборотов за произвольный период.
Приведенный в этом примере код составлен по таблицам конфигурации ПУБ для Украины. Запрос получает остатки и обороты в разрезе продукции за период с "15.01.2006" по "15.02.2006", при условии что период хранения остатков в регистрах остатков - Месяц.
-- Объявление переменных Declare @PERIODR DateTime Declare @PERIODN Char(8) Declare @PERIODK Char(8) Declare @PERIODN2 Char(8) Declare @PERIODK2 Char(8) -- присвоение переменным значений SET @PERIODR = Convert(DateTime,'20051201',112) SET @PERIODN = '20051201' SET @PERIODK = '20060115' SET @PERIODK2 = '20060215' -- сам запрос SELECT TMP.Продукция AS Продукция, SUM(TMP.НачОст) As НачОст, SUM(TMP.Приход) As Приход, SUM(TMP.Расход) As Расход, SUM(TMP.НачОст + TMP.Приход - TMP.Расход) As КонОст FROM ( SELECT TabRegOst.SP1718 As Продукция, TabRegOst.SP1722 As НачОст, 0 As Приход, 0 As Расход FROM RG1714 As TabRegOst WHERE TabRegOst.PERIOD = @PERIODR UNION ALL SELECT TabReg.SP1718 As Товар, (TabReg.SP1722*((DEBKRED+1)%2))- (TabReg.SP1722*DEBKRED) As НачОст, 0 As Приход, 0 As Расход FROM RA1714 As TabReg INNER JOIN _1SJOURN As TabJ ON (TabReg.IDDOC = TabJ.IDDOC) WHERE TabJ.DATE_TIME_IDDOC >= @PERIODN AND TabJ.DATE_TIME_IDDOC < @PERIODK UNION ALL SELECT TabReg2.SP1718 As Товар, 0 As НачОст, (TabReg2.SP1722*((DEBKRED+1)%2)) As Приход, (TabReg2.SP1722*DEBKRED) As Расход FROM RA1714 As TabReg2 INNER JOIN _1SJOURN As TabJ ON (TabReg2.IDDOC = TabJ.IDDOC) WHERE TabJ.DATE_TIME_IDDOC >= @PERIODK AND TabJ.DATE_TIME_IDDOC < @PERIODK2 ) AS TMP GROUP BY TMP.Продукция
Пересчет итогов по одному из регистров остатков.
Данный код приведен как пример работы с регистрами.
-- Создадим процедуры по конвертации 10-36, 30-10 -- Проверим есть ли такие процедуры --IF EXISTS(SELECT name FROM sysobjects WHERE name = 'Convert10To36') DROP Procedure 'Convert10To36' CREATE PROCEDURE [Convert10To36] @Deci INT, @Res36 CHAR(9) OUTPUT AS SET NOCOUNT ON DECLARE @j INT DECLARE @Arr36 CHAR(36) SELECT @Arr36 = '0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZ' SELECT @Res36 = '' SELECT @j = LOG(@Deci)/LOG(36) +1 while @j>0 begin SELECT @Res36 = LTRIM(RTRIM(@Res36)) + SUBSTRING(@Arr36, @Deci/POWER(36,@j-1) +1 ,1) SELECT @Deci = @Deci%POWER(36,@j-1) SELECT @j =@j-1 end GO --IF EXISTS(SELECT name FROM sysobjects WHERE name = 'Convert36To10') DROP Procedure 'Convert36To10' CREATE PROCEDURE [Convert36To10] @Res36 CHAR(9), @Deci INT OUTPUT AS SET NOCOUNT ON DECLARE @j INT DECLARE @Arr36 CHAR(36) SELECT @Arr36 = '0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZ' SELECT @Deci = 0 SELECT @j = 1 while @j <= LEN(LTRIM(RTRIM(@Res36))) begin if @j <> 1 SELECT @Deci = @Deci*36 SELECT @Deci = @Deci + CHARINDEX(SUBSTRING(LTRIM(RTRIM(@Res36)), @j,1),@Arr36) -1 SELECT @j = @j+1 End GO DECLARE @DATE_POSL datetime DECLARE @PERIOD_POSL datetime DECLARE @TIME_POSL int DECLARE @IDDOC_POSL char(9) DECLARE @POZ_POSL char(23) DECLARE @POZ_POSLMAX char(23) DECLARE @DOC_POSL char(23) DECLARE @Res36 char(9) --SELECT @DATE_POSL = Convert(Varchar(23),(SELECT MAX(CURDATE) FROM _1SSYSTEM),112) -- Получаем дату ТА, время ТА в десятичном формате от начала суток*10000, ID -- документа последовательности SELECT @DATE_POSL = MAX(CURDATE), @TIME_POSL = MAX(CURTIME), @IDDOC_POSL = MAX(EVENTIDTA) FROM _1SSYSTEM IF @TIME_POSL <> 0 -- Конвертируем время в 36 ричный вид EXEC Convert10To36 @TIME_POSL, @Res36=@POZ_POSL OUTPUT ELSE SELECT @POZ_POSL= '' -- Если ТА установлена не на документ - тогда время и ID документа будут равны 0 IF @IDDOC_POSL = ' 0 ' SELECT @IDDOC_POSL= '' -- Получаем позицию ТА SELECT @POZ_POSL = Convert(Varchar(23),@DATE_POSL,112)+LTRIM(RTRIM(@POZ_POSL))+@IDDOC_POSL -- Получаем документ ТА, если такой имеется (если ТА находится на документе) SELECT @DOC_POSL = (SELECT DATE_TIME_IDDOC FROM _1SJOURN WHERE DATE_TIME_IDDOC = @POZ_POSL) -- Получаем начало месяца даты ТА SELECT @PERIOD_POSL = DATEADD(dd,1-DATEPART(dd,@DATE_POSL),@DATE_POSL) --SELECT @DATE_POSL,@TIME_POSL,@POZ_POSL,@DOC_POSL,@PERIOD_POSL -- Заполняем временную таблицу периодами пересчета SELECT DISTINCT RG8918.PERIOD As PERIOD, Convert(Varchar(8),RG8918.PERIOD,112) AS PERIODS, (CASE WHEN RG8918.PERIOD <> @PERIOD_POSL THEN Convert(Varchar(8),DATEADD(mm,1,RG8918.PERIOD),112) ELSE @POZ_POSL END) As PERIODPO INTO ##TabPeriod FROM RG8918 GO --SELECT * FROM ##TabPeriod ORDER BY PERIODS -- Получаем курсор для обхода каждой строки временной таблицы ##TabPeriod DECLARE MyCur cursor for SELECT PERIOD, PERIODS, PERIODPO FROM ##TabPeriod ORDER BY PERIOD OPEN MyCur -- переменные для периодов DECLARE @PERIOD datetime DECLARE @PERIODPRED datetime DECLARE @PERIODS varchar(23) DECLARE @PERIODPO varchar(23) -- все делаем в транзакции -- обходим в цикле по каждой строке таблицы (по каждому периоду) FETCH NEXT FROM MyCur INTO @PERIOD, @PERIODS, @PERIODPO WHILE @@FETCH_STATUS = 0 BEGIN -- Получим дату предыдущего периода (месяца) SELECT @PERIODPRED = DATEADD(mm,-1,@PERIOD) PRINT convert(Varchar(8),@PERIODPRED,112)+' # '+@PERIODS+' # '+@PERIODPO --SET HACT_ABORT ON BEGIN TRANSACTION -- очищаем текущий период DELETE FROM RG8918 WHERE PERIOD = @PERIOD -- заполняем текущий период по пересчитанным данным INSERT RG8918 SELECT @PERIOD, T1.SP8908, T1.SP8909, T1.SP8910, T1.SP8911, T1.SP8912, T1.SP8913, T1.SP8914, T1.SP8915, '', T1.SP11200, SUM(T1.SP8916), 0, 0, 0, 0 FROM ( SELECT TabRegOst.SP8908 As SP8908, TabRegOst.SP8909 As SP8909, TabRegOst.SP8910 As SP8910, TabRegOst.SP8911 As SP8911, TabRegOst.SP8912 As SP8912, TabRegOst.SP8913 As SP8913, TabRegOst.SP8914 As SP8914, TabRegOst.SP8915 As SP8915, TabRegOst.SP11200 As SP11200, SUM(TabRegOst.SP8916) As SP8916 FROM RG8918 As TabRegOst (HOLDLOCK) WHERE TabRegOst.PERIOD = @PERIODPRED GROUP BY TabRegOst.SP8908, TabRegOst.SP8909, TabRegOst.SP8910, TabRegOst.SP8911, TabRegOst.SP8912, TabRegOst.SP8913, TabRegOst.SP8914, TabRegOst.SP8915, TabRegOst.SP11200 UNION ALL SELECT TabRegOb.SP8908 As SP8908, TabRegOb.SP8909 As SP8909, TabRegOb.SP8910 As SP8910, TabRegOb.SP8911 As SP8911, TabRegOb.SP8912 As SP8912, TabRegOb.SP8913 As SP8913, TabRegOb.SP8914 As SP8914, TabRegOb.SP8915 As SP8915, TabRegOb.SP11200 As SP11200, SUM(TabRegOb.SP8916*(1-TabRegOb.DEBKRED*2)) As SP8916 FROM RA8918 AS TabRegOb (HOLDLOCK) INNER JOIN _1SJOURN As TabJourn ON (TabRegOb.IDDOC = TabJourn.IDDOC) WHERE TabJourn.DATE_TIME_IDDOC BETWEEN @PERIODS AND @PERIODPO GROUP BY TabRegOb.SP8908, TabRegOb.SP8909, TabRegOb.SP8910, TabRegOb.SP8911, TabRegOb.SP8912, TabRegOb.SP8913, TabRegOb.SP8914, TabRegOb.SP8915, TabRegOb.SP11200 ) AS T1 GROUP BY T1.SP8908, T1.SP8909, T1.SP8910, T1.SP8911, T1.SP8912, T1.SP8913, T1.SP8914, T1.SP8915, T1.SP11200 HAVING SUM(T1.SP8916) <> 0 COMMIT TRANSACTION FETCH NEXT FROM MyCur INTO @PERIOD, @PERIODS, @PERIODPO END CLOSE MyCur DEALLOCATE MyCur SELECT * FROM ##TabPeriod ORDER BY PERIODS DROP TABLE ##TabPeriod
Оригинал статьи расположен по адресу http://www.metaprog.co.ua/secrprog/.
Перейти на главную страничку сайта (список статей, файлы для скачивания)
© 2007 http://www.script-coding.info/ При любом использовании материалов сайта обязательна ссылка на него как на источник информации, а также сохранение целостности и авторства материалов.