Laravel 避免 Trying to get property of non-object 错误的六种方法 [新增第六种 data_get]

在使用链式操作的时候,例如:

return $user->avatar->url;

如果 $user->avatarnull,就会引起 (E_ERROR) Trying to get property 'url' of non-object 错误。

1. 常规方法是使用 isset 加以判断:

if(isset($user->avatar->url))
    return $user->avatar->url;
else
    return 'defaultUrl';

如果在 blade 模板的 echo 中,可以使用:

{{ $user->avatar->url or 'defaultUrl' }}

上述代码会被 Blade 引擎解析为:

echo e(isset($user->avatar->url) ? $user->avatar->url : 'defaultUrl');

Laravel 5.7 已经取消了这个特性。详见:https://github.com/laravel/framework/pull/... 。感谢 [[[[[[@jltxwesley](https://learnku.com/users/28596)](https://learnku.com/users/28596)](https://learnku.com/users/28596)](https://learnku.com/users/28596)](https://learnku.com/users/28596)](https://learnku.com/users/28596) 提醒。

2. PHP7 可以使用 ?? (NULL 合并操作符) :

// 如果 $user->avatar->url 为 null, 返回 'defaultUrl'
return $user->avatar->url ?? 'defaultUrl';

3. Laravel 5.5 及以上可以使用 optional 辅助函数:

/**
 * 如果给定的对象是 null , 那么属性和方法会简单地返回 null 而不是产生一个错误:
 */
return optional($user->avatar)->url;

详见 辅助函数《Laravel 5.5 中文文档》

Laravel 5.7 中,optional 函数还可以接受 匿名函数 作为第二个参数:

/**
 * 如果第一个参数不为 null, 则调用闭包
 * 详见 https://laravel\com/docs/5.7/helpers#method-optional
 */
return optional(User::find($id), function ($user) {
    return new DummyUser;
});

4. 使用 object_get 辅助函数

return object_get($user->avatar, 'url', 'default');

这个函数原意是用来已 . 语法来获取对象中的属性,例如:

return object_get($user, 'avatar.url', 'default');

也可以达到避免 non-object 错误的效果。

if (! function_exists('object_get')) {
    /**
     * Get an item from an object using "dot" notation.
     *
     * @param  object  $object
     * @param  string  $key
     * @param  mixed   $default
     * @return mixed
     */
    function object_get($object, $key, $default = null)
    {
        if (is_null($key) || trim($key) == '') {
            return $object;
        }

        foreach (explode('.', $key) as $segment) {
            if (! is_object($object) || ! isset($object->{$segment})) {
                return value($default);
            }

            $object = $object->{$segment};
        }

        return $object;
    }
}

详见 https://github.com/laravel/framework/blob/...

感谢 [[[[[@lovecn](https://learnku.com/users/87)](https://learnku.com/users/87)](https://learnku.com/users/87)](https://learnku.com/users/87)](https://learnku.com/users/87) 提供姿势!

5. 使用 data_get 辅助函数

return data_get($user, 'avatar.url', 'default');

return data_get($user, ['avatar', 'url'], 'default');

. 语法来获取对象属性或数组元素。

if (! function_exists('data_get')) {
    /**
     * Get an item from an array or object using "dot" notation.
     *
     * @param  mixed   $target
     * @param  string|array  $key
     * @param  mixed   $default
     * @return mixed
     */
    function data_get($target, $key, $default = null)
    {
        if (is_null($key)) {
            return $target;
        }
        $key = is_array($key) ? $key : explode('.', $key);
        while (! is_null($segment = array_shift($key))) {
            if ($segment === '*') {
                if ($target instanceof Collection) {
                    $target = $target->all();
                } elseif (! is_array($target)) {
                    return value($default);
                }
                $result = [];
                foreach ($target as $item) {
                    $result[] = data_get($item, $key);
                }
                return in_array('*', $key) ? Arr::collapse($result) : $result;
            }
            if (Arr::accessible($target) && Arr::exists($target, $segment)) {
                $target = $target[$segment];
            } elseif (is_object($target) && isset($target->{$segment})) {
                $target = $target->{$segment};
            } else {
                return value($default);
            }
        }
        return $target;
    }
}

详见 https://github.com/laravel/framework/blob/...

感谢 [@Hachiko](https://learnku.com/users/22249) 提供姿势!

6.除此之外,还可以使用 Null Object Pattern(空对象模式):

《點燈坊:如何實現 Null Object Pattern ?》

感谢群里大佬 @盒子 和 @Outshine 提供的姿势。:kissing: :kissing: :kissing:

本作品采用《CC 协议》,转载必须注明作者和本文链接
原创。 所有 Laravel 文章均已收录至 Github laravel-tips 项目。
本帖由系统于 5年前 自动加精
Ίκαρος
《L04 微信小程序从零到发布》
从小程序个人账户申请开始,带你一步步进行开发一个微信小程序,直到提交微信控制台上线发布。
《G01 Go 实战入门》
从零开始带你一步步开发一个 Go 博客项目,让你在最短的时间内学会使用 Go 进行编码。项目结构很大程度上参考了 Laravel。
讨论数量: 19
jltxwesley

友情提醒:

  1. 因为 PHP7 的 NULL 合并操作符,Laravel 5.7 已经移除了 Blade “or” 操作符https://github.com/laravel/framework/pull/...
  2. NULL 合并操作符还可以嵌套使用,比如:return $user->avatar->url ?? $user->defaulUrl ?? 'otherUrl';
5年前 评论
阿麦

很实用

5年前 评论

第四种涨姿势了

5年前 评论
jltxwesley

友情提醒:

  1. 因为 PHP7 的 NULL 合并操作符,Laravel 5.7 已经移除了 Blade “or” 操作符https://github.com/laravel/framework/pull/...
  2. NULL 合并操作符还可以嵌套使用,比如:return $user->avatar->url ?? $user->defaulUrl ?? 'otherUrl';
5年前 评论
Ίκαρος

@jltxwesley Thanks♪(・ω・)ノ,我修改下~

5年前 评论

點燈坊,那种中文名的测试方法,很骚。。。。

5年前 评论
Ίκαρος

@largezhou :laughing: 简单暴力,不需要写注释了都

5年前 评论

object_get($user->avatar,'url','default');

5年前 评论
Ίκαρος

@lovecn 谢谢! :thumbsup: 明天补上它!

5年前 评论

js里有没有类似方法呢?

5年前 评论

花括号花括号。。。

5年前 评论
Ίκαρος

@Nick :wink: 要多打四个字符,手指不同意啊哈哈哈~

5年前 评论

data_get 辅助函数不更好,数组和对象通用,源代码参见:https://github.com/laravel/framework/blob/...

5年前 评论
Ίκαρος

@Hachiko 给力。。。一会儿加上。。。

5年前 评论
Destiny

optional() 用得比较多。

5年前 评论

用withDefault更好

5年前 评论

?? 和 optional()用的比较多。

每回看同事代码写if (isset($data) && !empty($data))的时候,我就想改成 if ($data ?? [])if ($data ?? '')
反正用php7以后,就很少用isset函数和empty函数了

5年前 评论

讨论应以学习和精进为目的。请勿发布不友善或者负能量的内容,与人为善,比聪明更重要!