// bad
Example::calculate ($ obj )
// good
(new Example ($ obj ))->calculate ()
// bad
public static int $ foo ;
// good
public int $ foo ;
$ countries = ['Europe ' => 'Germany ' ];
// bad
foreach ($ countries as $ countries__key => $ countries__value ) { }
// good
foreach ($ countries as $ continent => $ country ) { }
$ dataToTest = [7 , 42 ];
// bad
foreach ($ dataToTest as $ dataToTest__value ) { }
// good
foreach ($ dataToTest as $ testValue ) { }
// bad
$ model ->calc()
// good
$ model ->calculate ()
// camel case
$ variableFooBar
function fooBarBaz () {}
// pascal case
class ExampleClass {}
// snake case
$ variable_in_template
Foo::first ()->variable_from_db
// bad
exampleFunction (['foo ' => 'bar ' , 'bar ' => 'baz ' ])
// good
exampleFunction (foo: 'bar ' , bar: 'baz ' )
// bad
$ foo = (object ) ['foo ' => 'bar ' , 'bar ' => 'baz ' ]; // anonymous object with properties 'foo' and 'bar'
// good
$ foo = (new Foo ())->foo ('bar ' )->bar ('baz ' )->get (); // Foo object with properties 'foo' and 'bar'
use classes / use getters / prevent arrays
// bad
$ mapping = getMapping ('test ' ); // ['foo' => 42, 'bar' => 7]
// good
$ mapping = (new MappingGetter ())->value ('test ' )->get (); // Mapping object
class Mapping {
public ?int $ foo ;
public ?int $ bar ;
}
class MappingGetter {
public string $ value ;
public function value ($ value ) { $ this ->value = $ value ; return $ this ; }
public function get () {
$ mapping = (new Mapping ());
$ mapping ->foo = 42 ;
$ mapping ->bar = 7 ;
/* ... more logic ... */
return $ mapping ;
}
}
prevent monster classes / outsource in separate classes
// bad
$ objectInBigModel ->calculateSpecialValue ();
// good
(new SpecialValueCalculator ($ objectInBigModel ))->calculate ()
DTOs (Data Transfer Objects) are little "packages" that hold data meant to be moved around
the data is consistent, structured and typed
they improve IDE support
// bad
$ userData = [
'id ' => 42 ,
'name ' => 'David ' ,
'email ' => '[email protected] '
];
print_r ($ userData ['id ' ]); // 42
// best (php >= 8.1)
$ userData = UserDTO::create (
42 ,
'David ' ,
'[email protected] '
);
print_r ($ userData ->id ); // 42
class UserDTO {
private function __construct (
public readonly int $ id ,
public readonly string $ name ,
public readonly string $ email
) {}
public static function create (int $ id , string $ name , string $ email ): UserDTO {
// potentially modify data before storing
// ...
return new self ($ id , $ name , $ email );
}
}
// good (php <8.1)
$ userData = UserDTO::create (
42 ,
'David ' ,
'[email protected] '
);
print_r ($ userData ->id ()); // 42
class UserDTO {
private int $ id ;
private string $ name ;
private string $ email ;
private function __construct (int $ id , string $ name , string $ email ) {
$ this ->id = $ id ;
$ this ->name = $ name ;
$ this ->email = $ email ;
}
public static function create (int $ id , string $ name , string $ email ): UserDTO {
// potentially modify data before storing
// ...
return new self ($ id , $ name , $ email );
}
public function id (): int {
return $ this ->id ;
}
public function name (): string {
return $ this ->name ;
}
public function email (): string {
return $ this ->email ;
}
}
VOs (value objects) represent one or multiple values
Cannot be changed once instantiated
Validate the value when instantiating
Have methods that convert, compare or format the value
DTOs and VOs serve different purposes and actually complement each other very well
DTOs transfer data (carry data between layers or systems, without holding much logic or responsibility)
VOs ensure the integrity of data (focus on modeling concepts within your domain)
// bad
$ price = (float ) 42 ;
$ price = '$ ' .number_format ($ price /100 , 2 );
echo $ price ;
// good
$ price = Price::fromInteger (42 );
$ price ->formattedWithSymbol ();
$ price ->value ();
final readonly class Price
{
public function __construct (private int $ value ) {
if ($ value < 0 ) {
throw new PriceCannotBeBelowZero ();
}
}
public static fromInteger (int $ value ): self {
return self($ value );
}
private function convertToDollars (): float {
return $ this ->value / 100 ;
}
public function formattedWithSymbol (): string {
return '$ ' . number_format ($ this ->convertToDollars (), 2 );
}
public function value (): int {
return $ this ->value ;
}
}
// bad
$ value = $ foo ->calc (save: true );
// good
$ foo ->save ()->calc ()->getValue ();
// bad
(new Foo (data: 'bar ' ))->calc ()
// good
(new Foo ())->setData (data: 'bar ' )->calc ()
// bad
$ str = trim ($ str );
$ str = mb_strtoupper ($ str );
$ str = str_replace (' ' , '' , $ str );
// good
$ str = Str::of ($ str )->trim ()->upper ()->replace (' ' , '' )->toString ();
prevent duplicate database calls
// bad
$ model = Model::orderBy ('id ' );
$ count = $ model ->count (); // counts in db
$ model = $ model ->get ();
// good
$ model = Model::orderBy ('id ' );
$ model = $ model ->get ();
$ count = $ model ->count (); // counts in collection
// bad
echo $ msg . PHP_EOL ; // with newline in command
echo $ msg . ' ' ; // without newline in command
echo str_pad ($ msg , 99 , ' ' , STR_PAD_RIGHT ) . "\r" ; // with overwrite in command
(new \Symfony \Component \Console \Output \ConsoleOutput ())->writeln ($ msg ); // with newline in controller
(new \Symfony \Component \Console \Output \ConsoleOutput ())->info ($ msg ); // without newline in controller
(new \Symfony \Component \Console \Output \ConsoleOutput ())->write (str_pad ($ msg , 99 , ' ' , STR_PAD_RIGHT ) . "\r" ); // with overwrite in controller
// good
$ this ->info ($ msg ); // with newline in command
$ this ->output ->write ($ msg ); // without newline in command
$ this ->output ->write (str_pad ($ msg , 99 , ' ' , STR_PAD_RIGHT ) . "\r" ); // with overwrite in command
ConsoleOutputHelper::info ($ msg ); // with newline in controller
ConsoleOutputHelper::write ($ msg ); // without newline in controller
ConsoleOutputHelper::overwrite ($ msg ); // with overwrite in controller
// bad
function log ($ msg ) : void {
if (. .. ) {
echo $ msg ;
}
}
// good
function log ($ msg ) {
if (!. .. ) {
return ;
}
echo $ msg ;
}
prevent too much nesting (2)
// bad
if ($ a === true ) {
if ($ b === true ) {
if ($ c === true ) {
return 42 ;
}
}
}
return 7 ;
// good
if ($ a !== true ) { return 7 ; }
if ($ b !== true ) { return 7 ; }
if ($ c !== true ) { return 7 ; }
return 42 ;
// bad
if ($ foo ) {
}
elseif ($ bar ) {
}
// good
if ($ foo ) {
}
if ($ bar && !$ foo ) {
}
prevent too much function calls
// bad
foo ($ value , true , true , false , 42 )
// good
foo (value: $ value , save: true , cache: true , force: false , amount: 42 )
$ foo ->withSave ()->withCache ()->withoutForce ()->withAmount (42 )->calculate ($ value );
outsource long code parts
// bad
$ isSpecial = false ;
if ($ foo ) { $ isSpecial = true ; }
if ($ bar && !$ foo ) { $ isSpecial = false ; }
if ($ isSpecial ) {
//...
}
// good
function isSpecial () {
if ($ foo ) { return true ; }
if ($ bar ) { return false ; }
return false ;
}
if ($ this ->isSpecial ()) {
//...
}
use carbon instead of strings
// bad
date ('Y ' )-10
// good
Carbon::now ()->subYears (10 )->year
// bad
class NonDerivativeClass {}
// good
final class NonDerivativeClass {}
// bad
public static function foo ($ a , $ b = null , $ c = false ) {}
// good
protected function foo (int $ a , ?string $ b = null , bool $ c = false ): void {}
// bad
public $ foo ;
// good
protected ?int $ foo ;
use jobs to dispatch long running tasks
// bad
longRunningTask ();
// good
CalculateLongRunningTaskQueue::dispatch ();
multiline comments
// bad
// this is
// a multiline
// comment
// bad
/*
this is
a multiline
comment
*/
// good
/**
* this is
* a multiline
* comment
*/
comment functions (e.g. using copilot)
// bad
public function foo (): ?float {}
// good
/**
* Calculate the value for our example.
*
* @return float|null
*/
public function foo (): ?float {}
phpunit tests: the expected value should come first as an argument
// bad
$ this ->assertSame ($ foo , 'bar ' );
// good
$ this ->assertSame ('bar ' , $ foo );
always write tests (tdd), use factories
// good
Model::factory ()->count (10 )->create (['birth_date ' => '1980-01-01 ' ]);
Model::factory ()->count (10 )->create (['birth_date ' => '1990-01-01 ' ]);
Model::factory ()->count (10 )->create (['birth_date ' => '2000-01-01 ' ]);
$ this ->assertEquals (30 , Model::whereBornAfter (1970 )->count ());
$ this ->assertEquals (30 , Model::whereBornAfter (1980 )->count ());
$ this ->assertEquals (20 , Model::whereBornAfter (1990 )->count ());
always catch the most specific exception
// bad
catch (\Throwable $ e ) {
print_r ($ e ->getMessage ());
}
// good
catch (\PDOException $ e ) {
print_r ($ e ->getMessage ());
}
// bad
class Environment
{
static function get ($ str ) {
if ( $ str === 'Entwicklung ' ) { return 'dev ' ; }
if ( $ str === 'Test ' ) { return 'stage ' ; }
if ( $ str === 'Produktion ' ) { return 'prod ' ; }
}
}
echo Environment::get ('Produktion ' ); // "prod"
// good
enum Environment: string
{
case Entwicklung = 'dev ' ;
case Test = 'stage ' ;
case Produktion = 'prod ' ;
}
echo Environment::Produktion->value ; // "prod"