Форум программистов «Весельчак У»
  *
Добро пожаловать, Гость. Пожалуйста, войдите или зарегистрируйтесь.
Вам не пришло письмо с кодом активации?

  • Рекомендуем проверить настройки временной зоны в вашем профиле (страница "Внешний вид форума", пункт "Часовой пояс:").
  • У нас больше нет рассылок. Если вам приходят письма от наших бывших рассылок mail.ru и subscribe.ru, то знайте, что это не мы рассылаем.
   Начало  
Наши сайты
Помощь Поиск Календарь Почта Войти Регистрация  
 
Страниц: [1]   Вниз
  Печать  
Автор Тема: Excel/msquery+sql+vba  (Прочитано 11676 раз)
0 Пользователей и 1 Гость смотрят эту тему.
vanillavera
Новенький

ru
Offline Offline

« : 15-07-2010 07:53 » 

Уважаемые Гуру и все кто сможет помочь...

Не уверена что выбрала правильный раздел..

Расскажу суть проблемы с самого начала.
В организации 1С ТиС7.7. Много пользователей, работает вяло, минимальный объем данных выгружается достаточно долго.
Но при этом большинство пользователей извлекает по большей части сходные данные, просто по-разному организованные.
Возникла идея ограничить доступ пользователям к 1С, обеспечив их при этом необходимыми данными в виде готовых динамических отчетов в Excel(в форме сводных таблиц, чтобы каждый смотрел как ему удобно). Для этого выгружать некие срезы БД (за день, неделю, месяц),  делать из них выбор необходимой информации посредством SQL запроса в готовые отчеты, где потом посредством макросов рассчитывать необходимые аналитики.
Данные выгружаются в DBF файлы.
Ввиду моего скудного знания SQL(и написания запросов SQL на VBA) запросы делаю посредством MS Query.
И собственно всё неплохо, если требуется простое соединение таблиц MS Query справляется нормально, все работает.
Проблемы возникают, если нужно сделать полное внешнее соединение таблиц. MS Query умирает.
Да собственно и если соединять более двух таблиц уже та же история.
Что вообще лучше сделать?
Если запрос SQL будет написан текстом непосредственно в макросе, то больше шансов что он выполнится, чем с использованием MS Query?
Если не использовать выгрузку в DBF файлы, а подключаться к тем же данным в SQL формате на сервере, поможет ли это? ускорит процесс? есть шанс, что  MS Query сможет с этим справится?

А может есть какие более изящные решения, подскажите пожалуйста!
Записан
Dimka
Деятель
Команда клуба

ru
Offline Offline
Пол: Мужской

« Ответ #1 : 15-07-2010 09:15 » 

Цитата: vanillavera
если нужно сделать полное внешнее соединение таблиц. MS Query умирает.
Да собственно и если соединять более двух таблиц уже та же история.
Не очень понятно, что именно ты под этим подразумеваешь. Можешь построить запрос в MS Query, не выполняя его, и потом скопировать сюда SQL код твоего запроса?

У меня есть подозрение, что умирает всё из-за попытки получить полное декартово произведение двух и более таблиц.
Записан

Программировать - значит понимать (К. Нюгард)
Невывернутое лучше, чем вправленное (М. Аврелий)
Многие готовы скорее умереть, чем подумать (Б. Рассел)
vanillavera
Новенький

ru
Offline Offline

« Ответ #2 : 15-07-2010 09:36 » 

У меня есть подозрение, что умирает всё из-за попытки получить полное декартово произведение двух и более таблиц.

Не думаю...
SELECT access.ZONA, access.KANAL, Sum(access.SUMPROD) AS 'Продажи', Sum(access.SUMDOG) AS 'Продажи_в_ДогЦен'
FROM {oj access access LEFT OUTER JOIN `Y:\Червакова\Выгрузки\Неделя1`\dz.DBF dz ON access.KODCLI = dz.KODCLI}
GROUP BY access.ZONA, access.KANAL
ORDER BY access.ZONA, access.KANAL
UNION
SELECT access.ZONA, access.KANAL, Sum(access.SUMPROD) AS 'Продажи', Sum(access.SUMDOG) AS 'Продажи_в_ДогЦен'
FROM access access, `Y:\Червакова\Выгрузки\Неделя1`\dz.DBF dz
WHERE access.KODCLI <> dz.KODCLI
GROUP BY access.ZONA, access.KANAL
ORDER BY access.ZONA, access.KANAL
« Последнее редактирование: 15-07-2010 09:45 от Джон » Записан
Dimka
Деятель
Команда клуба

