二、PHP四種序列化方案橫向?qū)Ρ?/strong>
數(shù)據(jù)的序列化是一個(gè)非常有用的功能,然而目測(cè)很多人跟我一樣,在剛接觸這玩意的時(shí)候壓根就不理解這貨色到底是干啥用的,反正老師說(shuō)了,實(shí)在理解不了就先背過(guò)再說(shuō)。
其實(shí)將數(shù)據(jù)序列化的作用無(wú)外乎有兩個(gè):
- 方便傳輸
- 方便存儲(chǔ)
方便存儲(chǔ)如何理解呢?比如我們有個(gè)PHP對(duì)象或者一個(gè)PHP數(shù)組需要存儲(chǔ)到數(shù)據(jù)庫(kù)甚至文件中,這顯然是不可能的,這個(gè)時(shí)候必須要將PHP對(duì)象或者PHP數(shù)組序列化后再執(zhí)行存儲(chǔ)操作。不過(guò)這將PHP數(shù)組序列化后存起來(lái)還能理解,這對(duì)象也能存儲(chǔ)啊?這操作是否過(guò)于風(fēng)騷?少年,這一點(diǎn)兒都不風(fēng)騷。有些時(shí)候?qū)?duì)象直接存儲(chǔ)起來(lái),用的時(shí)候只需要簡(jiǎn)單的反序列化后就可以投產(chǎn)使用了,避免了new一次帶來(lái)的性能耗費(fèi)。
方便傳輸如何理解呢?其實(shí)序列化在傳輸中應(yīng)用的相對(duì)更多更常見(jiàn)些許。最簡(jiǎn)單的一個(gè)例子,一個(gè)碼前端的碼了一個(gè)ajax找你給TA提供一個(gè)API,那么這個(gè)時(shí)候你倆得商量返回什么數(shù)據(jù),比如json或者xml,甚至你倆自己作死約定私有數(shù)據(jù)格式。比如在一個(gè)比較典型的服務(wù)架構(gòu)中,網(wǎng)關(guān)服務(wù)器和內(nèi)部RPC服務(wù)器之間通過(guò)msgpack傳遞數(shù)據(jù)。這都是典型的序列化為了傳輸?shù)牡湫蛻?yīng)用案例。
這里序列化的概念可能更為廣泛和籠統(tǒng)一些,包括傳統(tǒng)的serialize、json、msgpack、protobuf等。( 如果你覺(jué)得序列化這個(gè)稱呼不太嚴(yán)謹(jǐn)?shù)脑挘梢杂胑ncode來(lái)代替;反序列化則用decode來(lái)代替。反正我就用統(tǒng)統(tǒng)用序列化和反序列化來(lái)稱呼了,如果你覺(jué)得實(shí)在不舒服,可以順著網(wǎng)線來(lái)砍我!)。
實(shí)際上,從更高的層面看,數(shù)據(jù)的序列化可以分為兩種:
- 文本序列化,常見(jiàn)如json、serialize、xml等
- 二進(jìn)制序列化,常見(jiàn)如msgpack、protobuf、thrift等
一般說(shuō)來(lái),考驗(yàn)序列化技術(shù)的性能指標(biāo)一共有兩個(gè),一個(gè)是序列化的速度,一個(gè)是序列化后數(shù)據(jù)的大小,自然是序列化速度越快、序列化后的數(shù)據(jù)越小為佳。就目前來(lái)看,protobuf、msgpack等二進(jìn)制序列化無(wú)論是速度上還是數(shù)據(jù)大小上,都要比文本序列化更好。不過(guò)話說(shuō)回來(lái),文本序列化有更好的可讀性,一眼就能瞪出來(lái)數(shù)據(jù)內(nèi)容大概是啥玩意。
今天帶到這里的這里的有四個(gè)具體的方案,這四種方案都是簡(jiǎn)單粗暴、開(kāi)箱即用類型的,我們分別測(cè)試感受下,看哪個(gè)更適合我們。
參會(huì)的四個(gè)哥們:PHP內(nèi)置的serialize、PHP內(nèi)置的JSON解析器、PHP擴(kuò)展JSOND、PHP擴(kuò)展msgpack。其中前三個(gè)都是文本類型的,msgpack則是二進(jìn)制類型的。
JSOND作為PHP內(nèi)置的JSON解析器的高級(jí)版本,坊間一直傳聞速度上要比內(nèi)置的更牛X一些,作為擴(kuò)展,這貨需要額外安裝,附送地址:https://pecl.php.net/get/jsond-1.4.0.tgz。
msgpack是一個(gè)鳥(niǎo)哥等人搞的一套二進(jìn)制序列化工具,slogan就是“It's like JSON.but fast and small.”,附送地址:https://pecl.php.net/get/msgpack-2.0.2.tgz
1、serialize用法
serialize(),序列化方法。
unserialize(),反序列化方法。
2、json用法
json_encode(),沒(méi)啥好說(shuō)的吧?
json_decode(),沒(méi)啥好說(shuō)的吧?
3、jsond用法
jsond_encode(),和json_encode()一樣,后面多個(gè)字母d而已。
jsond_decode(),和json_decode()一樣,后面多個(gè)字母d而已。
4、msgpack用法
msgpack_pack(),序列化方法。
msgpack_unpack(),反序列化方法。
測(cè)試代碼如下:
<?php
// 故意搞了一個(gè)還算大的php數(shù)組,更容易看出差距來(lái)
$arr
=
array
(
array
(
'uid'
=> 22193123,
'gender'
=>
'famale'
,
'username'
=>
'elarity'
,
'password'
=> md5(
'www123'
),
'relation'
=>
array
(
array
(
'uid'
=> 22193123,
'gender'
=>
'famale'
,
'username'
=>
'elarity'
,
'password'
=> md5(
'www123'
),
),
array
(
'uid'
=> 22193123,
'gender'
=>
'famale'
,
'username'
=>
'elarity'
,
'password'
=> md5(
'www123'
),
),
array
(
'uid'
=> 22193123,
'gender'
=>
'famale'
,
'username'
=>
'elarity'
,
'password'
=> md5(
'www123'
),
),
array
(
'uid'
=> 22193123,
'gender'
=>
'famale'
,
'username'
=>
'elarity'
,
'password'
=> md5(
'www123'
),
),
array
(
'uid'
=> 22193123,
'gender'
=>
'famale'
,
'username'
=>
'elarity'
,
'password'
=> md5(
'www123'
),
),
array
(
'uid'
=> 22193123,
'gender'
=>
'famale'
,
'username'
=>
'elarity'
,
'password'
=> md5(
'www123'
),
),
array
(
'uid'
=> 22193123,
'gender'
=>
'famale'
,
'username'
=>
'elarity'
,
'password'
=> md5(
'www123'
),
),
array
(
'uid'
=> 22193123,
'gender'
=>
'famale'
,
'username'
=>
'elarity'
,
'password'
=> md5(
'www123'
),
),
array
(
'uid'
=> 22193123,
'gender'
=>
'famale'
,
'username'
=>
'elarity'
,
'password'
=> md5(
'www123'
),
),
),
)
);
// 每種序列化方案都執(zhí)行100000次
$counter
= 100000;
// json序列化方案,執(zhí)行100000次
echo
PHP_EOL.PHP_EOL;
$start
= microtime( true );
for
(
$i
= 1;
$i
<=
$counter
;
$i
++ ){
$json
= json_encode(
$arr
);
}
$size
=
strlen
(
$json
);
$end
= microtime( true );
$cost_time
=
$end
-
$start
;
echo
"json_encode : 耗費(fèi)時(shí)間為{$cost_time} , 數(shù)據(jù)體積為{$size}"
.PHP_EOL;
// jsond序列化方案,執(zhí)行100000次
$start
= microtime( true );
for
(
$i
= 1;
$i
<=
$counter
;
$i
++ ){
$jsond
= jsond_encode(
$arr
);
}
$size
=
strlen
(
$jsond
);
$end
= microtime( true );
$cost_time
=
$end
-
$start
;
echo
"jsond_encode : 耗費(fèi)時(shí)間為{$cost_time} , 數(shù)據(jù)體積為{$size}"
.PHP_EOL;
// serialize序列化方案,執(zhí)行100000次
$start
= microtime( true );
for
(
$i
= 1;
$i
<=
$counter
;
$i
++ ){
$serialize
= serialize(
$arr
);
}
$size
=
strlen
(
$serialize
);
$end
= microtime( true );
$cost_time
=
$end
-
$start
;
echo
"serialize : 耗費(fèi)時(shí)間為{$cost_time} , 數(shù)據(jù)體積為{$size}"
.PHP_EOL;
// msgpack序列化方案,執(zhí)行100000次
$start
= microtime( true );
for
(
$i
= 1;
$i
<=
$counter
;
$i
++ ){
$msgpack
= msgpack_pack(
$arr
);
}
$size
=
strlen
(
$msgpack
);
$end
= microtime( true );
$cost_time
=
$end
-
$start
;
echo
"msgpack耗費(fèi)時(shí)間為 : {$cost_time} , 數(shù)據(jù)體積為{$size}"
.PHP_EOL;
echo
PHP_EOL.PHP_EOL;
將文件保存為test.php,然后php test.php執(zhí)行,結(jié)果如下圖所示:
總結(jié)一下:
- jsond確實(shí)是要比json快一些的
- 總有刁民張嘴就來(lái)json要比serialize()快
- serialize()數(shù)據(jù)體積確實(shí)大(因?yàn)檫€保留了數(shù)據(jù)類型說(shuō)明)
- msgpack最佳???不知道昂,你們自己感受
- Php兩點(diǎn)地理坐標(biāo)距離的計(jì)算方法和具體代碼
- PHP獲取HTTP body內(nèi)容的方法
- PHP面向?qū)ο蟪绦蛟O(shè)計(jì)中獲取對(duì)象屬性的3種方法實(shí)例分析
- php5.5新增的yield關(guān)鍵字功能與相關(guān)使用技巧
- Windows7下IIS+php配置教程詳細(xì)介紹
- PHP基于Redis消息隊(duì)列實(shí)現(xiàn)的消息推送的方法
- Linux服務(wù)器下 php7安裝redis的方法
- PHP判斷電子郵件是否正確的簡(jiǎn)單方法介紹
- 在PHP中進(jìn)行curl開(kāi)啟操作的具體教程
- PHP中間件ICE,ICE的安裝配置,ICE常見(jiàn)編譯和運(yùn)行(異常)