# Swoole编程指南-EOF协议 [原文链接](http://www.catplanet.me/?id=12) [TOC] ## 什么是EOF协议 EOF(End of File)是一个结束标记,意思为在逐个读取数据流中的数据时,如果发现读到EOF标记,就代表已经读到了数据末尾。在TCP的数据流中,使用EOF协议的数据流特征如下: > | 数据 | EOF | 数据 |EOF| 在每一串正常数据的末尾,添加一个预先规定的、绝对不会出现在数据中的字符串作为结束标记。这样接收到的数据就可以根据这个EOF标记来切分数据。 ## 开启EOF支持 在Swoole中,可以使用如下配置选项来开启EOF功能: ``` $server->set([ 'open_eof_split' => true, // 开启EOF检测 'package_eof' => '/r/n' , // 设置EOF标记 ]); ``` 其中,open_eof_split选项会开启Swoole底层对接收到的数据从头开始依次扫描检查,当找到第一个EOF标记时,将已经扫描过的数据作为一个完整的数据包通过onReceive回调发送给PHP层处理。这里需要注意的是,package_eof只允许设置长度不超过8的字符串。 这里要注意到,open_eof_split选项是依次扫描数据中的EOF标记的,这样虽然保证每次回调都只会收到一个完整的数据包,但是性能较差。因此Swoole还提供了另外一个不同的选项: ``` $server->set([ 'open_eof_check' => true, // 开启EOF检测 ]); ``` open_eof_check同样会开启EOF检测。不同的是,open_eof_check只检查接收数据的末尾是否为EOF标记。相比于open_eof_split,这种方式性能最好,几乎没有损耗。但是如果同时收到了N条带有EOF标记的数据,这种方式会同时将N个数据包合并为一个回调给PHP层处理,因此需要在PHP层通过EOF标记对数据做二次拆分。 ## 实战 为了演示效果,我将直接使用open_eof_check选项,并演示如何拆分数据包。 首先是服务器端代码,如下: ``` class Server { private $serv; public function __construct() { $this->serv = new swoole_server("0.0.0.0", 9501); $this->serv->set(array( 'worker_num' => 1, 'open_eof_check' => true, 'package_eof' => "\r\n", )); $this->serv->on('Connect', array($this, 'onConnect')); $this->serv->on('Receive', array($this, 'onReceive')); $this->serv->on('Close', array($this, 'onClose')); $this->serv->start(); } public function onConnect( $serv, $fd, $from_id ) { echo "Client {$fd} connect\n"; } public function onReceive( swoole_server $serv, $fd, $from_id, $data ) { $data_list = explode("\r\n", $data); foreach ($data_list as $item) { if(empty($item)) continue; var_dump($item); $serv->send($fd, $item . "\r\n"); } } public function onClose( $serv, $fd, $from_id ) { echo "Client {$fd} close connection\n"; } } new Server(); ``` 可以看到,在接收到数据后,我先将接收到的数据用EOF标记作为分隔符做了一次拆分,这样就能得到一个个独立的完整数据包,然后依次处理即可。