目录
一、背景
使用php+zmq实现前后端通信,出现只能发布、请求数据,不能订阅到消息及接收请求回复的消息
这是订阅端$connectAddr = 'tcp://127.0.0.1:5566';,负责发布消息

这是接收端,负责订阅消息

二、原因
ZMQ 字符串缓冲区与 Zend MM 内存布局冲突(嵌入式 PHP 8.0 NTS 专属)。接收 1 条消息后必触发 zend_mm_heap corrupted,说明 ZMQ 1.1.3 的字符串缓冲区释放逻辑与 PHP 8.0 NTS 的 Zend MM 内存布局存在底层不兼容------ 即使做了字符串复制和延迟,ZMQ 释放缓冲区时仍会意外触碰 Zend MM 管理的内存区域(嵌入式环境内存地址更紧凑,冲突概率极高)。出现Segment fault 修改代码排查也是由于zend_mm_heap corrupted 引起,也会造成只收到一条数据就发生错误。
三、解决方法
升级PHP-ZMQ 1.2.0 或者在控制端输入命令行执行export USE_ZEND_ALLOC=0,然后在运行php zmq_subscriber.php即可(临时生效)
ps:不用USE_ZEND_ALLOC性能会急剧下降,这是php自带的内存管理机制,作用是禁用 PHP 底层 Zend 引擎的内置内存分配器(Zend Memory Manager,ZMM)
但该方法重启后失效,需要写入开机自启动,导出该环境变量,在嵌入式linux中/etc/profile加入即可。

重启成功运行
