最近遇到的几个PHP小知识点

最近遇到的几个PHP小知识点

author: he xiaodong date: 2019-12-18

好的框架能提升语言生态,也有时候能把让PHP程序员成为框架程序员,而不是更好的后端程序员,例如我这样的。

测试环境为 PHP7.4 Windows 10系统

str_replace 函数

一个小需求是类似单词填空题,从单词中替换到几个字母,变成 _,然后给出一些选项,例如 ax n, au h, ab y这样的,将选项一一对应的替换 _,和原单词比较,如果一致就是正确选项,本以为事情很简单,str_replace 函数支持两个数组之间的替换,例如

$str = 'ab';
str_replace(['a', 'b'], ['c', 'd'], $str);

# output: cd

同位置交换,所以我写了这个

<?php
# 原单词是 laugh

$str = 'l__g_';
$replace = 'au h';
$needle = array_fill(0, substr_count($str, '_') - 1, '_');
$replaceArray = array_filter(str_split($replace));

echo str_replace($needle, $replaceArray, $str);

# output: laaga

本以为很简单就可以输出 laugh,没有多想,实际上会在第一次 _ 替换 a 的情况下,直接将所有的 _ 替换为了 a, 再一次扫描替换已经没有 _ 了,执行完毕,所以最终结果毫无问题,是我想的简单了,换个思路只能一个个替换。

<?php
# 原单词是 laugh

$str = 'l__g_';
$replace = 'au h';

$replaceArray = array_filter(str_split($replace));
for ($i = 0; $i < count($replaceArray); $i++) {
    $position = strpos($str, '_');
    $str = substr_replace($str, $replaceArray[$i], $position, 1);
}

echo $str;
对象与数组的拷贝差异

原想的操作是根据最初的对象数值,例如有 10 个子属性,然后给 10 个属性附上一个新键值,最后得到的并不是想要的,一个 demo 代码 此问题常见于 laravel 框架

<?php
$object = (object)[
        (object)[
            'a' => 1,
            'b' => 2,
        ],
        (object)[
            'a' => 3,
            'b' => 4,
        ]
    ];

    $result = [];
    for ($i = 0 ; $i < 2; $i++) {
        foreach ($object as $key => $value) {
            $value->type = $i;
            $result[] = $value;
        }
    }

    return $result;

// output: [{"a":1,"b":2,"type":1},{"a":3,"b":4,"type":1},{"a":1,"b":2,"type":1},{"a":3,"b":4,"type":1}]

期望输出的前两个的 type 应该是 0,后两个是 1,然而结果全是 1,导致的原因是对象的复制是浅复制,他们实际指向的是统一内存地址,循环的最后把值改成 1 了,所以全都是 type : 1 , 参考 对象复制文档

测试代码:

<?php
$object = (object)[
        'a' => 1,
        'b' => 2,
    ];

    $a = $object;
    $object->a = 'a';
    echo $object->a . PHP_EOL;
    echo $a->a;

# output: a a 

与之相反的数组的复制,数组的拷贝是值传递。PHP 在管理内存方面有一个机制叫写时复制(COW,Copy On Write),保证了变量间复制值不浪费内存:当一个变量的值复制到另一个变量时,PHP 没有为复制值使用更多的内存,相反,它会更新符号表来说明两个变量拥有相同的内存块,所以当执行下面的代码时并没有创建一个新的数组,在修改其中的一个值时,PHP才会将分配所需的内存来复制。

测试代码:

<?php
$array = [
        'a' => 1,
        'b' => 2,
    ];

    $a = $array;
    $array['a'] = 'a';
    echo $array['a'] . PHP_EOL;
    echo $a['a'];

# output: a 1

主要是基础知识不够好,才踩坑,避免下次再犯错。