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

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

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/23532 。感谢 @jltxwesley 提醒。

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;

详见 https://laravel-china.org/docs/laravel/5.7/helpers/1320#method-optional

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/master/src/Illuminate/Support/helpers.php#L673

感谢 @lovecn 提供姿势!

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/master/src/Illuminate/Support/helpers.php#L450

感谢 @Hachiko 提供姿势!

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

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

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

原创。 所有 Laravel 文章均已收录至 laravel-tips 项目。

本帖由系统于 1个月前 自动加精
《L02 从零构建论坛系统》
以构建论坛项目 LaraBBS 为线索,展开对 Laravel 框架的全面学习。应用程序架构思路贴近 Laravel 框架的设计哲学。
《L03 构架 API 服务器》
你将学到如 RESTFul 设计风格、PostMan 的使用、OAuth 流程,JWT 概念及使用 和 API 开发相关的进阶知识。
讨论数量: 17
GhostCoder

很实用

1个月前
BradStevens

第四种涨姿势了

1个月前
jltxwesley

友情提醒:

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

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

1个月前

刘神威武

1个月前
largezhou

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

1个月前
Ίκαρος

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

1个月前

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

1个月前
Ίκαρος

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

1个月前
Insua

js里有没有类似方法呢?

1个月前
Nick

花括号花括号。。。

1个月前
Ίκαρος

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

1个月前

data_get 辅助函数不更好,数组和对象通用,源代码参见:https://github.com/laravel/framework/blob/master/src/Illuminate/Support/helpers.php#L450

1个月前
Ίκαρος

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

1个月前

實用

1个月前
Destiny

optional() 用得比较多。

1个月前

用withDefault更好

4天前

  • 请注意单词拼写,以及中英文排版,参考此页
  • 支持 Markdown 格式, **粗体**、~~删除线~~、`单行代码`, 更多语法请见这里 Markdown 语法
  • 支持表情,使用方法请见 Emoji 自动补全来咯,可用的 Emoji 请见 :metal: :point_right: Emoji 列表 :star: :sparkles:
  • 上传图片, 支持拖拽和剪切板黏贴上传, 格式限制 - jpg, png, gif
  • 发布框支持本地存储功能,会在内容变更时保存,「提交」按钮点击时清空
  请勿发布不友善或者负能量的内容。与人为善,比聪明更重要!