ru
Offline Offline
Пол: Мужской

« Ответ #3 : 16-07-2010 10:05 » 

Цитата: vanillavera
FROM access access, `Y:\Червакова\Выгрузки\Неделя1`\dz.DBF dz
WHERE access.KODCLI <> dz.KODCLI
Ты это называешь "полным внешним соединением"?

Это как раз и есть полное декартово произведение за вычетом внутреннего соединения. Результат будет размером от M*N-max(N,M) до M*N.

Сколько записей в таблицах? Например, 1000 на 1000 в таком запросе даст результат в 999000 записей.

Ты лучше объясни цель второго запроса (после UNION) - что ты хочешь получить? Тогда можно будет подобрать менее тяжёлое для машины решение.
Записан

Программировать - значит понимать (К. Нюгард)
Невывернутое лучше, чем вправленное (М. Аврелий)
Многие готовы скорее умереть, чем подумать (Б. Рассел)
vanillavera
Новенький

ru
Offline Offline

« Ответ #4 : 21-07-2010 10:00 » 

Цитата: vanillavera
FROM access access, `Y:\Червакова\Выгрузки\Неделя1`\dz.DBF dz
WHERE access.KODCLI <> dz.KODCLI
Ты это называешь "полным внешним соединением"?

Это как раз и есть полное декартово произведение за вычетом внутреннего соединения. Результат будет размером от M*N-max(N,M) до M*N.

Сколько записей в таблицах? Например, 1000 на 1000 в таком запросе даст результат в 999000 записей.

Ты лучше объясни цель второго запроса (после UNION) - что ты хочешь получить? Тогда можно будет подобрать менее тяжёлое для машины решение.

а как правильно?
хочу получить все данные из таблиц access и dz в одной таблице, при этом поля ZONA,KANAL должны быть общими и содержать как значения для непустых access.SUMPROD access.SUMDOG, так и для непустых DZ из DZ...
как-то так
Записан
Dimka
Деятель
Команда клуба

ru
Offline Offline
Пол: Мужской

« Ответ #5 : 21-07-2010 10:50 » new

vanillavera, вопрос не понял.

В тобою приведённых запросах таблица dz не фигурирует ни в результатах, ни в группировках, ни в условии отбора записей из таблицы access. В таком виде она не нужна.

Лучше на словах (без MS Query) опиши структуру таблиц access и dz и объясни:
1) По каким полям связываются или не связываются таблицы.
2) Какие именно поля ты хочешь получить в результате.
3) Какие группировки (и группировочные функции - суммы, количества, средние, минимумы, максимумы и т.п.), какие условия и какие сортировки тебе нужны.

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

Пример:

Формулировка проблемы в предметной области
Цитата
Выявить продавцов, чьи продажи с начала года превышают средние более, чем на 10%.

Формулировка соответствующего проблеме запроса к базе данных:
Цитата
Информация о продавцах хранится в таблице Salesman. Внутри этой таблицы: информация о накопленной сумме продаж с начала года для каждого продавца хранится в поле YearToDateSales. Имя продавца хранится в поле Name.

В результате должна получиться таблица с одним полем EffectSalesman, в которой перечислены интересующие имена продавцов.

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


Если двигаться дальше, на языке SQL согласно описанию получается следующий запрос:
Код: (SQL)
SELECT Salesman.Name AS EffectSalesman
FROM
  Salesman,
  (
    SELECT AVG(Salesman.YearToDateSales) AS VALUE
    FROM Salesman
  ) AS AverageSales
WHERE Salesman.YearToDateSales > AverageSales.VALUE * 1.1
Здесь используется подзапрос, который из таблицы Salesman получает как бы "временную" таблицу AverageSales, состоящую из 1 поля и 1 записи. Эта "временная" таблица соединяется с таблицей Salesman как полное декартово произведение, так что в результате к таблице Salesman будто бы прибавляют дополнительное поле AverageSales.Value, в котором хранится средняя сумма продаж с начала года в расчёте на одного продавца. Затем по условию отбираются нужные записи. И, наконец, из полученных записей оставляется лишь поле Salesman.Name которое переименовывается в нужное EffectSalesman.
« Последнее редактирование: 21-07-2010 10:52 от Dimka » Записан

Программировать - значит понимать (К. Нюгард)
Невывернутое лучше, чем вправленное (М. Аврелий)
Многие готовы скорее умереть, чем подумать (Б. Рассел)
Страниц: [1]   Вверх
  Печать  
 

Powered by SMF 1.1.21 | SMF © 2015, Simple Machines