震惊 PHP empty 函数判断结果为空,但实际值却为非空

最近我在一个项目中使用 empty 时获取到了一些意料之外的结果。下面是我处理后的调试记录,在这里与你分享了。

var_dump(
    $person->firstName,
    empty($person->firstName)
);

它的结果是:

string(5) "Freek"
bool(true)

结果出人意料。为什么变量的值为字符串,但同时会是空值呢?让我们在 $person->firstName 变量上尝试使用其它一些函数来进行判断吧:

var_dump(
    $person->firstName,
    empty($person->firstName),
    isset($person->firstName),
    is_null($person->firstName)
);

以上结果为:

string(5) "Freek"
bool(true) // empty
bool(true) // isset
bool(false) // is_null

译者注:这边的结果可能存在问题 isset 的结果同样为 false,可以到 这里 去运行下查看结果。

issetis_null 函数执行结果符合预期判断,唯独 empty 函数返回了错误结果。

这里让我们来看看 person 类的实现代码吧:

class person
{
    protected $attributes = [];

    public function __construct(array $attributes)
    {
        $this->attributes = $attributes;
    }

    public function __get($name)
    {
        return $this->attributes[$name] ?? null;
    }
}

从上述代码我们可以看到 Person 对象的成员变量是通过 __get 魔术方法从 $attributes 数组中检索出来的。

当将变量传入一个普通函数时,$person->firstName 会先进行取值处理,然后再将获取到的结果作为参数传入函数内。

但是 empty 不是一个函数,而是一种数据结构。所以当将 $person->firstName 传入 empty 时,并不会先进行取值处理。而是会先判断 $person 对象成员变量 firstName 的内容,由于这个变量并未真实存在,所以返回 false

在正中应用场景下,如果你希望 empty 函数能够正常处理变量,我们需要在类中实现 __isset 魔术方法。

class Person
{
    protected $attributes = [];

    public function __construct(array $attributes)
    {
        $this->attributes = $attributes;
    }

    public function __get($name)
    {
        return $this->attributes[$name] ?? null;
    }

    public function __isset($name)
    {
        $attribute = $this->$name;

        return !empty($attribute);
    }
}

这是当 empty 进行控制判断时,会使用这个魔术方法来判断最终的结果。

再让我们看看输出结果:

var_dump(
   $person->firstName, 
   empty($person->firstName)
);

新的检测结果:

string(5) "Freek"
bool(false)

完美!

原文:When empty is not empty

本作品采用《CC 协议》,转载必须注明作者和本文链接
本帖由系统于 1年前 自动加精
liuqing_hu
《L02 从零构建论坛系统》
以构建论坛项目 LaraBBS 为线索,展开对 Laravel 框架的全面学习。应用程序架构思路贴近 Laravel 框架的设计哲学。
《L04 微信小程序从零到发布》
从小程序个人账户申请开始,带你一步步进行开发一个微信小程序,直到提交微信控制台上线发布。
讨论数量: 16

UC的人什么时候也写PHP了?

1年前 评论
liuqing_hu

@我执 为了打下夯实的基础,9 年义务教务我上了 12 年 :)

1年前 评论

UC的人什么时候也写PHP了?

1年前 评论
白俊遥

亲;你这篇文章已经通过我们 UC 震惊部的审核;请携带个人简历前往公司报名;

1年前 评论
liuqing_hu

@白俊遥 :ghost:

1年前 评论
liuqing_hu

@XiaohuiLam 调皮

1年前 评论

6*6 ,标题满分

1年前 评论

居然没有举报

1年前 评论

empty 不敢乱用

$a = 0;
$b = '';
$c = '0';
$d = NUll;
dd(empty($a), empty($b), empty($c), empty($d), empty($e));

//true
//true
//true
//true
//true
1年前 评论

@liuqing_hu 同样是九年义务教育,为何你就这么秀?

1年前 评论
liuqing_hu

@我执 为了打下夯实的基础,9 年义务教务我上了 12 年 :)

1年前 评论

这个empty的争议一直都没停过。
项目中我从来不用empty。都用isset()和count()来代替需要使用empty的场景。

1年前 评论
liuqing_hu

@jobsssss 这也是由于历史原因所致,空值判断一直都是一个比较头痛的问题。
http://us3.php.net/manual/zh/types.compari...

1年前 评论
JeffLi

@XiaohuiLam 没明白。。UC浏览器?

1年前 评论

@jobsssss empty 用好了远比 isset 与 count 更优。如果是 PHP 7 及以上版本,基本可以用 empty 替代 isset,上面说的对象问题也不会存在 。 个人是能用 empty 绝不用 isset

1年前 评论

@FreeMason 能解决问题就好,不必在细节上纠结。

1年前 评论

请勿发布不友善或者负能量的内容。与人为善,比聪明更重要!