本文給大家介紹一個(gè)例子,有關(guān)PHP的tcp 粘包/拆包,希望對(duì)需要的朋友有所幫助~
php入門到就業(yè)線上直播課:進(jìn)入學(xué)習(xí)
Apipost = Postman + Swagger + Mock + Jmeter 超好用的API調(diào)試工具:點(diǎn)擊使用
tcp 長(zhǎng)鏈接模式下,使用固定消息頭長(zhǎng)度的方式進(jìn)行消息 拆包 ,解決 粘包 問(wèn)題。 固定消息頭協(xié)議
將消息頭的前 N 個(gè)字節(jié)固定為 消息長(zhǎng)度位 ,結(jié)合業(yè)務(wù)場(chǎng)景, 2bytes 或 4bytes ,讀取消息時(shí)先讀取 消息長(zhǎng)度位 ,即可按具體的 消息長(zhǎng)度 讀取 消息內(nèi)容 。【推薦學(xué)習(xí):PHP視頻教程】
pack/unpack 可以 打包數(shù)值至二進(jìn)制 / 解包二進(jìn)制至數(shù)值 ,具體的模式可以參考 pack/unpack 詳細(xì)用法,這里我們選用固定頭長(zhǎng)度為 2bytes 來(lái)表示 消息體長(zhǎng)度 ,最大能表示 2^16 – 1 長(zhǎng)度的消息體,不夠你就上 4bytes 好了。
組包
<?php // msg protocol // | ---- dataLen ---- | data | // | - fixed 2bytes - | // 模擬客戶端連續(xù)發(fā)送2條消息 $foo = "hello world"; $bar = "i am sqrt_cat"; $package = ""; // 使用 n 打包 固定2bytes $fooLenn = pack("n", strlen($foo)); $package = $fooLenn . $foo; $barLenn = pack("n", strlen($bar)); $package .= $barLenn . $bar;
粘包
// send // 傳輸 $package 由 $foo $bar 兩條消息組成 模擬粘包場(chǎng)景 // receive
拆包
<?php // 解析第1條消息 取前 2bytes 按 n 解包 $fooLen = unpack("n", substr($package, 0, 2))[1]; // 使用包消息體長(zhǎng)度定義讀取消息體 // 從第 3byte 開始讀 前 2bytes表示長(zhǎng)度 $foo = substr($package, 2, $fooLen); echo $foo . PHP_EOL; // 解析第2條消息 取前 2bytes 按 n 解包 // 0 ~ (2 + fooLen) - 1 字節(jié)序?yàn)?fooLen . foo // (2 + fooLen) ~ (2 + fooLen) + 2 - 1 為 barLen $barLen = unpack("n", substr($package, (2 + $fooLen), 2))[1]; $bar = substr($package, (2 + $fooLen) + 2, $barLen); echo $bar . PHP_EOL;
日常工作中經(jīng)常遇到的 tcp 場(chǎng)景可能是 短連接單個(gè)消息 的模式,客戶端發(fā)送一條消息后便關(guān)閉連接,服務(wù)端循環(huán)讀取到 EOF 即可得到一條完整的消息。但如果是 短連接多個(gè)消息 或 長(zhǎng)鏈接模式 下,就可能會(huì)發(fā)生粘包,客戶端不關(guān)閉服務(wù)端無(wú)法通過(guò) EOL 確定消息讀取完畢的問(wèn)題。這就需要定義協(xié)議和拆包。