Pomoc - Szukaj - Użytkownicy - Kalendarz
Pełna wersja: __get zwracający tablicę w php 5.2.0
Forum PHP.pl > Forum > PHP > Object-oriented programming
cicik
Jak wiadomo ukazała się niedawno wersja 5.2.0 php
Zainstalowałem ją czym prędzej ale napotkałem problem.

Jest klasa:

  1. <?php
  2. class test {
  3. private $zmienne = array();
  4.  
  5. public function & __get($nazwa) {
  6. return $this->$zmienne[$nazwa];
  7. }
  8. }
  9. ?>


Do tej pory gdy robiłem

  1. <?php
  2. $test = new test;
  3. $test->dane['cos_tam'] = 'wartosc';
  4. ?>


otrzymywałem efekt zgodny z zamierzeniem (wiadomo, że klucz $this->zmienne['dane'] powinien już wcześniej istnieć i istnieje). Metoda __get zwraca referencję do odpowiedniego obszaru w pamięci i przypisywana jest tam wartość 'wartosc'.

W php 5.1.x działa bez problemu.
W php 5.2.0 nie przypisuje.

Spotkał się jeszcze ktoś z tym.

Dodam, że generalnie metody magiczne działają.
dr_bonzo
Nie probowalem ale dopiero od 5.2 mozna ponoc zmieniac zwrocona tablice przez __get(), a sproboj usunac '&' sprzed metody.

O:

Cytat
Fixed bug #38146 Cannot use array returned from foo::__get('bar') in write context). (Dmitry) (5.2)
cicik
Cytat(dr_bonzo @ 8.11.2006, 09:03:30 ) *
Nie probowalem ale dopiero od 5.2 mozna ponoc zmieniac zwrocona tablice przez __get(), a sproboj usunac '&' sprzed metody.


Już tego próbowałem zanim wysłałem posta na forum.
Wcześniej znalazłem też notkę o tym bugu, o którym mówisz i byłem mocno zaskoczony.

Poza tym ja nie dostaję komunikatu "Fatal error: Cannot use array returned from foo::__get
('bar') in write context", który jest zaqcytowany w noce o tym bugu.
U mnie przypisanie jest wykonywane tak jakby poprawnie. Tylko jakby nie widział, że to jest referencja.

Ha!
Zrobilem jeszcze testy.
mam klase jak ostatnio tylko lekko zmodyfikowana.
bardziej odzwierciedla przypadek z mojego rzeczywstego skryptu bo to tutaj chyba lezy bug.

  1. <?php
  2. class test {
  3. private $zmienne = array();
  4.  
  5. public function & __get($nazwa) {
  6. return $this->zmienne[$nazwa];
  7. }
  8. public function __set($nazwa, $wartosc) {
  9. $this->zmienne[$nazwa] = $wartosc;
  10. }
  11. }
  12. ?>


I teraz:

  1. <?php
  2. $test1 = new test;
  3. $test1->dane['cos1'] = 'cos1';
  4. $test1->dane['cos2'] = 'cos2';
  5.  
  6. $test2 = new test;
  7. $test2->dane['przepisane'] = $test1->dane;
  8. ?>


Teraz jesli zrobie:

  1. <?php
  2. echo($test2->dane['przepisane']['cos1']);
  3. ?>


To na ekranie jest drukowane:

  1. Array


Dodam, ze na php 5.1.x nie ma problemów.

?>

Z dalszych testów wynika, że następujący kod:
  1. <?php
  2. $test1 = new test;
  3. $test1->dane['cos1'] = 'cos1';
  4. $test1->dane['cos2'] = 'cos2';
  5.  
  6. $test2 = new test;
  7.  
  8. $zmienna['przepisane'] = $test1->dane;
  9. $test2->dane = $zmienna;
  10. ?>


Zachowuje się poprawnie.
dr_bonzo
A probowales BEZ '&' questionmark.gif

Cytat
Z dalszych testów wynika, że następujący kod:
  1. <?php
  2. ?php
  3. $test1 = new test;
  4. $test1->dane['cos1'] = 'cos1';
  5. $test1->dane['cos2'] = 'cos2';
  6.  
  7. $test2 = new test;
  8.  
  9. $zmienna['przepisane'] = $test1->dane;
  10. $test2->dane = $zmienna;
  11. ?>


To chyba oczywiste, bo

  1. <?php
  2. $test2->dane = $zmienna;
  3. ?>


tu dziala __set() a stare dane zostana usuniete
cicik
Cytat(dr_bonzo @ 8.11.2006, 09:54:39 ) *
A probowales BEZ '&' questionmark.gif


No przecież napisałem, że próbowałem.
Efekt ten sam.
dr_bonzo
Sorry - nie zauwazylem.

A czemu nie zrobisz po prostu:

  1. <?php
  2. $test->cos = "x";
  3. ?>


zamiast uzywania tablicy?

(sciagam i instaluje php 5.2)
cicik
Cytat(dr_bonzo @ 8.11.2006, 10:13:25 ) *
Sorry - nie zauwazylem.

