首先是服务端,一定要位于公网上,我们以udp为例,其实tcp也是一样的,代码如下:
<?php $arr = []; $server = new Swoole\Server('0.0.0.0', 9503, SWOOLE_PROCESS, SWOOLE_SOCK_UDP); $server->set([ "worker_num" => 1 ]); $server->on('packet', function ($server, $data, $clientInfo) use (&$arr){ $list = explode("\n", $data); foreach ($list as $value) { $data = json_decode($value, 1); if ($data) { if ($data["do"] == "reg") { $arr[$data["user"]] = $clientInfo; var_dump($arr); $server->sendTo($clientInfo['address'], $clientInfo['port'], "Server:"); // $server->send($fd, "ok\n"); } else if ($data["do"] == "get") { $server->sendTo($clientInfo['address'], $clientInfo['port'], json_encode($arr[$data["user"]])); // $server->send($fd, json_encode($arr[$data["user"]])); // // $server->send($arr[$data["user"]]["fd"], json_encode($arr[$data["user"]])); } else { } } } //$server->sendTo($clientInfo['address'], $clientInfo['port'], "Server:{$data}"); }); $server->start();其中一个client2位于局域网中,他发送消息给服务器后,其实已经将自己的路由器的对外ip与地址告知服务器了。
<?php $client = new Swoole\Client(SWOOLE_SOCK_UDP, SWOOLE_SOCK_SYNC); $r = $client->sendto('服务器ip', 9503, json_encode(["do"=>"reg", "user"=>"client2", "passwd" => "passwd"]) . "\n"); $ret = $client->recv(); while (1) { $ret = $client->recv(); if ($ret) { var_dump($ret); } sleep(1); } $client->close();这个时候我们在另外一个地方的client1想要直接发送消息给client2,先告知服务器他的对外ip与端口号,然后获取client1的对外ip与端口号,最后直接发送udp消息给client2的对外ip和端口,这个时候client2正好正在等待,于是就获取了client1直接发送的消息,完成的打洞,那么client2想直接发送信息给client1也要获取client1的对外ip与端口。
<?php $client = new Swoole\Client(SWOOLE_SOCK_UDP, SWOOLE_SOCK_SYNC); $r = $client->sendto('服务器ip', 9503, json_encode(["do"=>"reg", "user"=>"client1", "passwd" => "passwd"]) . "\n"); echo $client->recv(); $r = $client->sendto('服务器ip', 9503, json_encode(["do"=>"get", "user"=>"client2"]) . "\n"); $ret = $client->recv(); $data = json_decode($ret, 1); while (1) { $r = $client->sendto($data["address"], $data["port"], "test" . "\n"); var_dump($r); sleep(3); }tcp打洞也是一样。
网友回复
如何编写一个chrome插件实现多线程高速下载大文件?
cdn版本的vue在网页中出现typeerror错误无法找到错误代码位置怎么办?
pywebview能否使用webrtc远程控制共享桌面和摄像头?
pywebview6.0如何让窗体接受拖拽文件获取真实的文件路径?
如何在linux系统中同时能安装运行apk的安卓应用?
python有没有离线验证码识别ocr库?
各家的ai图生视频及文生视频的api价格谁最便宜?
openai、gemini、qwen3-vl、Doubao-Seed-1.6在ui截图视觉定位这款哪家更强更准?
如何在linux上创建一个沙箱隔离的目录让python使用?
pywebview如何使用浏览器自带语音识别与webspeech 的api?