Форум программистов, компьютерный форум, киберфорум
Delphi: Базы данных
Войти
Регистрация
Восстановить пароль
Блоги Сообщество Поиск Заказать работу  
 
Рейтинг 5.00/4: Рейтинг темы: голосов - 4, средняя оценка - 5.00
1076 / 989 / 340
Регистрация: 07.08.2012
Сообщений: 2,790
Interbase

SQL запрос

19.03.2020, 19:27. Показов 762. Ответов 10

Студворк — интернет-сервис помощи студентам
Не получается сочинить селективный запрос. Суть.
В таблице БД значимые для запроса два поля: дата (DAT) и идентификатор (ID_G).
Примерно так:
12.01.2019 10
16.01.2019 15
18.01.2019 16
20.01.2019 17
23.01.2019 20
25.01.2019 45

08.01.2020 11
09.01.2020 20
11.01.2020 22
20.01.2020 45
21.01.2020 55
22.01.2020 15
31.01.2020 98

Дата в возвращаемом результате не нужна, а используется для определения фильтра периодов в запросе.
Запрос должен вернуть только те значения id_g, которые совпадают в указанных периодах. Периодов всегда только два.
В примере выше периоды такие:
1. dat >= '01.01.2019' and dat < '01.02.2019' и
2. dat >= '01.01.2020' and dat < '01.02.2020'
И запрос должен вернуть значения:
15
20
45
Именно эти значения есть в обоих периодах.
Может ли кто сформировать болванку такого запроса?
0
cpp_developer
Эксперт
20123 / 5690 / 1417
Регистрация: 09.04.2010
Сообщений: 22,546
Блог
19.03.2020, 19:27
Ответы с готовыми решениями:

SQL запрос по фрагменту строки сохраненной в таблице SQL
Прошу помощи. Есть SQL таблица-1, в ней есть столбец по имени Model_Vagona с типом данных ntext Есть SQL таблица-2, в ней есть столбец...

SQL запрос, работающий в MS SQL Menegment'e не работает в делфи
Требуется выполнить запрос по нажатию кнопки, запрос сначала написал в Microsoft SQL Managment, где и написал свою БД, там запрос работает,...

