Каков наилучший способ определить, является ли строка результатом/результатом функции serialize()?
Проверьте, не сериализована ли строка?
Ответ 1
Я бы сказал, попробуйте unserialize
it; -)
Указание руководства:
Если переданная строка не является unserializeable, возвращается FALSE и Выдается E_NOTICE.
Итак, вы должны проверить, является ли возвращаемое значение false
или нет (с помощью ===
или !==
), чтобы не было проблем с 0
или null
или что-либо, что равно false
, я бы сказал).
Остерегайтесь уведомления: вам может понадобиться/нужно использовать @operator.
Например:
$str = 'hjkl';
$data = @unserialize($str);
if ($data !== false) {
echo "ok";
} else {
echo "not ok";
}
Вы получите:
not ok
EDIT: О, и как @Peter сказал (спасибо ему!), У вас могут возникнуть проблемы, если вы пытаетесь неэриализовать представление логического false: - (
Итак, проверка того, что ваша сериализованная строка не равна "b:0;
", также может быть полезна; что-то вроде этого должно сделать трюк, я полагаю:
$data = @unserialize($str);
if ($str === 'b:0;' || $data !== false) {
echo "ok";
} else {
echo "not ok";
}
проверка того, что особый случай перед попыткой несериализации будет оптимизацией, но, вероятно, не так уж и полезен, если вы не часто имеете ложное сериализованное значение.
Ответ 2
Я не писал этот код, это из WordPress. Думал, что я включу его для любого заинтересованного, это может быть слишком много, но оно работает:)
<?php
function is_serialized( $data ) {
// if it isn't a string, it isn't serialized
if ( !is_string( $data ) )
return false;
$data = trim( $data );
if ( 'N;' == $data )
return true;
if ( !preg_match( '/^([adObis]):/', $data, $badions ) )
return false;
switch ( $badions[1] ) {
case 'a' :
case 'O' :
case 's' :
if ( preg_match( "/^{$badions[1]}:[0-9]+:.*[;}]\$/s", $data ) )
return true;
break;
case 'b' :
case 'i' :
case 'd' :
if ( preg_match( "/^{$badions[1]}:[0-9.E-]+;\$/", $data ) )
return true;
break;
}
return false;
}
Ответ 3
Оптимизация ответа Pascal MARTIN
/**
* Check if a string is serialized
* @param string $string
*/
public static function is_serial($string) {
return (@unserialize($string) !== false);
}
Ответ 4
Если $ string является сериализованным значением false
, т.е. $string = 'b:0;'
Функция SoN9ne возвращает false
, это неправильно
поэтому функция будет
/**
* Check if a string is serialized
*
* @param string $string
*
* @return bool
*/
function is_serialized_string($string)
{
return ($string == 'b:0;' || @unserialize($string) !== false);
}
Ответ 5
Несмотря на превосходный ответ Паскаля МАРТИНА, мне было любопытно, можете ли вы подойти к этому по-другому, поэтому я сделал это как умственное упражнение.
<?php
ini_set( 'display_errors', 1 );
ini_set( 'track_errors', 1 );
error_reporting( E_ALL );
$valueToUnserialize = serialize( false );
//$valueToUnserialize = "a"; # uncomment this for another test
$unserialized = @unserialize( $valueToUnserialize );
if ( FALSE === $unserialized && isset( $php_errormsg ) && strpos( $php_errormsg, 'unserialize' ) !== FALSE )
{
echo 'Value could not be unserialized<br>';
echo $valueToUnserialize;
} else {
echo 'Value was unserialized!<br>';
var_dump( $unserialized );
}
И это действительно работает. Единственное предостережение в том, что он, скорее всего, сломается, если у вас есть зарегистрированный обработчик ошибок из-за того, как $php_errormsg работает.
Ответ 6
$data = @unserialize($str);
if($data !== false || $str === 'b:0;')
echo 'ok';
else
echo "not ok";
Правильно обрабатывает случай serialize(false)
.:)
Ответ 7
встраивается в функцию
function isSerialized($value)
{
return preg_match('^([adObis]:|N;)^', $value);
}
Ответ 8
Существует решение WordPress: (подробности здесь)
function is_serialized($data, $strict = true)
{
// if it isn't a string, it isn't serialized.
if (!is_string($data)) {
return false;
}
$data = trim($data);
if ('N;' == $data) {
return true;
}
if (strlen($data) < 4) {
return false;
}
if (':' !== $data[1]) {
return false;
}
if ($strict) {
$lastc = substr($data, -1);
if (';' !== $lastc && '}' !== $lastc) {
return false;
}
} else {
$semicolon = strpos($data, ';');
$brace = strpos($data, '}');
// Either ; or } must exist.
if (false === $semicolon && false === $brace)
return false;
// But neither must be in the first X characters.
if (false !== $semicolon && $semicolon < 3)
return false;
if (false !== $brace && $brace < 4)
return false;
}
$token = $data[0];
switch ($token) {
case 's' :
if ($strict) {
if ('"' !== substr($data, -2, 1)) {
return false;
}
} elseif (false === strpos($data, '"')) {
return false;
}
// or else fall through
case 'a' :
case 'O' :
return (bool)preg_match("/^{$token}:[0-9]+:/s", $data);
case 'b' :
case 'i' :
case 'd' :
$end = $strict ? '$' : '';
return (bool)preg_match("/^{$token}:[0-9.E-]+;$end/", $data);
}
return false;
}
Ответ 9
/**
* some people will look down on this little puppy
*/
function isSerialized($s){
if(
stristr($s, '{' ) != false &&
stristr($s, '}' ) != false &&
stristr($s, ';' ) != false &&
stristr($s, ':' ) != false
){
return true;
}else{
return false;
}
}
Ответ 10
Это отлично работает для меня
<?php
function is_serialized($data){
return (is_string($data) && preg_match("#^((N;)|((a|O|s):[0-9]+:.*[;}])|((b|i|d):[0-9.E-]+;))$#um", $data));
}
?>
Ответ 11
Я предпочитаю делать это так:
if (is_array(unserialize($serialized_string))):