Проверить, сериализуема ли переменная

#php #serialization #closures

#php #сериализация #замыкания

Вопрос:

Я ищу элегантный способ проверить, сериализуема ли переменная. Например, array( function() {} ) не удастся сериализовать.

В настоящее время я использую приведенный ниже код, но, похоже, это довольно неоптимальный способ сделать это.

 function isSerializable( $var )
{
    try {
        serialize( $var );
        return TRUE;
    } catch( Exception $e ) {
        return FALSE;
    }
}

var_dump( isSerializable( array() ) );                // bool(true)
var_dump( isSerializable( function() {} ) );          // bool(false)
var_dump( isSerializable( array( function() {} ) ) ); // bool(false)
  

Комментарии:

1. Это довольно хороший способ сделать это 🙂

2. Да, я согласен с Алексом, это выглядит совершенно нормально. Вас беспокоят накладные расходы при попытке сериализации?

3. я думаю, что это лучший способ, и другого способа не существует 🙂

4. Ну, я пытаюсь написать функцию, которая позволит сериализовать обычные сериализуемые переменные замыкания (используя это ). Итак, я пытаюсь обойти дерево объектов в поисках замыкания, которое может существовать (и предотвращающего сериализацию), используя функцию isSerializable для «обхода» переменной. Это может быть очень медленно, если мне придется пытаться выполнить глубокую сериализацию в каждой точке.

5. Почему функция не возвращает сериализованный объект или false? Тогда ваши данные уже сериализованы. Sry не прочитал код, на который вы ссылались.

Ответ №1:

Альтернативой может быть:

 function isSerializable ($value) {
  $return = true;
  $arr = array($value);

  array_walk_recursive($arr, function ($element) use (amp;$return) {
    if (is_object($element) amp;amp; get_class($element) == 'Closure') {
      $return = false;
    }
  });

  return $return;
}
  

Но из комментариев я думаю, что это то, что вы ищете:

 function mySerialize ($value) {
  $arr = array($value);

  array_walk_recursive($arr, function (amp;$element) {

    # do some special stuff (serialize closure) ...
    if (is_object($element) amp;amp; get_class($element) == 'Closure') {
      $serializableClosure = new SerializableClosure($element);
      $element = $serializableClosure->serialize();
    }

  });

  return serialize($arr[0]);
}
  

Комментарии:

1. При этом не проверяется, является ли она ресурсом.

2. Или любой другой объект, который не сериализуем.

Ответ №2:

Поздний ответ, но…

Согласно документации PHP, переменные типа Resource не могут быть сериализованы. Однако, по крайней мере в PHP 5.4, попытка сериализовать ресурс не вызовет никакой ошибки.

Я думаю, что лучшим подходом было бы протестировать замыкания и ресурсы без try / catch:

 $resource = fopen('composer.json', 'r');
$closure = function() {
    return 'bla';
};
$string = 'string';

function isSerializable($var)
{
    if (is_resource($var)) {
        return false;
    } else if ($var instanceof Closure) {
        return false;
    } else {
        return true;
    }
}

var_dump(isSerializable($resource));
var_dump(isSerializable($closure));
var_dump(isSerializable($string));
  

Выводит:

 boolean false 
boolean false
boolean true
  

Комментарии:

1. Понятия не имею, почему кто-то -1 это… но в любом случае. Вы также должны проверить, является ли она an object , и если да, то реализует ли она Serializable .

2. Это не выполняется, если $var является массивом, который содержит несериализуемые значения.

Ответ №3:

Это то, что я использую. Здесь объединено несколько предложений, плюс проверка instaneof ArrayAccess того, какая из них должна быть сериализуемой.

 if (!function_exists('is_iterable')) {
    function is_iterable($var) {
        return is_array($var) || (is_object($var) amp;amp; ($var instanceof Traversable));
    }
}

function is_serializable($var, $iterate=true) {
    if (is_resource($var)) {
        return false;
    } else if (is_object($var)) {
        if ($var instanceof Closure) {
            return false;
        } else if (!$var instanceof Serializable amp;amp; !$var instanceof ArrayAccess) {
            return false;
        }
    }

    if ($iterate amp;amp; is_iterable($var)) {
        foreach ($var as $key => $value) {
            if (!is_serializable($value, true)) {
                return false;
            }
        }
    }

    return true;
}