SQL и ComboBox в Delphi. Нужно чтобы при выборе специальности в комбобоксе это значение вносилось в SQL запрос
procedure TForm3.Button2Click(Sender: TObject); begin try ADOQuery1.SQL.Clear; ADOQuery1.SQL.Add('SELECT Абитуриент.фамилия,...

10
5980 / 4555 / 1095
Регистрация: 29.08.2013
Сообщений: 28,185
Записей в блоге: 3
19.03.2020, 22:01
самый простой вариант
SQL
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
CREATE TABLE #t (d DATE, i INT)
INSERT #t SELECT '12.01.2019', 10
INSERT #t SELECT '16.01.2019', 15
INSERT #t SELECT '18.01.2019', 16
INSERT #t SELECT '20.01.2019', 17
INSERT #t SELECT '23.01.2019', 20
INSERT #t SELECT '25.01.2019', 45
INSERT #t SELECT '08.01.2020', 11
INSERT #t SELECT '09.01.2020', 20
INSERT #t SELECT '11.01.2020', 22
INSERT #t SELECT '20.01.2020', 45
INSERT #t SELECT '21.01.2020', 55
INSERT #t SELECT '22.01.2020', 15
INSERT #t SELECT '31.01.2020', 98
 
SELECT DISTINCT t.i
FROM #t t
LEFT JOIN #t t1 ON t1.i=t.i AND t1.d BETWEEN '01.01.2019' AND '01.02.2019'
LEFT JOIN #t t2 ON t2.i=t.i AND t2.d BETWEEN '01.01.2020' AND '01.02.2020'
WHERE t1.i IS NOT NULL AND t2.i IS NOT NULL
1
1076 / 989 / 340
Регистрация: 07.08.2012
Сообщений: 2,790
19.03.2020, 22:51  [ТС]
qwertehok, спасибо. Запрос рабочий (я что-то похожее пытался сделать, но запутался в деталях).
Единственный недостаток - это неприемлемое время выполнение такого запроса.
------ Информация о производительности ------
Время подготовки запроса = 32ms
Время выполнения запроса = 2m 23s 296ms
Среднее время на получение одной записи = 6 230.26 ms
Реальная таблица не большая: ~9000 записей, но 2,5 минуты на выполнение запроса - это, конечно, многовато.
Осторожно спрошу: может как-то можно оптимизировать...

Добавлено через 16 минут
qwertehok, ещё раз спасибо за подсказку (подсказки двигатель прогресса и неопытных разработчиков БД).
Запрос оптимизировал так, что он выполняется меньше чем за секунду (использовано CTE).
0
5980 / 4555 / 1095
Регистрация: 29.08.2013
Сообщений: 28,185
Записей в блоге: 3
19.03.2020, 22:59
индексы?

Добавлено через 6 минут
Цитата Сообщение от Скандербег Посмотреть сообщение
Запрос оптимизировал так, что он выполняется меньше чем за секунду (использовано CTE).
какое CTE в Interbase?
0
1076 / 989 / 340
Регистрация: 07.08.2012
Сообщений: 2,790
19.03.2020, 23:11  [ТС]
Да, нет, дружище, индексы какие были, такие и остались.
А CTE в Firebird'е есть. Пользуюсь уже много лет.
(префикс Interbase в теме лишь потому, что в списке префиксов нет FB)
Если запрос перевести на похожие наименования, то выглядит он так:
SQL
1
2
3
4
5
6
7
8
WITH tt AS
(SELECT id_g
 FROM t
 WHERE d > '01.01.2019' AND d < '01.02.2019')
 
SELECT DISTINCT id_g FROM tt
WHERE id_g IN (SELECT id_g FROM t
                  WHERE d > '01.01.2020' AND d < '01.02.2020')
------ Информация о производительности ------
Время подготовки запроса = 32ms
Время выполнения запроса = 734ms
Среднее время на получение одной записи = 48.93 ms
0
5980 / 4555 / 1095
Регистрация: 29.08.2013
Сообщений: 28,185
Записей в блоге: 3
19.03.2020, 23:20
ааа, Firebird


а так проверь, CTE то нужно что бы сформировать подзапрос для нескольких использований, а у тебя смысла от него нет
SQL
1
2
3
4
5
6
7
8
SELECT t.i
FROM (
SELECT * FROM #t t WHERE t.d BETWEEN '01.01.2019' AND '01.02.2019'
UNION
SELECT * FROM #t t WHERE t.d BETWEEN '01.01.2020' AND '01.02.2020'
) t
GROUP BY t.i
HAVING COUNT(*)=2
0
1076 / 989 / 340
Регистрация: 07.08.2012
Сообщений: 2,790
19.03.2020, 23:46  [ТС]
Запрос шустрый, но результат возвращается с ошибками: может быть одно значение в обоих периодах, но его нет в выходном НД, или в одном периоде есть, в другом нет, а в НД возвращается. Не все, но примерно треть значений не те, какие нужны.
Наши первый и второй запросы возвращают из реальной таблицы одинаковое количество строк и одинаковый набор значений - правильный.
0
5980 / 4555 / 1095
Регистрация: 29.08.2013
Сообщений: 28,185
Записей в блоге: 3
20.03.2020, 07:44
Да, если в одном периоде несколько значений 15, то будет неправильно.
Но тогда нужно просто группировку сделать по id
0
1076 / 989 / 340
Регистрация: 07.08.2012
Сообщений: 2,790
20.03.2020, 08:03  [ТС]
Да, их может быть много в периоде.
А в строке 7 твоего запроса это разве не группировка по id?
Проба сделать группировку во вложенных запросах привела к нулевому результату.
К сожалению, не совсем понимаю логику твоего запроса.
0
5980 / 4555 / 1095
Регистрация: 29.08.2013
Сообщений: 28,185
Записей в блоге: 3
20.03.2020, 09:19
SQL
1
2
3
4
5
6
7
8
SELECT t.i
FROM (
SELECT id FROM #t t WHERE t.d BETWEEN '01.01.2019' AND '01.02.2019' GROUP BY id
UNION
SELECT id FROM #t t WHERE t.d BETWEEN '01.01.2020' AND '01.02.2020' GROUP BY id
) t
GROUP BY t.i
HAVING COUNT(*)>1
вот так
0
1076 / 989 / 340
Регистрация: 07.08.2012
Сообщений: 2,790
20.03.2020, 10:11  [ТС]
Именно так я и пробовал. Получается пустой НД.
В вопросе была только часть основного запроса. Весь запрос затолкал в ХП и он прекрасно работает.
Спасибо за поддержку и внимание.
0
Надоела реклама? Зарегистрируйтесь и она исчезнет полностью.
raxper
Эксперт
30234 / 6612 / 1498
Регистрация: 28.12.2010
Сообщений: 21,154
Блог
20.03.2020, 10:11
Помогаю со студенческими работами здесь

Sql запрос
Вопрос такой Есть запрос: SELECT PFAMILIA AS &quot;Фамилия&quot;,NUMBER AS &quot;Номер участка&quot;' FROM PATIENT WHERE (NUMBER=4) ORDER BY...

sql запрос
Добрый день! Надо было сделать программу для вывода в &quot;StringGrid&quot; sql базы , И собственно нужно было отсортировать по месяцам...

SQL запрос
Здравствуйте! У меня есть две таблицы dat и dat1, есть SQL запрос на вывод из первой таблицы(dat) ср.значения из поля datniz во вторую...

sql запрос
привет всем ребята я создал sql 2000 хранимые процедуры set ANSI_NULLS ON set QUOTED_IDENTIFIER ON GO

Запрос в SQL
Скажите в чем ошибка у меня есть 2 таблицы: mer и mer_ur, в них поля: data, mer, lico. Я делаю запрос (точней пишу) select* from...


Искать еще темы с ответами

Или воспользуйтесь поиском по форуму:
11
Ответ Создать тему
Новые блоги и статьи
Кому нужен AOT?
DevAlt 26.03.2026
Решил сделать простой ланчер Написал заготовку: dotnet new console --aot -o UrlHandler var items = args. Split(":"); var tag = items; var id = items; var executable = args;. . .
Отправка уведомления на почту при изменении наименования справочника
Maks 24.03.2026
Программная отправка письма электронной почты на примере изменения наименования типового справочника "Склады" в конфигурации БП3. Перед реализацией необходимо выполнить настройку системной учетной. . .
модель ЗдравоСохранения 5. Меньше увольнений- больше дохода!
anaschu 24.03.2026
Теперь система здравосохранения уменьшает количество увольнений. 9TO2GP2bpX4 a42b81fb172ffc12ca589c7898261ccb/ https:/ / rutube. ru/ video/ a42b81fb172ffc12ca589c7898261ccb/ Слева синяя линия -. . .
Midnight Chicago Blues
kumehtar 24.03.2026
Такой Midnight Chicago Blues, знаешь?. . Когда вечерние улицы становятся ночными, а ты не можешь уснуть. Ты идёшь в любимый старый бар, и бармен наливает тебе виски. Ты смотришь на пролетающие. . .
SDL3 для Desktop (MinGW): Вывод текста со шрифтом TTF с помощью библиотеки SDL3_ttf на Си и C++
8Observer8 24.03.2026
Содержание блога Финальные проекты на Си и на C++: finish-text-sdl3-c. zip finish-text-sdl3-cpp. zip
Жизнь в неопределённости
kumehtar 23.03.2026
Жизнь — это постоянное существование в неопределённости. Например, даже если у тебя есть список дел, невозможно дойти до точки, где всё окончательно завершено и больше ничего не осталось. В принципе,. . .
Модель здравоСохранения: работники работают быстрее после её введения.
anaschu 23.03.2026
geJalZw1fLo Корпорация до введения программа здравоохранения имела много невыполненных работниками заданий, после введения программы количество заданий выросло. Но на выплатах по больничным это. . .
Контроль уникальности заводского номера
Maks 23.03.2026
Алгоритм контроля уникальности заводского (или серийного) номера на примере нетипового документа выдачи шин для спецтехники с табличной частью, разработанного в конфигурации КА2. Номеклатура. . .
КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin
Copyright ©2000 - 2026, CyberForum.ru