Я написал простой класс PHP, который реализует интерфейс ArrayAccess:
class MyArray implements ArrayAccess
{
public $value;
public function __construct($value = null)
{
$this->value = $value;
}
public function &offsetGet($offset)
{
var_dump(__METHOD__);
if (!isset($this->value[$offset])) {
throw new Exception('Undefined index: ' . $offset);
}
return $this->value[$offset];
}
public function offsetExists($offset)
{
var_dump(__METHOD__);
return isset($this->value[$offset]);
}
public function offsetSet($offset, $value)
{
var_dump(__METHOD__);
$this->value[$offset] = $value;
}
public function offsetUnset($offset)
{
var_dump(__METHOD__);
$this->value[$offset] = null;
}
}
Он работает нормально в PHP 7, но проблема в PHP 5.6 и HHVM.
Если я вызываю функцию isset()
по неопределенному индексу PHP вызовет offsetGet()
вместо offsetExists()
что приведет к Undefined index
уведомление.
В PHP 7 это вызывает offsetGet()
только если offsetExists()
возвращается true
, так что нет ошибки.
Я думаю, что это связано с PHP ошибка 62059.
Код доступен в 3V4L, так что вы можете увидеть, что не так. Я добавил еще несколько вызовов отладки и выбросил исключение, если индекс не определен, потому что уведомления не отображаются в 3V4L:
https://3v4l.org/7C2Fs
Не должно быть никакого уведомления, иначе тесты PHPUnit не пройдут.
Как я могу исправить эту ошибку?
Похоже, что это ошибка PHP в старых версиях PHP и HHVM. Поскольку PHP 5.6 больше не поддерживается, эта ошибка не будет исправлена.
Быстрое решение состоит в том, чтобы добавить дополнительную проверку в методе offsetGet()
и вернуться null
если индекс не определен:
class MyArray implements ArrayAccess
{
public $value;
public function __construct($value = null)
{
$this->value = $value;
}
public function &offsetGet($offset)
{
if (!isset($this->value[$offset])) {
$this->value[$offset] = null;
}
return $this->value[$offset];
}
public function offsetExists($offset)
{
return isset($this->value[$offset]);
}
public function offsetSet($offset, $value)
{
$this->value[$offset] = $value;
}
public function offsetUnset($offset)
{
$this->value[$offset] = null;
}
}
Смотрите код на 3V4L а также zerkmsкомментарии (первый, второй, в третьих).
Я не уверен, что понял ваш вопрос, но, возможно, вы могли бы попробовать
public function __construct($value =[]){
$this->value = $value;
}
вместо:
public function __construct($value = null){
$this->value = $value;
}