> 백엔드 개발 > PHP 튜토리얼 > php中unserialize返回false的解决方法_PHP教程

php中unserialize返回false的解决方法_PHP教程

WBOY
풀어 주다: 2016-07-13 10:18:37
원래의
911명이 탐색했습니다.

php中unserialize返回false的解决方法

 php 提供serialize(序列化) 与unserialize(反序列化)方法。

使用serialize序列化后,再使用unserialize反序列化就可以获取原来的数据。

先来看看如下程序实例:

1

2

3

4

5

6

7

8

9

10

11

12

13

$arr = array( 

  'name' => 'fdipzone', 

  'gender' => 'male' 

); 

   

$str = serialize($arr); //序列化 

echo 'serialize str:'.$str."\r\n\r\n"; 

   

$content = unserialize($str); // 反序列化 

echo "unserialize str:\r\n"; 

var_dump($content); 

?>

输出:

1

2

3

4

5

6

7

8

9

serialize str:a:2:{s:4:"name";s:8:"fdipzone";s:6:"gender";s:4:"male";} 

   

unserialize str: 

array(2) { 

 ["name"]=> 

 string(8) "fdipzone" 

 ["gender"]=> 

 string(4) "male" 

}

但下面这个例子反序列化会返回false

1

2

3

4

$str = 'a:9:{s:4:"time";i:1405306402;s:4:"name";s:6:"新晨";s:5:"url";s:1:"-";s:4:"word";s:1:"-";s:5:"rpage";s:29:"http://www.baidu.com/test.html";s:5:"cpage";s:1:"-";s:2:"ip";s:15:"117.151.180.150";s:7:"ip_city";s:31:"中国北京市 北京市移动";s:4:"miao";s:1:"5";}'; 

var_dump(unserialize($str)); // bool(false) 

?>

检查序列化后的字符串,发现出问题是在两处地方:

s:5:"url"
s:29:"http://www.baidu.com/test.html"
这两处应为
s:3:"url"
s:30:"http://www.baidu.com/test.html"

出现这种问题的原因是序列化数据时的编码与反序列化时的编码不一致导致,例如数据库是latin1和UTF-8字符长度不一样。
另外有可能出问题的还有单双引号,ascii字符"\0"被解析为 '\0',\0在C中是字符串的结束符等于chr(0),错误解析后算了2个字符。
\r在计算长度时也会出问题。

解决方法如下:

1

2

3

4

5

6

7

8

9

10

11

12

13

// utf8 

function mb_unserialize($serial_str) { 

  $serial_str= preg_replace('!s:(\d+):"(.*?)";!se', "'s:'.strlen('$2').':\"$2\";'", $serial_str ); 

  $serial_str= str_replace("\r", "", $serial_str); 

  return unserialize($serial_str); 

   

// ascii 

function asc_unserialize($serial_str) { 

  $serial_str = preg_replace('!s:(\d+):"(.*?)";!se', '"s:".strlen("$2").":\"$2\";"', $serial_str ); 

  $serial_str= str_replace("\r", "", $serial_str); 

  return unserialize($serial_str); 

}

例子:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

echo ''; 

   

// utf8 

function mb_unserialize($serial_str) { 

  $serial_str= preg_replace('!s:(\d+):"(.*?)";!se', "'s:'.strlen('$2').':\"$2\";'", $serial_str ); 

  $serial_str= str_replace("\r", "", $serial_str); 

  return unserialize($serial_str); 

   

$str = 'a:9:{s:4:"time";i:1405306402;s:4:"name";s:6:"新晨";s:5:"url";s:1:"-";s:4:"word";s:1:"-";s:5:"rpage";s:29:"http://www.baidu.com/test.html";s:5:"cpage";s:1:"-";s:2:"ip";s:15:"117.151.180.150";s:7:"ip_city";s:31:"中国北京市 北京市移动";s:4:"miao";s:1:"5";}'; 

   

var_dump(unserialize($str));  // false 

   

var_dump(mb_unserialize($str)); // 正确

使用处理过单双引号,过滤\r的mb_unserialize方法就能成功反序列化了。

使用unserialize:

bool(false) 
 
使用mb_unserialize

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

array(9) { 

 ["time"]=> 

 int(1405306402) 

 ["name"]=> 

 string(6) "新晨" 

 ["url"]=> 

 string(1) "-" 

 ["word"]=> 

 string(1) "-" 

 ["rpage"]=> 

 string(30) "http://www.baidu.com/test.html" 

 ["cpage"]=> 

 string(1) "-" 

 ["ip"]=> 

 string(15) "117.151.180.150" 

 ["ip_city"]=> 

 string(31) "中国北京市 北京市移动" 

 ["miao"]=> 

 string(1) "5" 

}

www.bkjia.comtruehttp://www.bkjia.com/PHPjc/882699.htmlTechArticlephp中unserialize返回false的解决方法 php 提供serialize(序列化) 与unserialize(反序列化)方法。 使用serialize序列化后,再使用unserialize反序列化就可以...
원천:php.cn
본 웹사이트의 성명
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.
인기 튜토리얼
더>
최신 다운로드
더>
웹 효과
웹사이트 소스 코드
웹사이트 자료
프론트엔드 템플릿