Главная страница
    Top.Mail.Ru    Яндекс.Метрика
Форум: "Базы";
Текущий архив: 2007.12.16;
Скачать: [xml.tar.bz2];

Вниз

Оптимизация запроса   Найти похожие ветки 

 
alsov ©   (2007-08-09 12:39) [0]

Приветствую, Мастера

Помогите оптимизировать запрос.

Есть таблица

CREATE TABLE TAB1 (
 obj_id            NUMBER(10,0)   NOT NULL,
 param_id  NUMBER(10,0)   NOT NULL,
 val                       VARCHAR2(2000) NULL
)


Надо сделать выборку объектов со всеми параметрами, у которых имеется связка (param_id=10 and val LIKE "Текст%").
Причем связок таких может быть как угодно много и соединять их можно как по И так и по ИЛИ.

У меня получилось примерно следующее, но мне кажентся этот способ не оптимален.


SELECT *
 FROM tab1
WHERE obj_id IN (
-- первое условие
select obj_id
 from tab1
WHERE (param_id=1 AND VAL LIKE "Пот%")
---
INTERSECT  -- если условие или поменять на UNION
-- второе услвие
select obj_id
 from tab1
WHERE (param_id=2 AND VAL IS NOT NULL)
---
--- если надо еще условий добавляем аналогично
)


Заранее спасибо за любу помощь


 
stanislav ©   (2007-08-09 13:31) [1]

Не совсем понял, а чем так не устраивает ?

SELECT *
from tab1
WHERE (param_id=1 AND VAL LIKE "Пот%") OR (param_id=2 AND VAL IS NOT NULL) OR ....

Какая СУБД?


 
Sergey13 ©   (2007-08-09 13:34) [2]

> [1] stanislav ©   (09.08.07 13:31)
Опередил! 8-)


 
alsov ©   (2007-08-09 15:11) [3]


> stanislav ©   (09.08.07 13:31) [1]

> Какая СУБД?
>

ORACLE


>
> Не совсем понял, а чем так не устраивает ?
>
> SELECT *
> from tab1
> WHERE (param_id=1 AND VAL LIKE "Пот%") OR (param_id=2 AND
> VAL IS NOT NULL) OR ....


подходит, но только для варианта ИЛИ, а для варианта И нет, т.к. нужно вывести obj_id у которых и  (param_id=1 AND VAL LIKE "Пот%"), и (param_id=2 AND VAL IS NOT NULL)  и ...

а если написать  (param_id=1 AND VAL LIKE "Пот%") AND (param_id=2 AND
> VAL IS NOT NULL) то естественно ничего не выведет.

Кроме того нужно чтобы все существующие param выводились для obj_id, попадающего под условие.


 
stanislav ©   (2007-08-09 15:23) [4]

stanislav ©   (09.08.07 13:31) [1]
Если нужно чтобы попадало
условие 1 и условие 2
значит нужно писать условие 1 OR условие 2, но не AND.
А вот что значит в вашем случае ИЛИ ? т.е. если попало условие 1, то условие 2 недолжно попасть?


 
Johnmen ©   (2007-08-09 15:29) [5]

Делай внутреннее соединение таблицы самой с собой столько раз, сколько означенных в [0] "условий", добавляя для каждого равенство на obj_id.


 
Sergey13 ©   (2007-08-09 15:34) [6]

> [3] alsov ©   (09.08.07 15:11)

Тогда у тебя нормальный запрос. Можно еще вместо IN попробовать EXIST и покрутить с ним - иногда производительность выше. Типа

SELECT *
FROM tab1
WHERE exists(
-- первое условие
select 1
from tab1
WHERE (param_id=1 AND VAL LIKE "Пот%")
)
---
AND  -- если условие или поменять на OR
-- второе услвие
(select 1
from tab1
WHERE (param_id=2 AND VAL IS NOT NULL)
)


 
alsov ©   (2007-08-09 16:21) [7]


> Sergey13 ©   (09.08.07 15:34) [6]

Спасибо - попробую что быстрее.

Может еще какие-нибудь варианты будут?


 
viman ©   (2007-08-10 09:26) [8]

select obj_id
from tab1
WHERE (:p_case1 <> 0  and (param_id=1 AND VAL LIKE "Пот%"))
      or (:p_case2 <> 0  and (param_id=2 AND VAL IS NOT NULL))
      or (:p_case3 <> 0  and (...))
      ....
Сделай так и включай нужное условие установкой нужного параметра <> 0, если же условие не нужно, то = 0.


 
alsov ©   (2007-08-10 12:01) [9]

остановился на вот таком варианте

select obj_id
 from tab1
group by obj_id
having count(case when (param_id=1 AND VAL LIKE "Пот%") then 1 end) > 0
    and count(case when (param_id=2 AND VAL IS NOT NULL) then 1 end ) > 0
    and ....


 
Johnmen ©   (2007-08-10 13:29) [10]


> alsov ©   (10.08.07 12:01) [9]
> остановился на вот таком варианте

А как же Кроме того нужно чтобы все существующие param выводились для obj_id, попадающего под условие.?
И зачем сэкономлено на else?


 
alsov ©   (2007-08-13 13:20) [11]


> А как же Кроме того нужно чтобы все существующие param выводились
> для obj_id, попадающего под условие.?

Чуть не дописал


select * from tab1
where obj_id in (
select obj_id
from tab1
group by obj_id
having count(case when (param_id=1 AND VAL LIKE "Пот%") then 1 end) > 0
   and count(case when (param_id=2 AND VAL IS NOT NULL) then 1 end ) > 0
   and ....
)



> И зачем сэкономлено на else?

А зачем писать, если он не нужен
ведь при сравнении с null всегда false


 
Кщд ©   (2007-08-13 14:24) [12]

>alsov ©   (13.08.07 13:20) [11]
если версия поддерживает аналитику, то можно и за один проход



Страницы: 1 вся ветка

Форум: "Базы";
Текущий архив: 2007.12.16;
Скачать: [xml.tar.bz2];

Наверх




Память: 0.47 MB
Время: 0.048 c
15-1194872844
Девушка
2007-11-12 16:07
2007.12.16
Гибрид DBGrid и TreeView


15-1195046851
vasIZmax
2007-11-14 16:27
2007.12.16
Оптимизация по Парето (имхо, неэффективно решается)


2-1195620885
Sergl
2007-11-21 07:54
2007.12.16
Корректность алгоритма программы(сетевой тест) на сокетах


3-1186590083
Shamansky_ne
2007-08-08 20:21
2007.12.16
файлы в базу данных


2-1195745591
Fraider
2007-11-22 18:33
2007.12.16
Лист бокс со скошенным углом





Afrikaans Albanian Arabic Armenian Azerbaijani Basque Belarusian Bulgarian Catalan Chinese (Simplified) Chinese (Traditional) Croatian Czech Danish Dutch English Estonian Filipino Finnish French
Galician Georgian German Greek Haitian Creole Hebrew Hindi Hungarian Icelandic Indonesian Irish Italian Japanese Korean Latvian Lithuanian Macedonian Malay Maltese Norwegian
Persian Polish Portuguese Romanian Russian Serbian Slovak Slovenian Spanish Swahili Swedish Thai Turkish Ukrainian Urdu Vietnamese Welsh Yiddish Bengali Bosnian
Cebuano Esperanto Gujarati Hausa Hmong Igbo Javanese Kannada Khmer Lao Latin Maori Marathi Mongolian Nepali Punjabi Somali Tamil Telugu Yoruba
Zulu
Английский Французский Немецкий Итальянский Португальский Русский Испанский