Índice
Zend_Db_Select é uma ferramenta para construir instruções SQL SELECT independentemente do banco de dados utilizado. Obviamente isso não pode ser perfeito, mas é um avanço para ajudar a fazer consultas portáveis entre banco de dados diferentes. Adicionalmente, ele ajuda a proteger suas consultas contra ataques de SQL injection.
A maneira mais fácil de criar uma instância de Zend_Db_Select é usar o método Zend_Db_Adapter::select().
<?php require_once 'Zend/Db.php'; $params = array ( 'host' => '127.0.0.1', 'username' => 'malory', 'password' => '******', 'dbname' => 'camelot' ); $db = Zend_Db::factory('pdoMysql', $params); $select = $db->select(); // $select é agora um objeto Zend_Db_Select_PdoMysql ?>
Então você constrói uma consulta SELECT usando o objeto e seus métodos, e em seguida gera uma string para devolver ao Zend_Db_Adapter para consultas ou buscas.
<?php // // SELECT * // FROM round_table // WHERE noble_title = "Sir" // ORDER BY first_name // LIMIT 10 OFFSET 20 // // você pode usar um estilo iterativo... $select->from('round_table', '*'); $select->where('noble_title = ?', 'Sir'); $select->order('first_name'); $select->limit(10,20); // ...ou um estilo "fluente": $select->from('round_table', '*') ->where('noble_title = ?', 'Sir') ->order('first_name') ->limit(10,20); // de um modo ou de outro, busque os resultados $sql = $select->__toString(); $result = $db->fetchAll($sql); // alternativamente, você pode passar o objeto $select; // Zend_Db_Adapter é esperto o suficiente para chamar __toString() nos // objetos Zend_Db_Select para pegar a string de consulta. $result = $db->fetchAll($select); ?>
Você também pode usar parâmetros vinculados nas suas consultas em vez de adicionar aspas a cada método.
<?php // // SELECT * // FROM round_table // WHERE noble_title = "Sir" // ORDER BY first_name // LIMIT 10 OFFSET 20 // $select->from('round_table', '*') ->where('noble_title = :title') ->order('first_name') ->limit(10,20); // de um modo ou de outro, busque os resultados usando os parãmetros vinculados $params = array('title' => 'Sir'); $result = $db->fetchAll($select, $params); ?>
Para selecionar colunas de uma tabela específica, use o método from(), specificando a tabela e as colunas que você quer. Você pode usar apelidos para a tabela e para colunas, e você pode usar from() quantas vezes precisar.
<?php // crie um objeto $db, usando Mysql como o adaptador. $select = $db->select(); // SELECT a, b, c FROM some_table $select->from('some_table', 'a, b, c'); // equivalente: $select->from('some_table', array('a', 'b', 'c'); // SELECT bar.col FROM foo AS bar $select->from('foo AS bar', 'bar.col'); // SELECT foo.col AS col1, bar.col AS col2 FROM foo, bar $select->from('foo', 'foo.col AS col1'); $select->from('bar', 'bar.col AS col2'); ?>
Para selecionar colunas usando tabelas unidas, use o método join(). Primeiro passe o nome da tabela que se unirá, então a condição da união, e finalmente as colunas que você quer unir. Você pode usar join() quantas vezes precisar.
<?php // crie um objeto $db, usando Mysql como adaptador. $select = $db->select(); // // SELECT foo.*, bar.* // FROM foo // JOIN bar ON foo.id = bar.id // $select->from('foo', '*'); $select->join('bar', 'foo.id = bar.id', '*'); ?>
No momento, apenas a sintaxe de JOIN é suportada; não LEFT JOINs, RIGHT JOINs, etc. Versões futuras vão suportar estas operações de um modo independente de bancos de dados.
Para adicionar condições WHERE, use o método where(). Você pode passar uma string regular, ou uma string com um ponto de interrogação marcando um espaço reservado e um valor para adicionar aspas nela (o valor receberá aspas usando Zend_Db_Adapter::quoteInto).
Chamadas múltiplas a where() adicionarão AND às condições; se você precisar de OR, use orWhere().
<?php // crie um objeto $db, e pega a ferramenta SELECT. $select = $db->select(); // // SELECT * // FROM round_table // WHERE noble_title = "Sir" // AND favorite_color = "yellow" // $select->from('round_table', '*'); $select->where('noble_title = "Sir"'); // embedded value $select->where('favorite_color = ?', 'yellow'); // quoted value // // SELECT * // FROM foo // WHERE bar = "baz" // OR id IN("1", "2", "3") // $select->from('foo', '*'); $select->where('bar = ?', 'baz'); $select->orWhere('id IN(?)', array(1, 2, 3); ?>
Para agrupar linhas, use o método group() quantas vezes você desejar.
<?php // crie um objeto $db, e pega a ferramenta SELECT. $select = $db->select(); // // SELECT COUNT(id) // FROM foo // GROUP BY bar, baz // $select->from('foo', 'COUNT(id)'); $select->group('bar'); $select->group('baz'); // chamada equivalente de group(): $select->group('bar, baz'); // outra chamada equivalente de group(): $select->group(array('bar', 'baz')); ?>
Para adicionar condições HAVING aos resultados selecionados, use o método having(). este método é id~entico em função ao método where().
Se você chamar having() múltiplas vezes, as condições são unidas usando AND; se você quiser uma condição OR, use orHaving().
<?php // crie um objeto $db, e pega a ferramenta SELECT. $select = $db->select(); // // SELECT COUNT(id) AS count_id // FROM foo // GROUP BY bar, baz // HAVING count_id > "1" // $select->from('foo', 'COUNT(id) AS count_id'); $select->group('bar, baz'); $select->having('count_id > ?', 1); ?>
Para ordenar colunas com ORDER, use o método order() quantas vezes desejar.
<?php // crie um objeto $db, e pega a ferramenta SELECT. $select = $db->select(); // // SELECT * FROM round_table // ORDER BY noble_title DESC, first_name ASC // $select->from('round_table', '*'); $select->order('noble_title DESC'); $select->order('first_name'); // chamada equivalente de order(): $select->order('noble_title DESC, first_name'); // outra chamada equivalente de order(): $select->order(array('noble_title DESC', 'first_name')); ?>
Zend_Db_Select oferece abstrações pra suporte da cláusula LIMIT. Para muitos bancos de dados, como MySQL e PostgreSQL, isso é relativamente fácil, pois eles suportam a sintaxe "LIMIT :count [OFFSET :offset]".
Para outros bancos de dados, como Microsoft SQL e Oracle, isso não é tão fácil, pois eles não suportam cláusulas LIMIT. MS-SQL tem apenas uma cláusula TOP, e Oracle requer que a consulta seja escrita de uma maneira especial para emular um LIMIT. Devido à maneira como o Zend_Db_Select trabalhar internamente, nós podemos reescrever SELECT para emular a funcionalidade LIMIT dos bancos de dados de código aberto mencionados anteriormente.
Para usar LIMIT nos resultados retornados usando count e offset, use o método limit() com um valor count e um offset opcional.
<?php // primeiro, um simples "LIMIT :count" $select = $db->select(); $select->from('foo', '*'); $select->order('id'); $select->limit(10); // // Em MySQL/PostgreSQL/SQLite, isso se traduz para: // // SELECT * FROM foo // ORDER BY id ASC // LIMIT 10 // // Mas em Microsoft SQL, isso se traduz para: // // SELECT TOP 10 * FROM FOO // ORDER BY id ASC // // // agora, um mais complexo "LIMIT :count OFFSET :offset" $select = $db->select(); $select->from('foo', '*'); $select->order('id'); $select->limit(10, 20); // // Em MySQL/PostgreSQL/SQLite, isso se traduz para: // // SELECT * FROM foo // ORDER BY id ASC // LIMIT 10 OFFSET 20 // // Mas em Microsoft SQL, que não tem suporte a offset, isso se traduz para // algo como: // // SELECT * FROM ( // SELECT TOP 10 * FROM ( // SELECT TOP 30 * FROM foo ORDER BY id DESC // ) ORDER BY id ASC // ) // // Zend_Db_Adapter faz a tradução da consulta automaticamente. // ?>
Zend_Db_Select também oferece limites baseados em páginas. Se você quer receber uma certa "página" de resultados, use o método limitPage(); primeiro passe o número da página que voc~e quer, e então o número de linhas que aparecerão em cada página.
<?php // construa o select básico... $select = $db->select(); $select->from('foo', '*'); $select->order('id'); // ... e limite o resultado à página 3, cada página com 10 linhas $select->limitPage(3, 10); // // Em MySQL/PostgreSQL/SQLite, isso se traduz para: // // SELECT * FROM foo // ORDER BY id ASC // LIMIT 10 OFFSET 20 // ?>