tcp 긴 링크 모드에서는 고정된 메시지 헤더 길이를 사용하여 메시지의 압축을 풀어 패킷 고착 문제를 해결합니다.
고정된 메시지 헤더 프로토콜
은 메시지 헤더의 첫 번째 N 바이트를 메시지 길이 비트로 수정합니다. 메시지를 읽을 때 메시지 길이 비트를 먼저 읽은 다음 메시지 길이 비트로 읽습니다. 특정 메시지 길이를 가져옵니다.
pack/unpack은 값을 바이너리로 압축하거나 바이너리를 값으로 압축 해제할 수 있습니다. 특정 모드의 경우 pack/unpack의 자세한 사용법을 참조하세요. 여기서는 메시지 본문 길이를 나타내기 위해 2바이트의 고정 헤더 길이를 선택합니다. 최대 2^16 - 1까지 표현할 수 있습니다. 메시지 본문의 길이가 부족할 경우 4바이트를 사용할 수 있습니다.
Assembling packages
<?php// msg protocol// | ---- dataLen ---- | data |// | - fixed 2bytes - |// 模拟客户端连续发送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;
Adhesive packages
// send// 传输 $package 由 $foo $bar 两条消息组成 模拟粘包场景 // receive
Unpacking
<?php // 解析第1条消息 取前 2bytes 按 n 解包 $fooLen = unpack("n", substr($package, 0, 2))[1]; // 使用包消息体长度定义读取消息体 // 从第 3byte 开始读 前 2bytes表示长度 $foo = substr($package, 2, $fooLen); echo $foo . PHP_EOL; // 解析第2条消息 取前 2bytes 按 n 解包 // 0 ~ (2 + fooLen) - 1 字节序为 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;
일상 작업에서 자주 접하는 tcp 시나리오는 짧은 연결과 단일 메시지 모드일 수 있으며, 클라이언트는 메시지를 보낸 후 연결을 끊고 서버는 이를 읽습니다. 루프 완전한 메시지를 얻으려면 EOF로 이동하세요. 그러나 짧은 연결이나 긴 연결 모드에서 여러 개의 메시지가 있는 경우 끈적한 패킷이 발생할 수 있으며, 클라이언트는 서버를 닫지 않으면 EOL을 통해 메시지를 읽었는지 확인할 수 없습니다. 이를 위해서는 프로토콜을 정의하고 압축을 풀어야 합니다.
PHP 관련 기술 기사를 더 보려면 PHP 튜토리얼 칼럼을 방문하여 알아보세요!
위 내용은 php - tcp 패킷 고정/압축 풀기 예제의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!