Best way to give a variable a default value (simulate Perl ||, ||= )

PhpPerlOperatorsDefault Value

Php Problem Overview


I love doing this sort of thing in Perl: $foo = $bar || $baz to assign $baz to $foo if $bar is empty or undefined. You also have $foo ||= $bletch which will only assign $bletch to $foo if $foo is not defined or empty.

The ternary operator in this situation is tedious and tiresome. Surely there's a simple, elegant method available in PHP?

Or is the only answer a custom function that uses isset()?

Php Solutions


Solution 1 - Php

PHP 5.3 has a shorthand ?: operator:

$foo = $bar ?: $baz;

Which assigns $bar if it's not an empty value (I don't know how this would be different in PHP from Perl), otherwise $baz, and is the same as this in Perl and older versions of PHP:

$foo = $bar ? $bar : $baz;

But PHP does not have a compound assignment operator for this (that is, no equivalent of Perl's ||=).

Also, PHP will make noise if $bar isn't set unless you turn notices off. There is also a semantic difference between isset() and empty(). The former returns false if the variable doesn't exist, or is set to NULL. The latter returns true if it doesn't exist, or is set to 0, '', false or NULL.

Solution 2 - Php

In PHP 7 we finally have a way to do this elegantly. It is called the Null coalescing operator. You can use it like this:

$name = $_GET['name'] ?? 'john doe';

This is equivalent to

$name = isset($_GET['name']) ? $_GET['name']:'john doe';

Solution 3 - Php

Thanks for all the great answers!

For anyone else coming here for a possible alternative, here are some functions that help take the tedium out of this sort of thing.

function set_if_defined(&$var, $test){
	if (isset($test)){
		$var = $test;
		return true;
	} else {
		return false;
	}
}
	
function set_unless_defined(&$var, $default_var){
	if (! isset($var)){
		$var = $default_var;
		return true;
	} else {
		return false;
	}
}
	
function select_defined(){
	$l = func_num_args();
	$a = func_get_args();
	for ($i=0; $i<$l; $i++){
		if ($a[$i]) return $a[$i];
	}
}

Examples:

// $foo ||= $bar;
set_unless_defined($foo, $bar);

//$foo = $baz || $bletch
$foo = select_defined($baz, $bletch);

I'm sure these can be improved upon.

Solution 4 - Php

A common idiom to stay compatible with older PHP versions is:

 $var = $bool   or   $var = "default";
 // If I use it, then only with excessive spaces for clarity.

This works for values that can be evaluated in boolean context. The advantage here is that it also gives you said debug e_notice should the variable be undefined.

Solution 5 - Php

In PHP earlier than 7.*, one may use ?: for an undefined variable having errors locally suppressed with an @:

$foo = @$bar ?: $baz;

Solution 6 - Php

this is another good format for the isset case

isset($foo) || $foo= $bar;

another simple way and will give you more control as you can add more conditions and assign to another variable in the same time

$foo = (isset($oData['foo']))?$bar['foo']:'default value';

Solution 7 - Php

A possible solution: defaultFor( )

Unless we have a factory solution (which is indeed very annoying), I'd recommend the following little helper. It does the job in most cases:


    function defaultFor(&$x,$default=null) {
        if(!isset($x)) $x = $default;
    }

    //--------------------  Now you can do this: --------------

    defaultFor($a,"Jack");  // if $a is not set, it will be "Jack"
    defaultFor($x);         // no more notices for $x but keep it !isset

I hope this is close to what you wanted to achieve. It will not give you any notices if you use it with a nonexistent variable, and it's quite convenient. Surely it has a drawback: the default value always gets calculated beforehand so don't use it with anything heavy as a second parameter, like a file_get_contents or something. In those cases, you're better off with isseting.

Solution 8 - Php

I think in general doing something like this:

$foo = $bar || $baz;

is a bad idea, unless $bar and $baz are both booleans. If they aren't:

$bar = 10;
$baz = 11;

then the question becomes: what determines if something is true or false? Most people would probably expect zero to be false, and everything else to be true. But with some languages (for example Ruby), only false and nil are false, which means both 0 and 1 would be true. Because of this cross language ambiguity, I think it best to be explicit in these cases:

if ($bar !== 0) {
   $foo = $bar;
} else {
   $foo = $baz;
}

or:

$foo = $bar !== 0 ? $bar : $baz;

Attributions

All content for this solution is sourced from the original question on Stackoverflow.

The content on this page is licensed under the Attribution-ShareAlike 4.0 International (CC BY-SA 4.0) license.

Content TypeOriginal AuthorOriginal Content on Stackoverflow
QuestionTom AugerView Question on Stackoverflow
Solution 1 - PhpBoltClockView Answer on Stackoverflow
Solution 2 - PhpjpschroederView Answer on Stackoverflow
Solution 3 - PhpTom AugerView Answer on Stackoverflow
Solution 4 - PhpmarioView Answer on Stackoverflow
Solution 5 - PhpAndreyS ScherbakovView Answer on Stackoverflow
Solution 6 - PhpBassem ShahinView Answer on Stackoverflow
Solution 7 - PhpdkellnerView Answer on Stackoverflow
Solution 8 - PhpZomboView Answer on Stackoverflow