Baza danych Nette
Nette Database to potężna i elegancka warstwa bazy danych dla PHP z naciskiem na prostotę i inteligentne funkcje. Oferuje dwa uzupełniające się sposoby pracy z danymi – przy użyciu eksploratora do szybkiego rozwoju lub sposobu SQL do pełnej kontroli nad zapytaniami.
Sposób SQL
- Bezpieczne, parametryzowane zapytania
- Precyzyjna kontrola nad strukturą zapytań SQL
- Idealny do pisania złożonych zapytań z zaawansowanymi funkcjami
- Optymalizacja wydajności przy użyciu określonych funkcji SQL
Droga odkrywcy
- Szybki rozwój bez pisania SQL
- Intuicyjna obsługa relacji między tabelami
- Automatyczna optymalizacja zapytań
- Doskonały do szybkich i wygodnych interakcji z bazą danych
Instalacja
Bibliotekę można pobrać i zainstalować za pomocą Composera:
composer require nette/database
Obsługiwane bazy danych
Nette Database obsługuje następujące bazy danych:
Serwer bazy danych | Nazwa DSN | Obsługa eksploratora |
---|---|---|
MySQL (>= 5.1) | mysql | TAK |
PostgreSQL (>= 9.0) | pgsql | TAK |
SQLite 3 (>= 3.8) | sqlite | TAK |
Oracle | oci | NIE |
MS SQL (PDO_SQLSRV) | sqlsrv | TAK |
MS SQL (PDO_DBLIB) | mssql | NIE |
ODBC | odbc | NIE |
Dwa podejścia do pracy z bazą danych
W Nette Database można pisać zapytania SQL bezpośrednio (sposób SQL) lub pozwolić, aby SQL był generowany automatycznie (sposób Explorer). Zobaczmy, jak oba podejścia rozwiązują te same zadania:
Sposób SQL – Pisanie zapytań SQL
// Wstawianie rekordu
$database->query('INSERT INTO books', [
'author_id' => $authorId,
'title' => $bookData->title,
'published_at' => new DateTime,
]);
// Pobieranie rekordów: autorzy książek
$result = $database->query('
SELECT authors.*, COUNT(books.id) AS books_count
FROM authors
LEFT JOIN books ON authors.id = books.author_id
WHERE authors.active = 1
GROUP BY authors.id
');
// Wyświetlanie (nieoptymalne, generuje N dodatkowych zapytań)
foreach ($result as $author) {
$books = $database->query('
SELECT * FROM books
WHERE author_id = ?
ORDER BY published_at DESC
', $author->id);
echo "Author $author->name has written $author->books_count books:\n";
foreach ($books as $book) {
echo "- $book->title\n";
}
}
Sposób Explorer – automatyczne generowanie SQL
// Wstawianie rekordu
$database->table('books')->insert([
'author_id' => $authorId,
'title' => $bookData->title,
'published_at' => new DateTime,
]);
// Pobieranie rekordów: autorzy książek
$authors = $database->table('authors')
->where('active', 1);
// Wyświetl (automatycznie generuje tylko 2 zoptymalizowane zapytania)
foreach ($authors as $author) {
$books = $author->related('books')
->order('published_at DESC');
echo "Author $author->name has written {$books->count()} books:\n";
foreach ($books as $book) {
echo "- $book->title\n";
}
}
Podejście Explorer automatycznie generuje i optymalizuje zapytania SQL. W powyższym przykładzie sposób SQL generuje N+1 zapytań (jedno dla autorów i jedno dla książek każdego autora), podczas gdy Explorer wykonuje tylko dwa zoptymalizowane zapytania – jedno dla autorów i drugie dla wszystkich ich książek.
W zależności od potrzeb można dowolnie łączyć oba podejścia w aplikacji.
Połączenie i konfiguracja
Aby połączyć się z bazą danych, wystarczy utworzyć instancję klasy Nette\Database\Connection:
$database = new Nette\Database\Connection($dsn, $user, $password);
Parametr $dsn
(Data Source Name) ma taki sam format jak PDO, np.
host=127.0.0.1;dbname=test
. Jeśli połączenie nie powiedzie się, zostanie wyświetlony komunikat
Nette\Database\ConnectionException
.
Jednak wygodniejszą metodą jest użycie konfiguracji aplikacji.
Dodaj sekcję database
, a zostaną utworzone wymagane obiekty, w tym panel bazy danych na pasku debugowania Tracy.
database:
dsn: 'mysql:host=127.0.0.1;dbname=test'
user: root
password: password
Następnie obiekt połączenia można pobrać jako usługę z kontenera DI, np:
class Model
{
public function __construct(
// or Nette\Database\Explorer
private Nette\Database\Connection $database,
) {
}
}
Aby uzyskać więcej informacji, patrz konfiguracja bazy danych.
Ręczne tworzenie eksploratora
Jeśli nie korzystasz z kontenera Nette DI, możesz ręcznie utworzyć instancję Nette\Database\Explorer
:
// database connection
$connection = new Nette\Database\Connection('mysql:host=127.0.0.1;dbname=mydatabase', 'user', 'password');
// cache storage, implements Nette\Caching\Storage, e.g.:
$storage = new Nette\Caching\Storages\FileStorage('/path/to/temp/dir');
// handles database structure reflection
$structure = new Nette\Database\Structure($connection, $storage);
// defines rules for mapping table names, columns, and foreign keys
$conventions = new Nette\Database\Conventions\DiscoveredConventions($structure);
$explorer = new Nette\Database\Explorer($connection, $structure, $conventions, $storage);
Zarządzanie połączeniami
Po utworzeniu obiektu Connection
automatycznie łączy się on z bazą danych. Jeśli chcesz opóźnić
połączenie, włącz tryb leniwy w konfiguracji, ustawiając
lazy
, lub zrób to w następujący sposób:
$database = new Nette\Database\Connection($dsn, $user, $password, ['lazy' => true]);
Aby zarządzać połączeniem, użyj metod connect()
, disconnect()
, i reconnect()
.
connect()
ustanawia połączenie, jeśli nie zostało jeszcze ustanowione i może rzucićNette\Database\ConnectionException
.disconnect()
rozłącza się z bazą danych.reconnect()
rozłącza się, a następnie ponownie łączy z bazą danych i może również rzucićNette\Database\ConnectionException
.
Dodatkowo można monitorować zdarzenia połączenia za pomocą zdarzenia onConnect
, które jest tablicą
wywołań zwrotnych wykonywanych po połączeniu z bazą danych.
// Wywoływana po połączeniu z bazą danych
$database->onConnect[] = function($database) {
echo "Connected to the database";
};
Pasek debugowania Tracy
Jeśli używasz Tracy, panel Database na pasku debugowania jest automatycznie włączony. Wyświetla on wszystkie wykonane zapytania, ich parametry, czas wykonania i lokalizację w kodzie, w której zostały wywołane.