A czemu nie zrobisz po prostu:

  1. <?php
  2. $test->cos = "x";
  3. ?>


zamiast uzywania tablicy?

(sciagam i instaluje php 5.2)


Bo kod, który podałem jest tylko przykładem ilustrującym błąd.
W rzeczywistości jest mi to po prostu potrzebne.
dr_bonzo
Testuje na 5.2.0 error_reporting( E_ALL | E_STRICT)

1.
  1. <?php
  2. class test {
  3. private $zmienne = array();
  4.  
  5. public function __get($nazwa) {
  6. return $this->zmienne[$nazwa]; // usunalem $ sprzed 'zmienne'
  7. }
  8. }
  9. $test = new test;
  10. echo $test->dane['cos_tam'];
  11. ?>


Kod
Notice: Undefined index: dane in /services/php.dev/www/test/ref_do_tabli_z__get.php on line 33

to jasne -- uzywa __get()'a a taki index nie istnieje
a przypisanie
  1. <?php
  2. $test->dane['cos_tam'] = 'costam';
  3. ?>

nie zadziala
Kod
Notice: Indirect modification of overloaded property test::$dane has no effect in /services/php.dev/www/test/ref_do_tabli_z__get.php on line 37


2.
  1. <pre><?php
  2. class test {
  3. private $zmienne = array();
  4.  
  5. public function & __get($nazwa) {
  6. return $this->zmienne[$nazwa];
  7. }
  8. public function __set($nazwa, $wartosc) {
  9. $this->zmienne[$nazwa] = $wartosc;
  10. }
  11. }
  12.  
  13. $test1 = new test;
  14. $test1->dane['cos1'] = 'cos1';
  15. $test1->dane['cos2'] = 'cos2';
  16.  
  17. $test2 = new test;
  18. $test2->dane['przepisane'] = $test1->dane;
  19. echo($test2->dane['przepisane']['cos1']);
  20. var_dump( $test2->dane['przepisane']['cos1'] );
  21. ?></pre>

wynik:
  1. <?php
  2. Notice: Indirect modification of overloaded property test::$dane has no effect in /services/php.dev/www/test/ref_do_tabli_z__get.php on line 73
  3.  
  4.  
  5.  
  6. Notice: Indirect modification of overloaded property test::$dane has no effect in /services/php.dev/www/test/ref_do_tabli_z__get.php on line 74
  7.  
  8.  
  9.  
  10. Notice: Indirect modification of overloaded property test::$dane has no effect in /services/php.dev/www/test/ref_do_tabli_z__get.php on line 77
  11.  
  12. NULL
  13. ?>


tobie to sie nie pojawia?


3. ta sama klasa ( z __set())
  1. <pre><?php
  2. class test {
  3. private $zmienne = array();
  4.  
  5. public function & __get($nazwa) {
  6. return $this->zmienne[$nazwa];
  7. }
  8. public function __set($nazwa, $wartosc) {
  9. $this->zmienne[$nazwa] = $wartosc;
  10. }
  11. }
  12.  
  13. $test1 = new test;
  14. var_dump( $test1 );
  15.  
  16. $test1->dane = 'aaa';
  17. var_dump( $test1 ); // $zmienne jest tablica, 'dane' -- jej indeksem
  18.  
  19. $test1->dane = array('aaa'); // $zmienne jest tablica, 'dane' -- indeksem,
  20. // $zmienne['dane'] jest tablica z jednym elementem 'aaa'
  21. var_dump( $test1 ); // co widac
  22.  
  23. $x = ( $test1->dane[0] == 'aaa' ); // tu tez
  24. var_dump( $x ); // true
  25.  
  26. $test1->xxx[8] == 'aaa'; // tak sie nie da
  27. var_dump( $test1 ); // no bo $zmienne['xxx'] nie jest tablica
  28. ?></pre>


wynik
Kod
object(test)#1 (1) {
  ["zmienne:private"]=>
  array(0) {
  }
}
object(test)#1 (1) {
  ["zmienne:private"]=>
  array(1) {
    ["dane"]=>
    string(3) "aaa"
  }
}
object(test)#1 (1) {
  ["zmienne:private"]=>
  array(1) {
    ["dane"]=>
    array(1) {
      [0]=>
      string(3) "aaa"
    }
  }
}
bool(true)
object(test)#1 (1) {
  ["zmienne:private"]=>
  array(2) {
    ["dane"]=>
    array(1) {
      [0]=>
      string(3) "aaa"
    }
    ["xxx"]=>
    NULL
  }
}


Linki
http://framework.zend.com/issues/browse/ZF-460
http://www.derickrethans.nl/overloaded_properties_get.php
cicik
Nie jestem teraz w domu ale sprawdzę jak wrócę. Pewnie mam wyłączone pokazywanie Notice.
Z linków, które podałeś wynika, że dane zwracane przez __get, nie będące obiektami są read-only.

Bez sensu...
envp
nie powinno byc __set() ?

Edit: sorka nie zauwazylem ;/
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-2025 Invision Power Services, Inc.