SQL Way
Можете да работите с Nette Database по два начина: като пишете SQL заявки (SQL начин) или като оставите SQL да се генерира автоматично(Explorer начин). SQL начинът ви позволява безопасно да изграждате заявки, като запазвате пълен контрол върху тяхната структура.
Вижте Свързване и конфигуриране за подробности относно настройката на връзката с базата данни.
Основно запитване
Методът query()
изпълнява заявки към базата данни и връща обект ResultSet, представящ резултата.
Ако заявката е неуспешна, методът хвърля
изключение. Можете да прегледате резултата от заявката с помощта на
цикъл foreach
или да използвате някоя от помощните
функции.
$result = $database->query('SELECT * FROM users');
foreach ($result as $row) {
echo $row->id;
echo $row->name;
}
За безопасно вмъкване на стойности в SQL заявките използвайте параметризирани заявки. Nette Database прави това много лесно: просто добавете запетая и стойност към SQL заявката.
$database->query('SELECT * FROM users WHERE name = ?', $name);
За множество параметри можете да преплетете SQL заявката с параметри:
$database->query('SELECT * FROM users WHERE name = ?', $name, 'AND age > ?', $age);
или първо да напишете цялата SQL заявка и след това да добавите всички параметри:
$database->query('SELECT * FROM users WHERE name = ? AND age > ?', $name, $age);
Защита срещу SQL инжектиране
Защо е важно да се използват параметризирани заявки? Защото те ви предпазват от атаки чрез SQL инжектиране, при които нападателите могат да инжектират злонамерени SQL команди, за да манипулират или да получат достъп до данни от базата данни.
Никога не вмъквайте променливи директно в SQL заявка! Винаги използвайте параметризирани заявки, за да се предпазите от SQL инжектиране.
// ❌ БЕЗОПАСЕН КОД - уязвим към SQL инжекция
$database->query("SELECT * FROM users WHERE name = '$name'");
// ✅ Безопасна параметризирана заявка
$database->query('SELECT * FROM users WHERE name = ?', $name);
Не забравяйте да се запознаете с потенциалните рискове за сигурността.
Техники за заявки
Условия WHERE
Можете да запишете условията WHERE
като асоциативен масив, в
който ключовете са имената на колоните, а стойностите – данните, които
трябва да се сравнят. Nette Database автоматично избира най-подходящия SQL
оператор въз основа на типа на стойността.
$database->query('SELECT * FROM users WHERE', [
'name' => 'John',
'active' => true,
]);
// WHERE `име` = 'John' AND `active` = 1
Можете също така изрично да посочите оператора в ключа:
$database->query('SELECT * FROM users WHERE', [
'age >' => 25, // използва оператора >
'name LIKE' => '%John%', // използва оператора LIKE
'email NOT LIKE' => '%example.com%', // използва оператора NOT LIKE
]);
// WHERE `age` > 25 AND `name` LIKE '%John%' AND `email` NOT LIKE '%example.com%'
Специални случаи като null
стойности или масиви се обработват
автоматично:
$database->query('SELECT * FROM products WHERE', [
'name' => 'Laptop', // използва оператора =
'category_id' => [1, 2, 3], // използва IN
'description' => null, // използва IS NULL
]);
// WHERE `name` = 'Laptop' AND `category_id` IN (1, 2, 3) AND `description` IS NULL
За отрицателни условия използвайте оператора NOT
:
$database->query('SELECT * FROM products WHERE', [
'name NOT' => 'Laptop', // използва оператора <>
'category_id NOT' => [1, 2, 3], // използва NOT IN
'description NOT' => null, // използва IS NOT NULL
'id' => [], // пропуснати
]);
// WHERE `name` <> 'Laptop' AND `category_id` NOT IN (1, 2, 3) AND `description` IS NOT NULL
По подразбиране условията се комбинират, като се използва операторът
AND
. Можете да промените това поведение, като използвате заместителя ?or.
ORDER BY Правила
Клаузата ORDER BY
може да бъде дефинирана като масив, в който
ключовете представляват колони, а стойностите са булеви стойности,
указващи възходящ ред:
$database->query('SELECT id FROM author ORDER BY', [
'id' => true, // възходящ
'name' => false, // низходящ
]);
// SELECT id FROM author ORDER BY `id`, `name` DESC
Вмъкване на данни (INSERT)
За да вмъкнете записи, използвайте оператора SQL INSERT
.
$values = [
'name' => 'John Doe',
'email' => 'john@example.com',
];
$database->query('INSERT INTO users ?', $values);
$userId = $database->getInsertId();
Методът getInsertId()
връща идентификатора на последния вмъкнат
ред. За някои бази данни (например PostgreSQL) трябва да посочите името на
последователността, като използвате $database->getInsertId($sequenceId)
.
Като параметри можете да подавате и специални стойности, като например файлове, DateTime обекти или enum типове.
Вмъкване на няколко записа наведнъж:
$database->query('INSERT INTO users ?', [
['name' => 'User 1', 'email' => 'user1@mail.com'],
['name' => 'User 2', 'email' => 'user2@mail.com'],
]);
Извършването на групово INSERT е много по-бързо, тъй като се изпълнява само една заявка към базата данни, а не няколко отделни заявки.
Забележка за сигурност: Никога не използвайте непотвърдени данни
като $values
. Запознайте се с възможните рискове.
Актуализиране на данни (UPDATE)
За актуализиране на записите използвайте изявлението SQL
UPDATE
.
// Актуализиране на единичен запис
$values = [
'name' => 'John Smith',
];
$result = $database->query('UPDATE users SET ? WHERE id = ?', $values, 1);
Можете да проверите броя на засегнатите редове, като използвате
$result->getRowCount()
.
Можете да използвате операторите +=
и -=
в UPDATE
:
$database->query('UPDATE users SET ? WHERE id = ?', [
'login_count+=' => 1, // увеличаване на login_count
], 1);
За да вмъкнете или актуализирате запис, ако той вече съществува,
използвайте техниката ON DUPLICATE KEY UPDATE
:
$values = [
'name' => $name,
'year' => $year,
];
$database->query('INSERT INTO users ? ON DUPLICATE KEY UPDATE ?',
$values + ['id' => $id],
$values,
);
// INSERT INTO users (`id`, `name`, `year`) VALUES (123, 'Jim', 1978)
// ON DUPLICATE KEY UPDATE `име` = 'Jim', `година` = 1978
Обърнете внимание, че Nette Database разпознава контекста на SQL командата, в
която се използва параметър с масив, и генерира съответния SQL код.
Например, тя конструира (id, name, year) VALUES (123, 'Jim', 1978)
от първия масив,
докато втория преобразува в name = 'Jim', year = 1978
. Това е разгледано
по-подробно в раздела Съвети за конструиране
на SQL.
Изтриване на данни (DELETE)
За да изтриете записи, използвайте SQL оператора DELETE
. Пример с
броя на изтритите редове:
$count = $database->query('DELETE FROM users WHERE id = ?', 1)
->getRowCount();
Съвети за изграждане на SQL
Заместващите символи в SQL ви позволяват да контролирате начина, по който стойностите на параметрите се включват в SQL изразите:
Загатване | Описание | Автоматично се използва за |
---|---|---|
?name |
Използва се за имена на таблици или колони | – |
?values |
Генерира (key, ...) VALUES (value, ...) |
INSERT ... ? , REPLACE ... ? |
?set |
Генерира задания key = value, ... |
SET ? , KEY UPDATE ? |
?and |
Обединява условия в масив с AND |
WHERE ? , HAVING ? |
?or |
Обединява условия в масив с OR |
– |
?order |
Генерира клаузата ORDER BY |
ORDER BY ? , GROUP BY ? |
За динамично вмъкване на имена на таблици или колони използвайте
заместителя ?name
. Базата данни Nette осигурява правилно ескапиране
според конвенциите на базата данни (напр. заграждане в задни тирета за
MySQL).
$table = 'users';
$column = 'name';
$database->query('SELECT ?name FROM ?name WHERE id = 1', $column, $table);
// SELECT `name` FROM `users` WHERE id = 1 (в MySQL)
Предупреждение: Използвайте заместителя ?name
само за
валидирани имена на таблици и колони. В противен случай рискувате уязвимости в сигурността.
Обикновено не е необходимо да се посочват други подсказки, тъй като
Nette използва интелигентно автоматично откриване при конструирането на
SQL заявки (вж. третата колона на таблицата). Въпреки това можете да ги
използвате в ситуации, в които искате да комбинирате условия, като
използвате OR
вместо AND
:
$database->query('SELECT * FROM users WHERE ?or', [
'name' => 'John',
'email' => 'john@example.com',
]);
// SELECT * FROM users WHERE `name` = 'John' OR `email` = 'john@example.com'
Специални стойности
В допълнение към стандартните скаларни типове (например string
,
int
, bool
) можете да предавате и специални стойности като
параметри:
- Файлове: Използвайте
fopen('file.png', 'r')
, за да вмъкнете двоичното съдържание на файл. - Дата и час: Обектите на
DateTime
се преобразуват автоматично във формата за дата на базата данни. - Стойности на енум: Инстанциите на
enum
се преобразуват в съответните им стойности. - SQL литерали: Създадени с помощта на
Connection::literal('NOW()')
, те се вмъкват директно в заявката.
$database->query('INSERT INTO articles ?', [
'title' => 'My Article',
'published_at' => new DateTime,
'content' => fopen('image.png', 'r'),
'state' => Status::Draft,
]);
За бази данни, които нямат собствена поддръжка на типа datetime
(например SQLite и Oracle), стойностите на DateTime
се преобразуват в
съответствие с опцията на конфигурацията formatDateTime
(по
подразбиране: U
за Unix timestamp).
SQL литерали
В някои случаи може да се наложи да вмъкнете суров SQL код като
стойност, без да го третирате като низ или да го ескапирате. За тази цел
използвайте обекти от клас Nette\Database\SqlLiteral
, които могат да бъдат
създадени чрез метода Connection::literal()
.
$result = $database->query('SELECT * FROM users WHERE', [
'name' => $name,
'year >' => $database::literal('YEAR()'),
]);
// SELECT * FROM users WHERE (`name` = 'Jim') AND (`year` > YEAR())
Алтернативно:
$result = $database->query('SELECT * FROM users WHERE', [
'name' => $name,
$database::literal('year > YEAR()'),
]);
// SELECT * FROM users WHERE (`name` = 'Jim') AND (year > YEAR())
SQL литералите могат да съдържат и параметри:
$result = $database->query('SELECT * FROM users WHERE', [
'name' => $name,
$database::literal('year > ? AND year < ?', $min, $max),
]);
// SELECT * FROM users WHERE `name` = 'Jim' AND (year > 1978 AND year < 2017)
Това дава възможност за гъвкави комбинации:
$result = $database->query('SELECT * FROM users WHERE', [
'name' => $name,
$database::literal('?or', [
'active' => true,
'role' => $role,
]),
]);
// SELECT * FROM users WHERE `name` = 'Jim' AND (`active` = 1 OR `role` = 'admin')
Извличане на данни
Съкращения за SELECT заявки
За да се опрости извличането на данни, класът Connection
предоставя
няколко преки пътища, които комбинират извикването на query()
с
последващо извикване на fetch*()
. Тези методи приемат същите
параметри като query()
, т.е. SQL заявка и незадължителни параметри.
Подробно описание на методите на fetch*()
можете да намерите по-долу.
fetch($sql, ...$params): ?Row |
Изпълнява заявката и извлича първия ред като обект Row . |
fetchAll($sql, ...$params): array |
Изпълнява заявката и извлича всички редове като масив от обекти
Row . |
fetchPairs($sql, ...$params): array |
Изпълнява заявката и извлича асоциативен масив, в който първата колона е ключът, а втората е стойността. |
fetchField($sql, ...$params): mixed |
Изпълнява заявката и извлича стойността на първата клетка в първия ред. |
fetchList($sql, ...$params): ?array |
Изпълнява заявката и извлича първия ред като индексиран масив. |
Пример:
// fetchField() - връща стойността на първата клетка
$count = $database->query('SELECT COUNT(*) FROM articles')
->fetchField();
foreach
– Итерация над редове
След изпълнение на заявка се връща обект ResultSet, който ви позволява да
итерирате над резултатите по различни начини. Най-простият и ефективен
от гледна точка на паметта начин за извличане на редове е чрез итерация
в цикъл foreach
. Този метод обработва редовете един по един и
избягва съхраняването на всички данни в паметта наведнъж.
$result = $database->query('SELECT * FROM users');
foreach ($result as $row) {
echo $row->id;
echo $row->name;
//...
}
Цикълът ResultSet
може да се итерира само веднъж. Ако трябва
да го повторите многократно, трябва първо да заредите данните в масив,
например чрез метода fetchAll()
.
fetch(): ?Row
Изпълнява заявката и извлича един ред като обект Row
. Ако не са
налични повече редове, връща null
. Този метод придвижва вътрешния
указател към следващия ред.
$result = $database->query('SELECT * FROM users');
$row = $result->fetch(); // извлича първия ред
if ($row) {
echo $row->name;
}
fetchAll(): array
Извлича всички останали редове от ResultSet
като масив от обекти
Row
.
$result = $database->query('SELECT * FROM users');
$rows = $result->fetchAll(); // извлича всички редове
foreach ($rows as $row) {
echo $row->name;
}
fetchPairs (string|int|null $key = null, string|int|null $value = null): array
Извлича резултатите като асоциативен масив. Първият аргумент указва колоната, която да се използва като ключ, а вторият – колоната, която да се използва като стойност:
$result = $database->query('SELECT id, name FROM users');
$names = $result->fetchPairs('id', 'name');
// [1 => "John Doe", 2 => "Jane Doe", ...]
Ако е посочен само първият параметър, стойността ще бъде целият ред
(като обект Row
):
$rows = $result->fetchPairs('id');
// [1 => Ред(id: 1, име: "John"), 2 => Ред(id: 2, име: "Jane"), ...]
Ако като ключ е подаден null
, масивът ще бъде индексиран цифрово,
започвайки от нула:
$names = $result->fetchPairs(null, 'name');
// [0 => "John Doe", 1 => "Jane Doe", ...]
fetchPairs (Closure $callback): array
Като алтернатива можете да предоставите обратно извикване, което определя двойките ключ-стойност или стойностите за всеки ред.
$result = $database->query('SELECT * FROM users');
$items = $result->fetchPairs(fn($row) => "$row->id - $row->name");
// ["1 - Джон", "2 - Джейн", ...]
// Обратното извикване може също да върне масив с двойка ключ и стойност:
$names = $result->fetchPairs(fn($row) => [$row->name, $row->age]);
// ["Джон" => 46, "Джейн" => 21, ...]
fetchField(): mixed
Извлича стойността на първата клетка в текущия ред. Ако няма повече
редове, се връща null
. Този метод придвижва вътрешния указател към
следващия ред.
$result = $database->query('SELECT name FROM users');
$name = $result->fetchField(); // извлича името от първия ред
fetchList(): ?array
Извлича реда като индексиран масив. Ако няма повече редове, се връща
null
. Този метод придвижва вътрешния указател към
следващия ред.
$result = $database->query('SELECT name, email FROM users');
$row = $result->fetchList(); // ["John", "john@example.com"]
getRowCount(): ?int
Връща броя на редовете, засегнати от последната заявка UPDATE
или
DELETE
. За заявките SELECT
връща броя на изтеглените редове, но
това не винаги е известно – в такива случаи връща null
.
getColumnCount(): ?int
Връща броя на колоните в ResultSet
.
Информация за заявката
За извличане на подробна информация за последно изпълнената заявка използвайте:
echo $database->getLastQueryString(); // извежда SQL заявката
$result = $database->query('SELECT * FROM articles');
echo $result->getQueryString(); // извежда SQL заявката
echo $result->getTime(); // извежда времето за изпълнение в секунди
За да покажете резултата като HTML таблица, използвайте:
$result = $database->query('SELECT * FROM articles');
$result->dump();
Можете също така да извлечете информация за типовете колони от
ResultSet
:
$result = $database->query('SELECT * FROM articles');
$types = $result->getColumnTypes();
foreach ($types as $column => $type) {
echo "$column is of type $type->type"; // напр., 'id е от тип int'
}
Регистриране на заявки
Можете да реализирате потребителско регистриране на заявки.
Събитието onQuery
представлява масив от обратни извиквания, които
се задействат след всяко изпълнение на заявката:
$database->onQuery[] = function ($database, $result) use ($logger) {
$logger->info('Query: ' . $result->getQueryString());
$logger->info('Time: ' . $result->getTime());
if ($result->getRowCount() > 1000) {
$logger->warning('Large result set: ' . $result->getRowCount() . ' rows');
}
};