onReceive
接收到數據時回調此函數,發生在worker進程中。函數原型: (推薦學習: swoole視頻教程)
function onReceive(swoole_server $server, int $fd, int $reactor_id, string $data);
$server,Server對象
$fd,TCP客戶端連接的唯一標識符
$reactor_id,TCP連接所在的Reactor線程ID
$data,收到的數據內容,可能是文本或者二進制內容
關于$fd和$reactor_id 詳細的解釋
未開啟自動協議選項,onReceive單次收到的數據最大為64K
開啟了自動協議處理選項,onReceive將收到完整的數據包,最大不超過 package_max_length
支持二進制格式,$data可能是二進制數據
使用底層提供的open_eof_check/open_length_check/open_http_protocol,可以保證數據包的完整性
不使用底層的協議處理,在onReceive后PHP代碼中自行對數據分析,合并/拆分數據包。
例如:代碼中可以增加一個 $buffer = array(),使用$fd作為key,來保存上下文數據。 每次收到數據進行字符串拼接,$buffer[$fd] .= $data,然后在判斷$buffer[$fd]字符串是否為一個完整的數據包。
默認情況下,同一個fd會被分配到同一個Worker中,所以數據可以拼接起來。使用dispatch_mode = 3時。
請求數據是搶占式的,同一個fd發來的數據可能會被分到不同的進程。所以無法使用上述的數據包拼接方法
關于粘包問題如SMTP協議,客戶端可能會同時發出2條指令。在Server中可能一次性收到,這時應用層需要自行拆包。SMTP是通過rn來分包的,所以業務代碼中需要 explode("rn", $data)來拆分數據包。
如果是請求應答式的服務,無需考慮拆分數據的問題。原因是客戶端在發起一次請求后,必須等到服務器端返回當前請求的響應數據,才會發起第二次請求,不會同時發送2個請求