Pomoc - Szukaj - Użytkownicy - Kalendarz
Pełna wersja: [PHP5] Rozszerzanie singletona?
Forum PHP.pl > Forum > PHP > Object-oriented programming
MacDada
Hej,

mam sobie klasę DB, która zajmuje się łączeniem z bazą danych oraz obsługą błędów. Gdy chcę zrobić coś z danymi tworzę nową klasę rozszerzającą klasę DB i w niej umieszczam odpowiednią metodę.

Zależy mi na możliwie dużej wydajności, stąd nie tworzę uniwersalnej klasy ze wszystkimi możliwymi metodami, ale każdy rodzaj działania na bazie wymaga oddzielnej klasy, żeby była ona możliwie najkrótsza.

Klasa DB:
http://wklej.to/O7pd

Przykładowa klasa pobierająca dane użytkownika:
http://wklej.to/Glmo

Przykładowe zastosowanie:
http://wklej.to/3TPK

Przykładowy wynik:
http://wklej.to/UOtk

Problem pojawia się, gdy chcę skorzystać z dwóch różnych operacji na bazie – niepotrzebnie wtedy uruchamiam dwie instancje klasy DB, czyli niepotrzebnie zajmuję pamięć i wielokrotnie łączę się z bazą zamiast skorzystać z jednego połączenia.

Chciałem zrobić klasę DB jako singleton i tu się pojawia pytanie:
czy mogę rozszerzać singletona? Jak to wtedy wywołać?

Chodzi o to, żeby mieć jedną instancję klasy bazowej DB, z którego korzystać może wiele innych klas ją rozszerzających.

Dzięki i pozdr.
Fifi209
Zależy co masz na myśli pisząc: "rozszerzać singletona".

Co do samego wywołania, tak samo jak z konstruktorem parent:: i jedziesz ;]

Mam nadzieję, że klasa getuser jest tylko przykładowa i nie rozbijesz wszystkiego na takie malutkie klasy.
zegarek84
jak masz takie same metody publiczne czy chronione u rodzica to nie musisz ich nadpisywać przy potomkach - dlatego potomek tutaj DB_getUser nie musiał mieć definicji __construct()...

jeśli chcesz mieć jeden uchwyt do a w zasadzie tworząc nowe instancje obiektu to daleko od singleton'a - jeśli tak chciałbyś zrobić to klasa bazy po prostu obsługuje bazę i instancje tej klasy przekazujesz innym klasom gdzie z db zwracasz uchwyt - oczywiście w modelu singleton klasę tworzy się tak by zawsze był zwracany stary obiekt a nie nowy [chyba, że nie istnieje...] - obiekt taki najczęściej jest zapamiętany w zmiennej statycznej...

ale na to co Ty zrobiłeś zamiast singleton'a to zrób deczko inny konstruktor i inaczej przechowuj zmienne i uchwyt do bazy - w zmiennych statycznych - jeśli one są ustawione to w konstruktorze nic nie rób... w metodach uchwyt do bazy odwołuj się do zmiennej statycznej - oczywiście zdefiniuj je u najwyższego rodzica by potomkowie mieli także dostęp do tych zmiennych statycznych...
czyli zamiast:
  1. class DB
  2. {
  3. protected $connection;
  4. protected $_printExceptions;
  5.  
  6. public function __construct($db, $printExceptions=FALSE)
  7. {
  8. $this->_printExceptions = $printExceptions;
  9.  
  10. try
  11. {
  12. $this->connection = new PDO('mysql:host='.$db['host'].'; port='.$db['port'].'; dbname='.$db['name'], $db['user'], $db['pass'], array(PDO::MYSQL_ATTR_INIT_COMMAND => "SET NAMES utf8"));
  13.  
  14. $this->connection->setAttribute(PDO::ATTR_ERRMODE, PDO::ERRMODE_EXCEPTION);
  15. }
  16. catch(PDOException $e)
  17. {
  18. $this->_handleException($e, 'Database class exception while connecting to the database:');
  19. }
  20. }
  21. }

możesz zrobić:
  1. class DB
  2. {
  3. protected static $connection;
  4. protected static $_printExceptions;
  5.  
  6. public function __construct($db, $printExceptions=FALSE)
  7. {
  8. if(!isset(self::$connection)){
  9. self::$_printExceptions = $printExceptions;
  10.  
  11. try
  12. {
  13. self::$connection = new PDO('mysql:host='.$db['host'].'; port='.$db['port'].'; dbname='.$db['name'], $db['user'], $db['pass'], array(PDO::MYSQL_ATTR_INIT_COMMAND => "SET NAMES utf8"));
  14.  
  15. self::$connection->setAttribute(PDO::ATTR_ERRMODE, PDO::ERRMODE_EXCEPTION);
  16. }
  17. catch(PDOException $e)
  18. {
  19. $this->_handleException($e, 'Database class exception while connecting to the database:');
  20. }
  21. }
  22. }
  23. }

no i oczywiście uchwyt do bazy we wszystkich klasach także potomnych masz pod self::$connection
#luq
Klasa do obsługi bazy danych nie powinna być imho singletonem, bo istnieje przecież możliwość, że jedna aplikacja będzie korzystała z kilku baz. Moim zdaniem powinieneś robić to tak:

  1. $db = new DB();
  2.  
  3. $user = new User( $db );
  4. $log = new Log( $db );
  5.  
  6. $user->getAll();
  7. $log->get( 1 );


Twoja klasa DB_getUser nie powinna dziedziczyć od DB bo ona właściwie nie rozszerza możliwości tej klasy, jest to zwykły model.
Pilsener
Dokładnie, obsługą samej bazy powinna zajmować się oddzielna klasa, natomiast główny model danych to powinna być inna klasa, która połączenie z bazą pobiera w konstruktorze i to ona dopiero obsługuje zapytania i błędy natomiast poszczególne metody dostępu do danych mogą być rozszerzeniem tej klasy. Moim zdaniem dobrze przyjąć koncepcje, że jedna tabela w bazie = jeden model zawierający metody korzystające z tej tabeli. Bo inaczej jak? Potężna klasa do obsługi każdego "select * from tabela"? Get user może wyglądać tak:
  1. class Model_User extends Model_Glowny{
  2. protected $table_name = 'Tusers';
  3.  
  4. public function getUser($id){
  5. $sql='SELECT `name`, `login`, `avatar`, `user_manager`, `event_manager` FROM `Tusers` WHERE id=$id';
  6. $sql = $this->prepare($sql);
  7. return $this->query($sql);
  8. }
  9.  
  10. public function countUsers(){
  11. //i tak dalej
  12.  
  13. }
  14. }


Spójna koncepcja nazewnictwa i komentarze też mile widziane.
To jest wersja lo-fi głównej zawartości. Aby zobaczyć pełną wersję z większą zawartością, obrazkami i formatowaniem proszę kliknij tutaj.
Invision Power Board © 2001-2024 Invision Power Services, Inc.