Nette Documentation Preview

syntax
Calea SQL
*********

.[perex]
Puteți lucra cu Nette Database în două moduri: scriind interogări SQL (metoda SQL) sau lăsând SQL să fie generat automat[(metoda Explorer |explorer]). Modul SQL vă permite să construiți în siguranță interogări, păstrând în același timp controlul deplin asupra structurii acestora.

.[note]
Consultați [Conectare și configurare |guide#Connection and Configuration] pentru detalii despre configurarea conexiunii la baza de date.


Interogare de bază .[#toc-basic-querying]
=========================================

Metoda `query()` execută interogări ale bazei de date și returnează un obiect [ResultSet | api:Nette\Database\ResultSet] care reprezintă rezultatul. Dacă interogarea eșuează, metoda aruncă [o excepție | exceptions].
Puteți parcurge în buclă rezultatul interogării folosind o buclă `foreach` sau folosind una dintre [funcțiile de ajutor | #Fetching Data].

```php
$result = $database->query('SELECT * FROM users');

foreach ($result as $row) {
	echo $row->id;
	echo $row->name;
}
```

Pentru a introduce valori în siguranță în interogările SQL, utilizați interogări parametrizate. Nette Database face acest lucru foarte simplu: trebuie doar să adăugați o virgulă și valoarea la interogarea SQL.

```php
$database->query('SELECT * FROM users WHERE name = ?', $name);
```

Pentru parametrii multipli, puteți fie să intercalați interogarea SQL cu parametrii:

```php
$database->query('SELECT * FROM users WHERE name = ?', $name, 'AND age > ?', $age);
```

Sau scrieți mai întâi întreaga interogare SQL și apoi adăugați toți parametrii:

```php
$database->query('SELECT * FROM users WHERE name = ? AND age > ?', $name, $age);
```


Protecție împotriva injectării SQL .[#toc-protection-against-sql-injection]
===========================================================================

De ce este important să utilizați interogări parametrizate? Pentru că vă protejează de atacurile de tip SQL injection, în care atacatorii pot injecta comenzi SQL rău intenționate pentru a manipula sau a accesa datele din baza de date.

.[warning]
**Niciodată nu introduceți variabile direct într-o interogare SQL!** Utilizați întotdeauna interogări parametrizate pentru a vă proteja împotriva injecțiilor SQL.

```php
// ❌ UNSAFE CODE - vulnerabil la injectarea SQL
$database->query("SELECT * FROM users WHERE name = '$name'");

// ✅ Interogare parametrizată sigură
$database->query('SELECT * FROM users WHERE name = ?', $name);
```

Asigurați-vă că vă familiarizați cu [potențialele riscuri de securitate | security].


Tehnici de interogare .[#toc-query-techniques]
==============================================


Condiții WHERE .[#toc-where-conditions]
---------------------------------------

Puteți scrie condițiile `WHERE` ca un array asociativ, în care cheile sunt numele coloanelor, iar valorile sunt datele de comparat. Nette Database selectează automat cel mai adecvat operator SQL în funcție de tipul valorii.

```php
$database->query('SELECT * FROM users WHERE', [
	'name' => 'John',
	'active' => true,
]);
// WHERE `name` = 'John' AND `active` = 1
```

De asemenea, puteți specifica explicit operatorul în cheie:

```php
$database->query('SELECT * FROM users WHERE', [
	'age >' => 25,           // utilizează operatorul >
	'name LIKE' => '%John%', // utilizează operatorul LIKE
	'email NOT LIKE' => '%example.com%', // utilizează operatorul NOT LIKE
]);
// WHERE `age` > 25 AND `name` LIKE '%John%' AND `email` NOT LIKE '%example.com%'
```

Cazurile speciale, cum ar fi valorile `null` sau matricele, sunt gestionate automat:

```php
$database->query('SELECT * FROM products WHERE', [
	'name' => 'Laptop',         // utilizează operatorul =
	'category_id' => [1, 2, 3], // utilizează IN
	'description' => null,      // uses IS NULL
]);
// WHERE `name` = 'Laptop' AND `category_id` IN (1, 2, 3) AND `description` IS NULL
```

Pentru condiții negative, utilizați operatorul `NOT`:

```php
$database->query('SELECT * FROM products WHERE', [
	'name NOT' => 'Laptop',         // utilizează operatorul <>
	'category_id NOT' => [1, 2, 3], // utilizează NOT IN
	'description NOT' => null,      // uses IS NOT NULL
	'id' => [],                     // sărit
]);
// WHERE `name` <> 'Laptop' AND `category_id` NOT IN (1, 2, 3) AND `description` IS NOT NULL
```

În mod implicit, condițiile sunt combinate utilizând operatorul `AND`. Puteți schimba acest comportament utilizând marcajul [?or | #SQL Construction Hints].


ORDER BY Reguli .[#toc-order-by-rules]
--------------------------------------

Clauza `ORDER BY` poate fi definită ca un array, unde cheile reprezintă coloane, iar valorile sunt booleeni care indică ordinea crescătoare:

```php
$database->query('SELECT id FROM author ORDER BY', [
	'id' => true,  // ascendentă
	'name' => false, // descrescător
]);
// SELECT id FROM author ORDER BY `id`, `name` DESC
```


Inserarea datelor (INSERT) .[#toc-inserting-data-insert]
--------------------------------------------------------

Pentru a introduce înregistrări, utilizați instrucțiunea SQL `INSERT`.

```php
$values = [
	'name' => 'John Doe',
	'email' => 'john@example.com',
];
$database->query('INSERT INTO users ?', $values);
$userId = $database->getInsertId();
```

Metoda `getInsertId()` returnează ID-ul ultimului rând introdus. Pentru anumite baze de date (de exemplu, PostgreSQL), trebuie să specificați numele secvenței utilizând `$database->getInsertId($sequenceId)`.

De asemenea, puteți trece [valori speciale |#special values], cum ar fi fișiere, obiecte DateTime sau tipuri enum, ca parametri.

Inserarea simultană a mai multor înregistrări:

```php
$database->query('INSERT INTO users ?', [
	['name' => 'User 1', 'email' => 'user1@mail.com'],
	['name' => 'User 2', 'email' => 'user2@mail.com'],
]);
```

Efectuarea unei INSERTĂRI pe loturi este mult mai rapidă deoarece se execută o singură interogare a bazei de date în loc de mai multe interogări individuale.

**Nota de securitate:** Nu utilizați niciodată date nevalidate ca `$values`. Familiarizați-vă cu [riscurile posibile | security#Array Keys are Not Secure API].


Actualizarea datelor (UPDATE) .[#toc-updating-data-update]
----------------------------------------------------------

Pentru a actualiza înregistrările, utilizați instrucțiunea SQL `UPDATE`.

```php
// Actualizarea unei singure înregistrări
$values = [
	'name' => 'John Smith',
];
$result = $database->query('UPDATE users SET ? WHERE id = ?', $values, 1);
```

Puteți verifica numărul de rânduri afectate utilizând `$result->getRowCount()`.

Puteți utiliza operatorii `+=` și `-=` în `UPDATE`:

```php
$database->query('UPDATE users SET ? WHERE id = ?', [
	'login_count+=' => 1, // increment login_count
], 1);
```

Pentru a introduce sau actualiza o înregistrare dacă aceasta există deja, utilizați tehnica `ON DUPLICATE KEY UPDATE`:

```php
$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 `name` = 'Jim', `year` = 1978
```

Rețineți că Nette Database recunoaște contextul comenzii SQL în care este utilizat un parametru cu un array și generează codul SQL în consecință. De exemplu, a construit `(id, name, year) VALUES (123, 'Jim', 1978)` din primul array, în timp ce l-a convertit pe al doilea în `name = 'Jim', year = 1978`. Acest aspect este tratat mai detaliat în secțiunea [Sfaturi pentru construirea SQL |#SQL Construction Hints].


Ștergerea datelor (DELETE) .[#toc-deleting-data-delete]
-------------------------------------------------------

Pentru a șterge înregistrări, utilizați instrucțiunea SQL `DELETE`. Exemplu cu numărul de rânduri șterse:

```php
$count = $database->query('DELETE FROM users WHERE id = ?', 1)
	->getRowCount();
```


Sugestii de construcție SQL .[#toc-sql-construction-hints]
----------------------------------------------------------

Lockerele SQL vă permit să controlați modul în care valorile parametrilor sunt încorporate în expresiile SQL:

| Indicație | Descriere | Folosit automat pentru
|------------|-------------------------------------------------|-----------------------------
| `?name` | Folosit pentru nume de tabele sau coloane | -
| `?values` | Generează `(key, ...) VALUES (value, ...)` | `INSERT ... ?`, `REPLACE ... ?`
| `?set` | Generează atribuiri `key = value, ...` | `SET ?`, `KEY UPDATE ?`
| `?and` | Unește condițiile dintr-un array cu `AND` | `WHERE ?`, `HAVING ?`
| `?or` | Unește condițiile într-un array cu `OR` | -
| `?order` | Generează clauza `ORDER BY` | `ORDER BY ?`, `GROUP BY ?`

Pentru inserarea dinamică a numelor de tabele sau coloane, utilizați marcajul `?name`. Nette Database asigură scăparea corectă în conformitate cu convențiile bazei de date (de exemplu, includerea în backticks pentru MySQL).

```php
$table = 'users';
$column = 'name';
$database->query('SELECT ?name FROM ?name WHERE id = 1', $column, $table);
// SELECT `name` FROM `users` WHERE id = 1 (în MySQL)
```

**Avertisment:** Folosiți marcajul `?name` numai pentru numele validate ale tabelelor și coloanelor. În caz contrar, riscați [vulnerabilități de securitate | security#Dynamic Identifiers].

De obicei, nu este necesar să specificați alte indicii, deoarece Nette utilizează detectarea automată inteligentă atunci când construiește interogări SQL (a se vedea a treia coloană din tabel). Cu toate acestea, le puteți utiliza în situațiile în care doriți să combinați condițiile utilizând `OR` în loc de `AND`:

```php
$database->query('SELECT * FROM users WHERE ?or', [
	'name' => 'John',
	'email' => 'john@example.com',
]);
// SELECT * FROM users WHERE `name` = 'John' OR `email` = 'john@example.com'
```


Valori speciale .[#toc-special-values]
--------------------------------------

În plus față de tipurile scalare standard (de exemplu, `string`, `int`, `bool`), puteți trece și valori speciale ca parametri:

- Fișiere: Utilizați `fopen('file.png', 'r')` pentru a introduce conținutul binar al unui fișier.
- Data și ora: Obiectele `DateTime` sunt convertite automat în formatul de dată al bazei de date.
- Valori enum: Instanțele din `enum` sunt convertite în valorile corespunzătoare.
- Literale SQL: Create utilizând `Connection::literal('NOW()')`, acestea sunt inserate direct în interogare.

```php
$database->query('INSERT INTO articles ?', [
	'title' => 'My Article',
	'published_at' => new DateTime,
	'content' => fopen('image.png', 'r'),
	'state' => Status::Draft,
]);
```

Pentru bazele de date care nu oferă suport nativ pentru tipul `datetime` (de exemplu, SQLite și Oracle), valorile `DateTime` sunt convertite în conformitate cu opțiunea de configurare `formatDateTime` (implicit: `U` pentru timestamp Unix).


Literali SQL .[#toc-sql-literals]
---------------------------------

În unele cazuri, poate fi necesar să introduceți cod SQL brut ca valoare fără a-l trata ca pe un șir de caractere sau fără a-l scăpa. Pentru aceasta, utilizați obiecte din clasa `Nette\Database\SqlLiteral`, care pot fi create utilizând metoda `Connection::literal()`.

```php
$result = $database->query('SELECT * FROM users WHERE', [
	'name' => $name,
	'year >' => $database::literal('YEAR()'),
]);
// SELECT * FROM users WHERE (`name` = 'Jim') AND (`year` > YEAR())
```

Alternativ:

```php
$result = $database->query('SELECT * FROM users WHERE', [
	'name' => $name,
	$database::literal('year > YEAR()'),
]);
// SELECT * FROM users WHERE (`name` = 'Jim') AND (year > YEAR())
```

Literalele SQL pot conține și parametri:

```php
$result = $database->query('SELECT * FROM users WHERE', [
	'name' => $name,
	$database::literal('year > ? AND year &lt; ?', $min, $max),
]);
// SELECT * FROM users WHERE `name` = 'Jim' AND (an > 1978 AND an < 2017)
```

Acest lucru permite combinații flexibile:

```php
$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')
```


Preluarea datelor .[#toc-fetching-data]
=======================================


Scurtături pentru interogările SELECT .[#toc-shortcuts-for-select-queries]
--------------------------------------------------------------------------

Pentru a simplifica recuperarea datelor, clasa `Connection` oferă mai multe scurtături care combină un apel la `query()` cu un apel ulterior la `fetch*()`. Aceste metode acceptă aceiași parametri ca și `query()`, și anume o interogare SQL și parametri opționali.
O descriere detaliată a metodelor `fetch*()` poate fi găsită [mai jos |#fetch()].

| `fetch($sql, ...$params): ?Row` | Execută interogarea și extrage primul rând ca obiect `Row`.
| `fetchAll($sql, ...$params): array` | Execută interogarea și extrage toate rândurile sub forma unei serii de obiecte `Row`.
| `fetchPairs($sql, ...$params): array` | Execută interogarea și extrage un array asociativ în care prima coloană este cheia, iar a doua este valoarea.
| `fetchField($sql, ...$params): mixed` | Execută interogarea și extrage valoarea primei celule din primul rând.
| `fetchList($sql, ...$params): ?array` | Execută interogarea și extrage primul rând ca un array indexat.

Exemplu:

```php
// fetchField() - returnează valoarea primei celule
$count = $database->query('SELECT COUNT(*) FROM articles')
	->fetchField();
```


`foreach` - Iterarea peste rânduri
----------------------------------

După executarea unei interogări, este returnat un obiect [ResultSet |api:Nette\Database\ResultSet], care vă permite să iterați peste rezultate în diverse moduri. Cel mai simplu și mai eficient mod de a prelua rânduri din memorie este prin iterarea într-o buclă `foreach`. Această metodă procesează rândurile unul câte unul și evită stocarea simultană a tuturor datelor în memorie.

```php
$result = $database->query('SELECT * FROM users');

foreach ($result as $row) {
	echo $row->id;
	echo $row->name;
	//...
}
```

.[note]
Rețeaua `ResultSet` poate fi iterată o singură dată. Dacă trebuie să o iterați de mai multe ori, trebuie mai întâi să încărcați datele într-un array, de exemplu, utilizând metoda `fetchAll()`.


fetch(): ?Row .[method]
-----------------------

Execută interogarea și extrage un singur rând ca obiect `Row`. Dacă nu mai sunt disponibile alte rânduri, se returnează `null`. Această metodă avansează pointerul intern la rândul următor.

```php
$result = $database->query('SELECT * FROM users');
$row = $result->fetch(); // preia primul rând
if ($row) {
	echo $row->name;
}
```


fetchAll(): array .[method]
---------------------------

Preia toate rândurile rămase din `ResultSet` sub forma unei serii de obiecte `Row`.

```php
$result = $database->query('SELECT * FROM users');
$rows = $result->fetchAll(); // preia toate rândurile
foreach ($rows as $row) {
	echo $row->name;
}
```


fetchPairs(string|int|null $key = null, string|int|null $value = null): array .[method]
---------------------------------------------------------------------------------------

Preia rezultatele sub forma unui array asociativ. Primul argument specifică coloana care urmează să fie utilizată ca cheie, iar al doilea specifică coloana care urmează să fie utilizată ca valoare:

```php
$result = $database->query('SELECT id, name FROM users');
$names = $result->fetchPairs('id', 'name');
// [1 => "John Doe", 2 => "Jane Doe", ...]
```

Dacă este furnizat doar primul parametru, valoarea va fi întregul rând (ca obiect `Row` ):

```php
$rows = $result->fetchPairs('id');
// [1 => rând (id: 1, nume: "John"), 2 => rând (id: 2, nume: "Jane"), ...]
```

Dacă `null` este trecut ca cheie, matricea va fi indexată numeric începând de la zero:

```php
$names = $result->fetchPairs(null, 'name');
// [0 => "John Doe", 1 => "Jane Doe", ...]
```


fetchPairs(Closure $callback): array .[method]
----------------------------------------------

Alternativ, puteți furniza un callback care determină perechile cheie-valoare sau valorile pentru fiecare rând.

```php
$result = $database->query('SELECT * FROM users');
$items = $result->fetchPairs(fn($row) => "$row->id - $row->name");
// ['1 - John', '2 - Jane', ...]

// Callback-ul poate returna, de asemenea, un array cu o pereche cheie/valoare:
$names = $result->fetchPairs(fn($row) => [$row->name, $row->age]);
// ['John' => 46, 'Jane' => 21, ...]
```


fetchField(): mixed .[method]
-----------------------------

Preia valoarea primei celule din rândul curent. Dacă nu mai sunt disponibile alte rânduri, se returnează `null`. Această metodă avansează pointerul intern la rândul următor.

```php
$result = $database->query('SELECT name FROM users');
$name = $result->fetchField(); // preia numele din primul rând
```


fetchList(): ?array .[method]
-----------------------------

Preia rândul ca un array indexat. Dacă nu mai sunt disponibile alte rânduri, se returnează `null`. Această metodă avansează pointerul intern la rândul următor.

```php
$result = $database->query('SELECT name, email FROM users');
$row = $result->fetchList(); // ['John', 'john@example.com']
```


getRowCount(): ?int .[method]
-----------------------------

Returnează numărul de rânduri afectate de ultima interogare `UPDATE` sau `DELETE`. Pentru interogările `SELECT`, se returnează numărul de rânduri extrase, dar acesta nu poate fi întotdeauna cunoscut - în astfel de cazuri, se returnează `null`.


getColumnCount(): ?int .[method]
--------------------------------

Returnează numărul de coloane din `ResultSet`.


Informații despre interogare .[#toc-query-information]
======================================================

Pentru a obține detalii despre cea mai recentă interogare executată, utilizați:

```php
echo $database->getLastQueryString(); // afișează interogarea SQL

$result = $database->query('SELECT * FROM articles');
echo $result->getQueryString();    // afișează interogarea SQL
echo $result->getTime();           // afișează timpul de execuție în secunde
```

Pentru a afișa rezultatul ca un tabel HTML, utilizați:

```php
$result = $database->query('SELECT * FROM articles');
$result->dump();
```

De asemenea, puteți obține informații despre tipurile de coloane din `ResultSet`:

```php
$result = $database->query('SELECT * FROM articles');
$types = $result->getColumnTypes();

foreach ($types as $column => $type) {
	echo "$column is of type $type->type"; // de exemplu, "id este de tip int
}
```


Înregistrarea interogărilor .[#toc-query-logging]
-------------------------------------------------

Puteți implementa înregistrarea interogărilor personalizate. Evenimentul `onQuery` este o serie de callback-uri care sunt invocate după fiecare execuție a interogării:

```php
$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');
	}
};
```

Calea SQL

Puteți lucra cu Nette Database în două moduri: scriind interogări SQL (metoda SQL) sau lăsând SQL să fie generat automat(metoda Explorer). Modul SQL vă permite să construiți în siguranță interogări, păstrând în același timp controlul deplin asupra structurii acestora.

Consultați Conectare și configurare pentru detalii despre configurarea conexiunii la baza de date.

Interogare de bază

Metoda query() execută interogări ale bazei de date și returnează un obiect ResultSet care reprezintă rezultatul. Dacă interogarea eșuează, metoda aruncă o excepție. Puteți parcurge în buclă rezultatul interogării folosind o buclă foreach sau folosind una dintre funcțiile de ajutor.

$result = $database->query('SELECT * FROM users');

foreach ($result as $row) {
	echo $row->id;
	echo $row->name;
}

Pentru a introduce valori în siguranță în interogările SQL, utilizați interogări parametrizate. Nette Database face acest lucru foarte simplu: trebuie doar să adăugați o virgulă și valoarea la interogarea SQL.

$database->query('SELECT * FROM users WHERE name = ?', $name);

Pentru parametrii multipli, puteți fie să intercalați interogarea SQL cu parametrii:

$database->query('SELECT * FROM users WHERE name = ?', $name, 'AND age > ?', $age);

Sau scrieți mai întâi întreaga interogare SQL și apoi adăugați toți parametrii:

$database->query('SELECT * FROM users WHERE name = ? AND age > ?', $name, $age);

Protecție împotriva injectării SQL

De ce este important să utilizați interogări parametrizate? Pentru că vă protejează de atacurile de tip SQL injection, în care atacatorii pot injecta comenzi SQL rău intenționate pentru a manipula sau a accesa datele din baza de date.

Niciodată nu introduceți variabile direct într-o interogare SQL! Utilizați întotdeauna interogări parametrizate pentru a vă proteja împotriva injecțiilor SQL.

// ❌ UNSAFE CODE - vulnerabil la injectarea SQL
$database->query("SELECT * FROM users WHERE name = '$name'");

// ✅ Interogare parametrizată sigură
$database->query('SELECT * FROM users WHERE name = ?', $name);

Asigurați-vă că vă familiarizați cu potențialele riscuri de securitate.

Tehnici de interogare

Condiții WHERE

Puteți scrie condițiile WHERE ca un array asociativ, în care cheile sunt numele coloanelor, iar valorile sunt datele de comparat. Nette Database selectează automat cel mai adecvat operator SQL în funcție de tipul valorii.

$database->query('SELECT * FROM users WHERE', [
	'name' => 'John',
	'active' => true,
]);
// WHERE `name` = 'John' AND `active` = 1

De asemenea, puteți specifica explicit operatorul în cheie:

$database->query('SELECT * FROM users WHERE', [
	'age >' => 25,           // utilizează operatorul >
	'name LIKE' => '%John%', // utilizează operatorul LIKE
	'email NOT LIKE' => '%example.com%', // utilizează operatorul NOT LIKE
]);
// WHERE `age` > 25 AND `name` LIKE '%John%' AND `email` NOT LIKE '%example.com%'

Cazurile speciale, cum ar fi valorile null sau matricele, sunt gestionate automat:

$database->query('SELECT * FROM products WHERE', [
	'name' => 'Laptop',         // utilizează operatorul =
	'category_id' => [1, 2, 3], // utilizează IN
	'description' => null,      // uses IS NULL
]);
// WHERE `name` = 'Laptop' AND `category_id` IN (1, 2, 3) AND `description` IS NULL

Pentru condiții negative, utilizați operatorul NOT:

$database->query('SELECT * FROM products WHERE', [
	'name NOT' => 'Laptop',         // utilizează operatorul <>
	'category_id NOT' => [1, 2, 3], // utilizează NOT IN
	'description NOT' => null,      // uses IS NOT NULL
	'id' => [],                     // sărit
]);
// WHERE `name` <> 'Laptop' AND `category_id` NOT IN (1, 2, 3) AND `description` IS NOT NULL

În mod implicit, condițiile sunt combinate utilizând operatorul AND. Puteți schimba acest comportament utilizând marcajul ?or.

ORDER BY Reguli

Clauza ORDER BY poate fi definită ca un array, unde cheile reprezintă coloane, iar valorile sunt booleeni care indică ordinea crescătoare:

$database->query('SELECT id FROM author ORDER BY', [
	'id' => true,  // ascendentă
	'name' => false, // descrescător
]);
// SELECT id FROM author ORDER BY `id`, `name` DESC

Inserarea datelor (INSERT)

Pentru a introduce înregistrări, utilizați instrucțiunea SQL INSERT.

$values = [
	'name' => 'John Doe',
	'email' => 'john@example.com',
];
$database->query('INSERT INTO users ?', $values);
$userId = $database->getInsertId();

Metoda getInsertId() returnează ID-ul ultimului rând introdus. Pentru anumite baze de date (de exemplu, PostgreSQL), trebuie să specificați numele secvenței utilizând $database->getInsertId($sequenceId).

De asemenea, puteți trece valori speciale, cum ar fi fișiere, obiecte DateTime sau tipuri enum, ca parametri.

Inserarea simultană a mai multor înregistrări:

$database->query('INSERT INTO users ?', [
	['name' => 'User 1', 'email' => 'user1@mail.com'],
	['name' => 'User 2', 'email' => 'user2@mail.com'],
]);

Efectuarea unei INSERTĂRI pe loturi este mult mai rapidă deoarece se execută o singură interogare a bazei de date în loc de mai multe interogări individuale.

Nota de securitate: Nu utilizați niciodată date nevalidate ca $values. Familiarizați-vă cu riscurile posibile.

Actualizarea datelor (UPDATE)

Pentru a actualiza înregistrările, utilizați instrucțiunea SQL UPDATE.

// Actualizarea unei singure înregistrări
$values = [
	'name' => 'John Smith',
];
$result = $database->query('UPDATE users SET ? WHERE id = ?', $values, 1);

Puteți verifica numărul de rânduri afectate utilizând $result->getRowCount().

Puteți utiliza operatorii += și -= în UPDATE:

$database->query('UPDATE users SET ? WHERE id = ?', [
	'login_count+=' => 1, // increment login_count
], 1);

Pentru a introduce sau actualiza o înregistrare dacă aceasta există deja, utilizați tehnica 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 `name` = 'Jim', `year` = 1978

Rețineți că Nette Database recunoaște contextul comenzii SQL în care este utilizat un parametru cu un array și generează codul SQL în consecință. De exemplu, a construit (id, name, year) VALUES (123, 'Jim', 1978) din primul array, în timp ce l-a convertit pe al doilea în name = 'Jim', year = 1978. Acest aspect este tratat mai detaliat în secțiunea Sfaturi pentru construirea SQL.

Ștergerea datelor (DELETE)

Pentru a șterge înregistrări, utilizați instrucțiunea SQL DELETE. Exemplu cu numărul de rânduri șterse:

$count = $database->query('DELETE FROM users WHERE id = ?', 1)
	->getRowCount();

Sugestii de construcție SQL

Lockerele SQL vă permit să controlați modul în care valorile parametrilor sunt încorporate în expresiile SQL:

Indicație Descriere Folosit automat pentru
?name Folosit pentru nume de tabele sau coloane
?values Generează (key, ...) VALUES (value, ...) INSERT ... ?, REPLACE ... ?
?set Generează atribuiri key = value, ... SET ?, KEY UPDATE ?
?and Unește condițiile dintr-un array cu AND WHERE ?, HAVING ?
?or Unește condițiile într-un array cu OR
?order Generează clauza ORDER BY ORDER BY ?, GROUP BY ?

Pentru inserarea dinamică a numelor de tabele sau coloane, utilizați marcajul ?name. Nette Database asigură scăparea corectă în conformitate cu convențiile bazei de date (de exemplu, includerea în backticks pentru MySQL).

$table = 'users';
$column = 'name';
$database->query('SELECT ?name FROM ?name WHERE id = 1', $column, $table);
// SELECT `name` FROM `users` WHERE id = 1 (în MySQL)

Avertisment: Folosiți marcajul ?name numai pentru numele validate ale tabelelor și coloanelor. În caz contrar, riscați vulnerabilități de securitate.

De obicei, nu este necesar să specificați alte indicii, deoarece Nette utilizează detectarea automată inteligentă atunci când construiește interogări SQL (a se vedea a treia coloană din tabel). Cu toate acestea, le puteți utiliza în situațiile în care doriți să combinați condițiile utilizând OR în loc de 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'

Valori speciale

În plus față de tipurile scalare standard (de exemplu, string, int, bool), puteți trece și valori speciale ca parametri:

  • Fișiere: Utilizați fopen('file.png', 'r') pentru a introduce conținutul binar al unui fișier.
  • Data și ora: Obiectele DateTime sunt convertite automat în formatul de dată al bazei de date.
  • Valori enum: Instanțele din enum sunt convertite în valorile corespunzătoare.
  • Literale SQL: Create utilizând Connection::literal('NOW()'), acestea sunt inserate direct în interogare.
$database->query('INSERT INTO articles ?', [
	'title' => 'My Article',
	'published_at' => new DateTime,
	'content' => fopen('image.png', 'r'),
	'state' => Status::Draft,
]);

Pentru bazele de date care nu oferă suport nativ pentru tipul datetime (de exemplu, SQLite și Oracle), valorile DateTime sunt convertite în conformitate cu opțiunea de configurare formatDateTime (implicit: U pentru timestamp Unix).

Literali SQL

În unele cazuri, poate fi necesar să introduceți cod SQL brut ca valoare fără a-l trata ca pe un șir de caractere sau fără a-l scăpa. Pentru aceasta, utilizați obiecte din clasa Nette\Database\SqlLiteral, care pot fi create utilizând metoda Connection::literal().

$result = $database->query('SELECT * FROM users WHERE', [
	'name' => $name,
	'year >' => $database::literal('YEAR()'),
]);
// SELECT * FROM users WHERE (`name` = 'Jim') AND (`year` > YEAR())

Alternativ:

$result = $database->query('SELECT * FROM users WHERE', [
	'name' => $name,
	$database::literal('year > YEAR()'),
]);
// SELECT * FROM users WHERE (`name` = 'Jim') AND (year > YEAR())

Literalele SQL pot conține și parametri:

$result = $database->query('SELECT * FROM users WHERE', [
	'name' => $name,
	$database::literal('year > ? AND year &lt; ?', $min, $max),
]);
// SELECT * FROM users WHERE `name` = 'Jim' AND (an > 1978 AND an < 2017)

Acest lucru permite combinații flexibile:

$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')

Preluarea datelor

Scurtături pentru interogările SELECT

Pentru a simplifica recuperarea datelor, clasa Connection oferă mai multe scurtături care combină un apel la query() cu un apel ulterior la fetch*(). Aceste metode acceptă aceiași parametri ca și query(), și anume o interogare SQL și parametri opționali. O descriere detaliată a metodelor fetch*() poate fi găsită mai jos.

fetch($sql, ...$params): ?Row Execută interogarea și extrage primul rând ca obiect Row.
fetchAll($sql, ...$params): array Execută interogarea și extrage toate rândurile sub forma unei serii de obiecte Row.
fetchPairs($sql, ...$params): array Execută interogarea și extrage un array asociativ în care prima coloană este cheia, iar a doua este valoarea.
fetchField($sql, ...$params): mixed Execută interogarea și extrage valoarea primei celule din primul rând.
fetchList($sql, ...$params): ?array Execută interogarea și extrage primul rând ca un array indexat.

Exemplu:

// fetchField() - returnează valoarea primei celule
$count = $database->query('SELECT COUNT(*) FROM articles')
	->fetchField();

foreach – Iterarea peste rânduri

După executarea unei interogări, este returnat un obiect ResultSet, care vă permite să iterați peste rezultate în diverse moduri. Cel mai simplu și mai eficient mod de a prelua rânduri din memorie este prin iterarea într-o buclă foreach. Această metodă procesează rândurile unul câte unul și evită stocarea simultană a tuturor datelor în memorie.

$result = $database->query('SELECT * FROM users');

foreach ($result as $row) {
	echo $row->id;
	echo $row->name;
	//...
}

Rețeaua ResultSet poate fi iterată o singură dată. Dacă trebuie să o iterați de mai multe ori, trebuie mai întâi să încărcați datele într-un array, de exemplu, utilizând metoda fetchAll().

fetch(): ?Row

Execută interogarea și extrage un singur rând ca obiect Row. Dacă nu mai sunt disponibile alte rânduri, se returnează null. Această metodă avansează pointerul intern la rândul următor.

$result = $database->query('SELECT * FROM users');
$row = $result->fetch(); // preia primul rând
if ($row) {
	echo $row->name;
}

fetchAll(): array

Preia toate rândurile rămase din ResultSet sub forma unei serii de obiecte Row.

$result = $database->query('SELECT * FROM users');
$rows = $result->fetchAll(); // preia toate rândurile
foreach ($rows as $row) {
	echo $row->name;
}

fetchPairs(string|int|null $key = null, string|int|null $value = null)array

Preia rezultatele sub forma unui array asociativ. Primul argument specifică coloana care urmează să fie utilizată ca cheie, iar al doilea specifică coloana care urmează să fie utilizată ca valoare:

$result = $database->query('SELECT id, name FROM users');
$names = $result->fetchPairs('id', 'name');
// [1 => "John Doe", 2 => "Jane Doe", ...]

Dacă este furnizat doar primul parametru, valoarea va fi întregul rând (ca obiect Row ):

$rows = $result->fetchPairs('id');
// [1 => rând (id: 1, nume: "John"), 2 => rând (id: 2, nume: "Jane"), ...]

Dacă null este trecut ca cheie, matricea va fi indexată numeric începând de la zero:

$names = $result->fetchPairs(null, 'name');
// [0 => "John Doe", 1 => "Jane Doe", ...]

fetchPairs(Closure $callback)array

Alternativ, puteți furniza un callback care determină perechile cheie-valoare sau valorile pentru fiecare rând.

$result = $database->query('SELECT * FROM users');
$items = $result->fetchPairs(fn($row) => "$row->id - $row->name");
// ['1 - John', '2 - Jane', ...]

// Callback-ul poate returna, de asemenea, un array cu o pereche cheie/valoare:
$names = $result->fetchPairs(fn($row) => [$row->name, $row->age]);
// ['John' => 46, 'Jane' => 21, ...]

fetchField(): mixed

Preia valoarea primei celule din rândul curent. Dacă nu mai sunt disponibile alte rânduri, se returnează null. Această metodă avansează pointerul intern la rândul următor.

$result = $database->query('SELECT name FROM users');
$name = $result->fetchField(); // preia numele din primul rând

fetchList(): ?array

Preia rândul ca un array indexat. Dacă nu mai sunt disponibile alte rânduri, se returnează null. Această metodă avansează pointerul intern la rândul următor.

$result = $database->query('SELECT name, email FROM users');
$row = $result->fetchList(); // ['John', 'john@example.com']

getRowCount(): ?int

Returnează numărul de rânduri afectate de ultima interogare UPDATE sau DELETE. Pentru interogările SELECT, se returnează numărul de rânduri extrase, dar acesta nu poate fi întotdeauna cunoscut – în astfel de cazuri, se returnează null.

getColumnCount(): ?int

Returnează numărul de coloane din ResultSet.

Informații despre interogare

Pentru a obține detalii despre cea mai recentă interogare executată, utilizați:

echo $database->getLastQueryString(); // afișează interogarea SQL

$result = $database->query('SELECT * FROM articles');
echo $result->getQueryString();    // afișează interogarea SQL
echo $result->getTime();           // afișează timpul de execuție în secunde

Pentru a afișa rezultatul ca un tabel HTML, utilizați:

$result = $database->query('SELECT * FROM articles');
$result->dump();

De asemenea, puteți obține informații despre tipurile de coloane din ResultSet:

$result = $database->query('SELECT * FROM articles');
$types = $result->getColumnTypes();

foreach ($types as $column => $type) {
	echo "$column is of type $type->type"; // de exemplu, "id este de tip int
}

Înregistrarea interogărilor

Puteți implementa înregistrarea interogărilor personalizate. Evenimentul onQuery este o serie de callback-uri care sunt invocate după fiecare execuție a interogării:

$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');
	}
};