分类目录归档:PHP

PHP序列化和反序列化

序列化

序列化格式

在PHP中,序列化用于存储或传递 PHP 的值的过程中,同时不丢失其类型和结构。

序列化函数原型如下:

string serialize ( mixed $value )

例子:

class TEST {
    public $TEST_data = 'test_data';
}
class C_TEST extends TEST{
    const SECOND = 60;

    public $data;
    private $pass;

    public function __construct($data, $pass)
    {
        $this->data = $data;
        $this->pass = $pass;
    }

    public function setPass($pass)
    {
        $this->pass = $pass;
    }
}
$test = new C_TEST('test', true);
$number = 100;
$str = 'test';
$bool = true;
$null = NULL;
$arr = array('a' => 1, 'b' => 2);
var_dump(serialize($number));
var_dump(serialize($str));
var_dump(serialize($bool));
var_dump(serialize($null));
var_dump(serialize($arr));
var_dump(serialize($test));

输出结果为:

string(6)"i:100;"
string(11)"s:4:"test";"
string(4)"b:1;"
string(2)"N;"
string(30)"a:2:{s:1:"a";i:1;s:1:"b";i:2;}"
string(87)"O:6:"C_TEST":3{s:4:"data";s:4:"test";s:12:"C_TESTpass";b:1;s:9:"TEST_data";s:9:"test_data";}"

序列化对象时,不会保存常量的值。对于父类中的变量,则会保留。

序列化对于不同类型得到的字符串格式为:

  • String : s:size:value;
  • Integer : i:value;
  • Boolean : b:value;(保存1或0)
  • Null : N;
  • Array : a:size:{key definition;value definition;(repeated per element)}
  • Object : O:strlen(object name):object name:object size:{s:strlen(property name):property name:property definition;(repeated per property)}

对象序列化自定义

在序列化对象的时候,对于对象中的一些敏感属性,我们不需要保存时,要用到一个魔术方法,当调用serialize()函数序列化对象时,该函数会检查类中是否存在魔术方法__sleep()。如果存在,该方法会先被调用,然后才执行序列化操作。可以通过重载这个方法,从而自定义序列化行为。该方法原型如下:

public array __sleep ( void )
  • 该方法返回一个包含对象中所有应被序列化的变量名称的数组
  • 该方法未返回任何内容,则 NULL 被序列化,并产生一个E_NOTICE级别的错误
  • __sleep()不能返回父类的私有成员的名字。这样做会产生一个E_NOTICE级别的错误。这时只能用Serializable接口来替代。
  • 常用于保存那些大对象时的清理工作,避免保存过多冗余数据

例子:

class User{
    const SITE = 'uusama';

    public $username;
    public $nickname;
    private $password;

    public function __construct($username, $nickname, $password)
    {
        $this->username = $username;
        $this->nickname = $nickname;
        $this->password = $password;
    }

    // 重载序列化调用的方法
    public function __sleep()
    {
        // 返回需要序列化的变量名,过滤掉password变量
        return array('username', 'nickname');
    }
}
$user = new User('uusama', 'uu', '123456');
var_dump(serialize($user));

返回结果如下,序列化的时候忽略了 password 字段的值。

string(67) "O:4:"User":2:{s:8:"username";s:6:"uusama";s:8:"nickname";s:2:"uu";}"

序列化对象存储

通过上面的介绍,我们可以把一个复制的对象或者数据序列化成一个序列字符串,保存值的同事还保存了他们的结构。

我们可以把序列化之后的值保存起来,存在文件或者缓存里面。不推荐存在数据库里面,可读性查,而且不便于迁移维护,不便于查询。

$user = new User('uusama', 'uu', '123456');
$ser = serialize($user);
// 保存在本地
file_put_contents('user.ser', $ser);

反序列化

反序列函数

mixed unserialize ( string $str )

unserialize()反序列化函数用于将单一的已序列化的变量转换回 PHP 的值。

  • 如果传递的字符串不可解序列化,则返回 FALSE,并产生一个E_NOTICE
  • 返回的是转换之后的值,可为integer“float、string、array或object
  • 若被反序列化的变量是一个对象,在成功重新构造对象之后,PHP会自动地试图去调用__wakeup()成员函数(如果存在的话)

