| FAQ |
| Kalender |
|
|
|
|
#1 | ||
|
|||
|
Klarade millennium-buggen
|
Nu har jag fått till det som jag vill ha det, ett tydligt felmeddelande när man använder odefinierade egenskaper på en klass (det borde vara inbyggt i ett objektorienterat språk):
Kod:
<?php
class Person
{
private $firstname;
private $lastname;
public function __construct($pFirstname, $pLastname)
{
$this->firstname = $pFirstname;
$this->lastname = $pLastname;
}
public function __set($name, $value)
{
switch ($name)
{
case 'firstname':
$this->firstname = $value;
break;
case 'lastname':
$this->lastname = $value;
break;
default:
# By throwing Exception for undefined property we can enforce strict property definition rules
# we also achive better error messages for improved debugging during development
# this is necessary because of lack of support for strict properties in php 5.3.15
$error = "Error: __set property: " . $name . " not supported by class: " . __CLASS__ ;
throw new Exception($error);
break;
}
}
public function __get($name)
{
switch ($name)
{
case 'firstname':
return $this->firstname;
break;
case 'lastname':
return $this->lastname;
break;
default:
# By throwing Exception for undefined property we can enforce strict property definition rules
# we also achive better error messages for improved debugging during development
# this is necessary because of lack of support for strict properties in php 5.3.15
$error = "Error: __get property: " . $name . " not supported by class: " . __CLASS__ . ".";
throw new Exception($error);
break;
}
return null;
}
/*
* a String representation for all Persons.
*/
public function __toString()
{
return $this->firstname . " " . $this->lastname;
}
}
?>
<?php
$rad = 0;
try
{
echo "\nTest of errormessage wehen faulty property ...\n";
echo "(row) - Firstname Lastname\n";
# create a PHP Array and initialize it with Person objects
$persons = array
(
new Person("Fredrik", "Framberg"),
new Person("Greta", "Gavelstam"),
new Person("Urban", "Urberg"),
new Person("Anna", "Ambtesteg"),
new Person("Henrik", "Hammarberg"),
new Person("Kristina", "Karlestam"),
new Person("Hans", "Hallin"),
new Person("Berit", "Bygdén")
);
# print out the results - calls Person->__toString().
foreach($persons as $person) echo "(" . ++$rad . ") - " . "$person\n";
}
catch (Exception $e)
{
echo 'Caught exception (1): ', $e->getMessage(), "\n";
}
try
{
echo "\nTest of creating new object of class Person ...\n";
$faultyPerson = new Person("Dennis","Harper");
echo "This message will hopefully be written to console.\n";
}
catch (Exception $e)
{
echo 'Caught exception (2): ', $e->getMessage(), "\n";
}
try
{
echo "\nTest of errormessage when faulty __set property ...\n";
echo "(" . ++$rad . ") - " . "$faultyPerson\n";
$faultyPerson->Age = 34;
$faultyPerson->ShoeSize = 45;
echo "Age: " . $faultyPerson->Age . " years.\n";
echo "ShoeSize: " . $faultyPerson->ShoeSize . ".\n";
echo "This message will not be written to console.\n";
}
catch (Exception $e)
{
echo 'Caught exception (3): ', $e->getMessage(), "\n";
}
try
{
echo "\nTest of errormessage when faulty __get property ...\n";
echo "(" . ++$rad . ") - " . "$faultyPerson\n";
echo "ShoeSize: " . $faultyPerson->ShoeSize . ".\n";
echo "Age: " . $faultyPerson->Age . " years.\n";
echo "This message will not be written to console.\n";
}
catch (Exception $e)
{
echo 'Caught exception (4): ', $e->getMessage(), "\n";
}
echo "\nDone testing class: Person.\n---------------------------------------\n\n";
?>
Senast redigerad av Conny Westh den 2012-07-24 klockan 01:39 |
||
|
|
Svara med citat
|
|
|
#2 | ||
|
|||
|
Flitig postare
|
Edit: äh, glöm det....
Senast redigerad av HenrikAI den 2012-07-24 klockan 10:59 |
||
|
|
Svara med citat
|
|
|
#3 | ||
|
|||
|
Flitig postare
|
Edit: glöm det...
|
||
|
|
Svara med citat
|
|
|
#4 | |||
|
||||
|
Flitig postare
|
Nu vet jag inte vad du egentligen är ute efter men att behöva ange alla godkända egenskaper i __set och __get känns omständigt.
Jag skulle gjort såhär: http://pastebin.com/ZDzJM0NL Och för att lägga till nya egenskaper som ska kunna sättas definierar du först scopet och sen sätter dem till något defaultvärde i konstruktorn. |
|||
|
|
Svara med citat
|
|
|
#5 | ||
|
|||
|
Klarade millennium-buggen
|
Citat:
Det är en del av styrkan med att ha den objektorienterde paradigmen. Man ökar kvaliten på koden mångfalt genom att programspråket stöder denna princip. Jag har aldrig sett något annat sråk som tillåter odefinierade egenskaper i en klass. I enklare imperativa skriptspråk är det vanligt att ha odeklareade/dynamiska variabler, men i objektorienterade språk brukar det i vart fall finnas den implicita tvingande egenskapen (kanske 'restriktionen' på ren svenska). Att ha getters och setters är till för att man ska kunna ha beräknade egenskaper som ibland kan vara ganska komplexa. Men de ska vara enkla att använda. Man kan vidare ha validering av status på en egenskap genom att använda getters och setters, så man kan kasta en exception om det blir tokigheter. Det finns även situationer där man ha komplicerade formateringsregler av caption (dvs displayvärden) men man har fortfarande en ganska teknisk grundinformation, typexempel kan vara datum som kan vara lagrat som ett mycket kompakt lagringsformat men som visas med en text som är lätt för en människa att förstå. Andra exempel är "scientific notation" o.s.v. .... Det är kompilatorns styrka att den hjälper mig att hitta fel under utvecklingsarbetet, jag vill ALDRIG råka ut för att fel smyger med och upptäcks först i produktion. Det är ofta 100-1000 gånger dyrare att åtgärda fel som upptäcks i produktion jämfört med under kodningsfasen eller kravfasen. Senast redigerad av Conny Westh den 2012-07-24 klockan 20:48 |
||
|
|
Svara med citat
|
|
|
#6 | ||
|
|||
|
Flitig postare
|
Citat:
Och magi är just oftast sådant vi vill undvika såvida det inte finns en annan lösning.Du behöver ju faktiskt inte använda dem och de har ett par nackdelar så som: - De är långsammare än direkt access eller access via en vanlig funktion - Det är svårare att skriva bra dokumentation för dem Den klassiska metoden är att skriva get/set metoder för varje värde du vill exponera och även den metoden jag rekommenderar även om det blir mer knackande på tangentbordet. Kod:
class Person
{
private $firstname;
public function getFirstname()
{
return $this->firstname;
}
public function setFirstname($firstname)
{
$this->firstname = $firstname;
}
}
|
||
|
|
Svara med citat
|
|
|
#7 | |||
|
||||
|
Flitig postare
|
Citat:
Jag menade att det känns som en dålig lösning att ha en switch-/if-sats i __get och __set. Min lösning visar ett sätt att slippa det. |
|||
|
|
Svara med citat
|
|
|
#8 | |||
|
||||
|
Nykomling
|
Citat:
Det är enligt mig självklart att man ska ha en switch/if i en get/set för att styra vad som händer, annars öppnar du ju upp klassen helt, och det kan ju inte vara tanken med get/set-metoderna. |
|||
|
|
Svara med citat
|
|
|
#9 | |||
|
||||
|
Flitig postare
|
Citat:
Som jag uppfattade det var målet att det inte ska gå att sättaegenskaper som objektet inte redan har. Alla som finns ska kunna sättas/hämtas. |
|||
|
|
Svara med citat
|
|
|
#10 | ||
|
|||
|
Klarade millennium-buggen
|
Citat:
Helst skulle jag önska att man kan deklarera: public class Person { // declataion of private variables private _firstname=""; private _lastname=""; // Jag skippar konstruktorn i detta exempel.... public __set_firstname($value) { $this->_firstname = $value; } public __get_firstname() { return $this->_firstname; } public __set_lastname($value) { $this->_lastname = $value; } public __get_lastname() { return $this->_lastname; } // Jag skippar även ToString-metoden.... } När man sen accessar egenskapen så vill jag göra så här: private $kalle = new Person(); $kalle.firstname = "Kalle"; $kalle.lastname = "Pettersson"; echo $kalle.firstname . " " . $kalle.lastname Resultatet ska då bli: "Kalle Pettersson" OBSERVERA att det inte alls är samma sak som att ha en funktion som geter getFirstName och en funktion som heter setFirstName, det är INTE getters och setters.... Ni som har jobbat med andra objektorienterade språk vet nog vad jag menar... . . . Senast redigerad av Conny Westh den 2012-07-31 klockan 03:46 |
||
|
|
Svara med citat
|
| Svara |
|
|