[Moscow.pm] конструкторе sql запросов
Nick Knutov
mail на knutov.com
Чт Апр 3 11:06:35 PDT 2014
Мне уже стало интересно и таки добил.
Полный пример - https://gist.github.com/knutov/9959392
Вопросы:
1) как и можно ли обойтись без временной переменной $t? Читал
документацию, не придумал.
2) как бы вы это улучшили?
3) Этот конструктор не очень хорошо позволяет работать с уловиями IN ()
в where и совсем не позволяет сделать опциональный вложенный селект со
своими параметрами в where. Может кто-нибудь придумает как это сделать?
ps: Нашел SQL::OOP -
https://metacpan.org/pod/distribution/SQL-OOP/README.pod - который почти
идеально мне подходит, кроме того, что много букв. Но интереснее сделать
своё )
pps: Главный кусок кода из гиста:
sub query2 (@) {
my ($in, $p, $o, $l) = @_;
my @aa = map {my $t = $_; map { $_ .' '. $t->{$_}[0] . ' ? '} keys %$t}
@$p;
my @bb = map {my $t = $_; map { $t->{$_}[1] } keys %$t} @$p;
(join' ', $in, map @$_, grep $_->[1],
$p ? [where => join ' AND ', @aa] : (),
$o ? ['ORDER BY' => join ',', @$o] : (),
$l ? [limit => join ',', ('?') x @$l] : ()),
$p ? @bb : (), $l ? @$l : ()
}
say Dumper query2 'SELECT i FROM t', [{'x' => ['=',99]}, {'a' =>
['=',1]},{'b' => ['>',3]}], ['id DESC','a ASC'], [0,1];
}
03.04.2014 8:11, Nick Knutov пишет:
>
> Спасибо, я наконец проникся.
>
> Но при таком конструкторе не учитывается порядок следования полей в
> условии AND, который иногда очень даже следует учитывать в случае с
> MySQL. Пример:
>
> query 'select i from t', {qw(x 99 a 1 b 3)}, ['id desc','a asc'], [0,1];
> даст
> where a = ? and b = ? and x = ?
>
> И, может быть у вас есть готовый код, позволяющий сделать
>
> query 'select i from t', {'a' => ['=',1],'b' => ['>',3]}, ['id desc','a
> asc'], [0,1];
>
> или чего-то такого по условиям, можно с другим входным форматом (надо
> задавать не только =, но и <,>,IS, IS NOT, IN и т.д.)?
>
> У меня получилось:
>
> (join' ', $in, map @$_, grep $_->[1],
> $p ? [where => join ' and ', map $_.' '.$p->{$_}[0].' ?', keys%$p] : (),
> $o ? ['order by' => join ',', @$o] : (),
> $l ? [limit => join ',', ('?') x @$l] : ()),
> $p ? map($_->[1], values $p) : (), $l ? @$l : ()
>
> Но оно стало еще менее читабельным. И если добавить поддержку {'a' =>
> [1], ...}, считая, что если элемент один, то значение - единственный
> элемент, а действие по умолчанию - "=", то код получается еще мене красивым.
>
>
> upd: А если делать массив для соблюдения порядка, то для конструкции
> [
> {'x' => ['=',99]},
> {'a' => ['=',1]},
> {'b' => ['>',3]}
> ]
> слишком многоэтажные конструкции получаются.
>
>
> 18.12.2013 14:07, vividsnow пишет:
>> как вариант что-то вроде:
>> https://gist.github.com/vividsnow/8d294801959975730fc3
>> (возможно более специфичное к вашей задаче)
>>
>> On 12/18/2013 03:44 AM, Nick Knutov wrote:
>>> Не вижу, чем мне это может помочь.
>>>
>>> В данном примере posts_sql_view() возвращает нечто очень большое и
>>> сложное, например селект с джойнами по куче таблиц из селекта с джойнами.
>>>
>>> Как минимум, конструировать запрос мне надо начиная с того места, где
>>> может быть WHERE. И, например, как тут - наличие LIMIT и его количество
>>> зависит от других параметров.
>>>
>>>
>>> 18.12.2013 5:34, vividsnow пишет:
>>>> https://metacpan.org/pod/SQL::Abstract
>>>>
>>>>
>>>> 2013/12/18 Nick Knutov <mail на knutov.com <mailto:mail на knutov.com>>
>>>>
>>>> У меня в коде в некоторых функциях, в зависимости от входных параметров,
>>>> конструируются разные sql запросы. Пример:
>>>>
>>>> my $sql = posts_sql_view();
>>>> $sql .= ' WHERE ' if $type or $id or $main;
>>>> do { $sql .= ' ty.id =? '; push @params, $type;} if $type;
>>>> do { $sql .= ' AND p.id =? LIMIT 1'; push @params, $id
>>>> } if $id;
>>>> $sql .= ' ORDER BY id DESC ' unless $id;
>>>> do { $sql .= ' LIMIT ? ' ; push @params, $limit } if $limit and not $id;
>>>>
>>>> Мне не нравится как это визуально выглядит. Как бы вы это переписали,
>>>> чтобы было понятнее и читабельнее?
>>>>
>>>> ps: в некоторых других случаях можно было бы использовать && вместо
>>>> do{}, но не тут.
>>>>
>>>> pps: ORM предлагать и обсуждать не надо. Как минимум потому, что запросы
>>>> сложные и сложность/стоимость проверки того, что генерирует ORM
>>>> многократно выше любого профита от ORM в данной ситуации.
>>>>
>>>> --
>>>> Best Regards,
>>>> Nick Knutov
>>>> http://knutov.com
>>>> ICQ: 272873706
>>>> Voice: +7-904-84-23-130 <tel:%2B7-904-84-23-130>
>>>> --
>>>> Moscow.pm mailing list
>>>> moscow-pm на pm.org <mailto:moscow-pm на pm.org> | http://moscow.pm.org
>>>>
>>>>
>>>>
>>>>
>>>
>
--
Best Regards,
Nick Knutov
http://knutov.com
ICQ: 272873706
Voice: +7-904-84-23-130
Подробная информация о списке рассылки Moscow-pm