例子:

class User{
    const SITE = 'uusama';

    public $username;
    public $nickname;
    private $password;
    private $order;

    public function __construct($username, $nickname, $password)
    {
        $this->username = $username;
        $this->nickname = $nickname;
        $this->password = $password;
    }

    // 定义反序列化后调用的方法
    public function __wakeup()
    {
        $this->password = $this->username;
    }
}
$user_ser = 'O:4:"User":2:{s:8:"username";s:6:"uusama";s:8:"nickname";s:2:"uu";}';
var_dump(unserialize($user_ser));

输出结果为:

object(User)#1 (4) {
  ["username"]=>
  string(6) "uusama"
  ["nickname"]=>
  string(2) "uu"
  ["password":"User":private]=>
  string(6) "uusama"
  ["order":"User":private]=>
  NULL
}
  • __wakeup()函数在对象被构建以后执行,所以$this->username的值不为空
  • 反序列化时,会尽量将变量值进行匹配并复制给序列化后的对象

未定义类的处理

在上面的例子中,我们在调用反序列化函数unserialize()之前,提前定义了User类

没有定义例:

$user_ser = 'O:4:"User":2:{s:8:"username";s:6:"uusama";s:8:"nickname";s:2:"uu";}';
var_dump(unserialize($user_ser));

得到的结果:

object(__PHP_Incomplete_Class)#1 (3) {
  ["__PHP_Incomplete_Class_Name"]=>
  string(4) "User"
  ["username"]=>
  string(6) "uusama"
  ["nickname"]=>
  string(2) "uu"
}

这个例子中,没有定义任何的User类,反序列化正常执行,并没有报错,对比之前定义了User类的结果,这儿反序列化得到的对象是__PHP_Incomplete_Class,并指定了未定义类的类名。

如果这个时候去使用这个反序列化后的不明对象,则会抛出E_NOTICE。

有两种解决方案。

  • 定义__autoload()等函数,指定发现未定义类时加载类的定义文件
  • 可通过 php.ini、ini_set() 或 .htaccess 定义unserialize_callback_func。每次实例化一个未定义类时它都会被调用

以上两种方案的实现如下:

// unserialize_callback_func 从 PHP 4.2.0 起可用
ini_set('unserialize_callback_func', 'mycallback'); // 设置您的回调函数
function mycallback($classname)
{
   // 只需包含含有类定义的文件
   // $classname 指出需要的是哪一个类
}

// 建议使用下面的函数,代替__autoload()
spl_autoload_register(function ($class_name) {
    // 动态加载未定义类的定义文件
    require_once $class_name . '.php';
});

PHP预定义序列化接口Serializable

上面在将序列化过程中遇到的:无法在__sleep()方法中返回父类对象的问题吗,方法就是实现序列化接口Serializable。

该接口的原型如下:

Serializable {
    abstract public string serialize ( void )
    abstract public mixed unserialize ( string $serialized )
}

需要注意的是,如果定义的类实现了Serializable接口,那么序列化和反序列化的时候,PHP就不会再去调用__sleep()方法和__wakeup()方法。

class CB implements Serializable{
    public $CB_data = '';
    private $CB_password = 'ttt';
    public function setCBPassword($password)
    {
        $this->CB_password = $password;
    }

    public function serialize()
    {
        echo __METHOD__ . "\n";
        return serialize($this->CB_password);
    }

    public function unserialize($serialized)
    {
        echo __METHOD__ . "\n";
    }
}

class CC extends CB {
    const SECOND = 60;

    public $data;
    private $pass;
    public function __construct($data, $pass)
    {
        $this->data = $data;
        $this->pass = $pass;
    }
    public function __sleep()
    {
        // 输出调用了该方法名
        echo __METHOD__ . "\n";
    }

    public function __wakeup()
    {
        // 输出调用了该方法名
        echo __METHOD__ . "\n";
    }
}

$cc = new CC('uu', true);
$ser = serialize($cc);
var_dump($ser);
$un_cc = unserialize($ser);
var_dump($un_cc);

