Clearing warnings from PHP8.4

Created by: Lester Caine, Last modification: 31 Mar 2025 (07:39 BST)

Moving over to PHP8.4 throws up numerous problems, and while deprecated errors can be hidden, this is counter productive since we have had several points at which earlier deprecated warnings have resulted in those elements being removed alreay.


"Attempt to access property on null"

1. Nullsafe Operator (?->):
The "?->" null safe operator lets you safely access properties or methods of potentially null variables without throwing errors. If the variable is null, the expression returns null.

<code>{{ $blog-&gt;author?-&gt;name }}
</code>

2. optional():
The optional() helper function is available in Laravel starting from version 5.5. It allows you to gracefully access properties or methods of potentially null variables by returning either null or the value if the variable is not null.

<code>{{ optional($blog-&gt;author)-&gt;name }}
</code>

3. Null Coalescing Operator (??):
The null coalescing operator "??" returns the left-hand side if it's not null, otherwise, it returns the right-hand side default value.

<code>{{ $blog-&gt;author-&gt;name ?? &#39;Unknown&#39; }}
</code>

4. withDefault()
The withDefault() method in Laravel's Eloquent ORM creates a related model with default values if it doesn't exist.

<code>use App\Models\Author;

class Blog extends Model {
    public function author() {
        return $this-&gt;belongsTo(Author::class)-&gt;withDefault([
            &#39;name&#39; =&gt; &#39;Anonymous&#39;,
        ]);
    }
}
</code>

In this example, if the blog's author is null when accessed, Laravel will automatically create a new "Author" model with the name set to 'Anonymous'.


PHP Warning Deprecated: Creation of dynamic property

The warning is telling you that there is a property you are trying to set which isn't declared in the class or any of its parents.

When you run this:

<code>class database {

    public $username = &quot;root&quot;;
    public $password = &quot;pasword&quot;;
    public $port = 3306;

    public function __construct($params = array())
    {
        foreach ($params as $key =&gt; $value)
        {
            $this-&gt;{$key} = $value;
        }
    }
}

$db = new database(array(
    &#39;database&#39; =&gt; &#39;db_name&#39;,
    &#39;server&#39; =&gt; &#39;database.internal&#39;,
));</code>

It is roughly equivalent to this:

<code>class database {

    public $username = &quot;root&quot;;
    public $password = &quot;pasword&quot;;
    public $port = 3306;
}

$db = new database;
$db-&gt;database = &#39;db_name&#39;;
$db-&gt;server = &#39;database.internal&#39;;</code>

The warning is that there is no line in the class definition saying that $db->database or $db->server exist.

For now, they will be dynamically created as untyped public properties, but in future, you will need to declare them explicitly:

<code>class database {
    public $database;
    public $server;
    public $username = &quot;root&quot;;
    public $password = &quot;pasword&quot;;
    public $port = 3306;

    public function __construct($params = array())
    {
        foreach ($params as $key =&gt; $value)
        {
            $this-&gt;{$key} = $value;
        }
    }
}

$db = new database(array(
    &#39;database&#39; =&gt; &#39;db_name&#39;,
    &#39;server&#39; =&gt; &#39;database.internal&#39;,
));</code>

In some rare situations, you actually want to say "the properties of this class are whatever I decide to add at run-time"; in that case, you can use the #[AllowDynamicProperties] attribute, like this:

<code>#[\AllowDynamicProperties]
class objectWithWhateverPropertiesIWant {
    public function __construct($params = array())
    {
        foreach ($params as $key =&gt; $value)
        {
            $this-&gt;{$key} = $value;
        }
    }
}</code>