运行结果为:

CB::serialize
string(24) "C:2:"CC":10:{s:3:"ttt";}"
CB::unserialize
object(CC)#2 (4) {
  ["data"]=>
  NULL
  ["pass":"CC":private]=>
  NULL
  ["CB_data"]=>
  string(0) ""
  ["CB_password":"CB":private]=>
  string(3) "ttt"
}

可以完全定义serialize()方法,该方法返回的值就是序列化后大括号内的值,只要保证自定义序列化和反序列化的规则一致即可。

CentOS 7 安装 LNMP

记录在CentOS 7安装LNMP环境(PHP7 + MySQL5.7 + Nginx1.10)的过程。

一 、修改yum源

rpm -Uvh https://dl.Fedoraproject.org/pub/epel/epel-release-latest-7.noarch.rpm

rpm -Uvh https://mirror.webtatic.com/yum/el7/webtatic-release.rpm

rpm -Uvh http://dev.mysql.com/get/mysql57-community-release-el7-9.noarch.rpm

二 、安装 Nginx、MySQL、PHP

yum -y install nginx

yum -y install mysql-community-server

yum -y install php70w-devel php70w.x86_64 php70w-cli.x86_64 php70w-common.x86_64 php70w-gd.x86_64 php70w-ldap.x86_64 php70w-mbstring.x86_64 php70w-mcrypt.x86_64  php70w-pdo.x86_64   php70w-mysqlnd  php70w-fpm php70w-opcache php70w-pecl-redis php70w-pecl-mongo

三 、配置 MySQL

systemctl start mysqld #启动MySQL

grep 'temporary password' /var/log/mysqld.log #查找默认密码
#输出:2018-07-12T09:04:58.532626Z 1 [Note] A temporary password is generated for root@localhost: qrky5irl-y5O

mysql -uroot -p'qrky5irl-y5O' #登录MySQL

set password for 'root'@'localhost'=password('!@#123qwe'); #修改密码 包含字符,英文,数字

vim /etc/my.cnf #配置默认编码
[mysqld]
character_set_server=utf8
init_connect='SET NAMES utf8'

systemctl restart mysqld #重启MySQL

systemctl enable mysqld #设置开机启动

配置文件:/etc/my.cnf
日志文件:/var/log/mysqld.log
服务启动脚本:/usr/lib/systemd/system/mysqld.service
socket 文件:/var/run/mysqld/mysqld.pid

四 、配置 Nginx

systemctl status firewalld #查看防火墙配置,如果显示active(running),则需要调整防火墙规则的配置。

vim /etc/firewalld/zones/public.xml #配置防火墙规则
<zone>
<service name="nginx"/>
<zone>

systemctl reload firewalld #重启firewalld

vim /etc/nginx/nginx.conf #修改Nginx配置
#在server{}里添加:
location / {
    #定义首页索引文件的名称
    index index.php index.html index.htm;   
}
# PHP 脚本请求全部转发到 FastCGI处理. 使用FastCGI默认配置.
location ~ .php$ {
    fastcgi_pass 127.0.0.1:9000;
    fastcgi_index index.php;
    fastcgi_param  SCRIPT_FILENAME  $document_root$fastcgi_script_name;
    include fastcgi_params;
}

systemctl start nginx #启动Nginx

systemctl enable nginx #设置开机启动

五 、配置PHP

systemctl enable php-fpm #设置开机启动

systemctl start php-fpm #启动php-fpm

六 、测试

vim /usr/share/nginx/html/phpinfo.php #创建测试页
<?php
phpinfo();
?>

#页面访问http://IP地址/phpinfo.php,成功输入,配置成功

Memcached和Memcache

Memcache::set

bool Memcache::set ( string $key , mixed $var [, int $flag [, int $expire ]] )

参数
key
要设置值的key。
var
要存储的值,字符串和数值直接存储,其他类型序列化后存储。
flag
使用MEMCACHE_COMPRESSED指定对值进行压缩(使用zlib)。
expire
当前写入缓存的数据的失效时间。如果此值设置为0表明此数据永不过期。你可以设置一个UNIX时间戳或 以秒为单位的整数(从当前算起的时间差)来说明此数据的过期时间,但是在后一种设置方式中,不能超过 2592000秒(30天)。

 

Memcached::set

public bool Memcached::set ( string $key , mixed $value [, int $expiration ] )

参数
key
用于存储值的键名。
value
存储的值。
expiration
到期时间,默认为 0。 更多信息请参见到期时间。

WINDOWS环境安装PHP的RABBITMQ扩展

1.下载amqp扩展模块php_amqp.dll,放在\php\ext\目录下;rabbitmq扩展模块rabbitmq.1.dll,放在\php目录下

下载地址:http://pecl.php.net/package/amqp/1.4.0/windows

根据PHP版本(5.3,5.4,5.5,5.6),是否线程安全(TS,NTS),PHP服务的版本(X84,X64)

2.在PHP.INI中写入

[amqp]

extension=php_amqp.dll

3.在APACHE的HTTPD.CONF中写入

LoadFile  ”C:\xampp\php\rabbitmq.1.dll”

路径根据环境自行修改

4.重启服务

WINDOWS环境安装PHP的MEMCACHE扩展

1.下载memcache扩展模块php_memcache.dll,放在\php\ext\目录下

下载地址:http://pecl.php.net/package/memcache/3.0.8/windows

根据PHP版本(5.3,5.4,5.5,5.6),是否线程安全(TS,NTS),PHP服务的版本(X84,X64)

2.在PHP.INI中写入

extension=php_memcache.dll
[Memcache]
memcache.allow_failover=1
memcache.max_failover_attempts=20
memcache.chunk_size=8192
memcache.default_port=11211

3.重启服务

过滤xml中的不合法字符

public function filterStringXml($str) {
$str = @iconv(‘UTF-8′, ‘GBK’, $str);
$str = @iconv(‘GBK’, ‘UTF-8′, $str);
$str = str_replace(“\xC”, ”, $str);

$str = preg_replace(‘/[^\x{0000}-\x{FFFF}]/iu’, ”, $str);
$str = preg_replace(“/[\\x00-\\x08\\x0b-\\x0c\\x0e-\\x1f]/”, “”, $str);
return $str;
}

BOM头的处理

类似WINDOWS自带的记事本等软件,在保存一个以UTF-8编码的文件时,会在文件开始的地方插入三个不可见的字符(0xEF 0xBB 0xBF,即BOM)。它是一串隐藏的字符,用于让记事本等编辑器识别这个文件是否以UTF-8编码。对于 PHP来说,BOM是个大麻烦,尤其是在用PHP做接口输出的时候。

PHP并不会忽略BOM,在读取、包含或者引用这些文件时,会把BOM作为该文件开头正文的一部分,一并输出出去。所以,在调用带有BOM的接口的时候,时常会遇到报文无法正确解析,但是看上去,报文又没有任何问题。

所以,在用PHP做接口被调用方时,一定要避免出现BOM头,这样给大家带来的麻烦都会减少,如果是在调用对方接口时候,对方接口存在BOM头,无法让对方去掉的时候,我们就要自己去手动去掉BOM。

附上一段代码,去判断是否存在BOM,并且去掉。

function checkBOM($filename) {
$contents = file_get_contents($filename);
$charset[1] = substr($contents, 0, 1);
$charset[2] = substr($contents, 1, 1);
$charset[3] = substr($contents, 2, 1);
if (ord($charset[1]) == 239 && ord($charset[2]) == 187 && ord($charset[3]) == 191)         {
$contents = substr($contents, 3);
}
return $contents;
}

PHP接收文件流的HTTP请求

最近,跟JAVA系统进行交互,学习到了一个新东西,HTTP请求接口时,用文件流传输数据。

这种方式相当于在BODY体里边传输数据,而非REQUEST的KEY=>VALUE形式,经过查询后,这种形式的传输,用file_get_contents(“php://input”),这种方式可以获取。

file_get_contents(“php://input”)这种方式同样也可以获取POST和GET的内容,不过会以字符串形式输出。