当前位置: 首页 > news >正文

Linux 网络编程必知:setsockopt、缓冲区、地址重用、多播端口一次讲透

Linux 网络编程必知:setsockopt、缓冲区、地址重用、多播端口一次讲透

关键词:setsockopt、SO_RECVBUF、SO_SNDBUF、SO_REUSEADDR、多播端口、UDP 组播、TCP_NODELAY

一、setsockopt 到底能改什么?

setsockopt只能改 内核套接字缓冲区 及相关行为,改不到 你read/recv时自己传的那块用户空间数组。

常见误区:把“内核接收缓冲区”和“用户空间 buf”混为一谈——两者不在一层。

缓冲区位置 控制方式 默认大小来源
内核接收缓冲区setsockopt(fd, SOL_SOCKET, SO_RCVBUF, …)/proc/sys/net/core/rmem_default
内核发送缓冲区setsockopt(fd, SOL_SOCKET, SO_SNDBUF, …)/proc/sys/net/core/wmem_default
用户空间缓冲区read/recv/recvfrom调用时你自己传的数组 代码里写多大就是多大

二、地址重用 SO_REUSEADDR 的真实作用

一句话:让新进程可以立刻bind到“刚刚被关闭、仍处于 TIME_WAIT 的同一 IP:端口”,重启服务不再报 “Address already in use”。

  • UDP:立竿见影,多播/广播接收器常用,无需等待。
  • TCP:仅解决 “TIME_WAIT 拖延复用”,不能 创建真正重复的四元组。

代码模板(TCP/UDP 通用):

intfd=socket(AF_INET,SOCK_STREAM,0);/* 或 SOCK_DGRAM */inton=1;setsockopt(fd,SOL_SOCKET,SO_REUSEADDR,&on,sizeof(on));

三、多播(组播)到底该用哪个端口?

技术上 没有硬性限制,只要避开系统/已知服务端口即可。

推荐范围:1024–65535 里任意空闲 UDP 端口。

场景 常用端口示例
路由协议 224.0.0.5:89(OSPF)
车载、音视频 239.x.x.x + 30000 以上
局域网测试 8888、9999、22334

防火墙/云主机注意:安全组需放行 UDP 协议 + 你选定的端口;部分云厂商把 135、445、5900 等列为高危端口,避开即可。

四、Linux UDP 多播最小可运行示例

多播地址范围:224.0.0.0–239.255.255.255,本例用 224.1.1.1:8888。

  1. 发送端 multicast_send.c
#include<stdio.h>#include<string.h>#include<stdlib.h>#include<unistd.h>#include<arpa/inet.h>#include<sys/socket.h>#defineMULTICAST_IP"224.1.1.1"#defineMULTICAST_PORT8888#defineBUF_SIZE1024intmain(void){intsock=socket(AF_INET,SOCK_DGRAM,0);if(sock<0){perror("socket");return-1;}structsockaddr_indst={.sin_family=AF_INET,.sin_port=htons(MULTICAST_PORT),.sin_addr.s_addr=inet_addr(MULTICAST_IP)};unsignedcharttl=2;/* 允许跨路由器 */setsockopt(sock,IPPROTO_IP,IP_MULTICAST_TTL,&ttl,sizeof(ttl));charbuf[BUF_SIZE];while(fgets(buf,sizeof(buf),stdin)){sendto(sock,buf,strlen(buf),0,(structsockaddr*)&dst,sizeof(dst));if(strncmp(buf,"quit",4)==0)break;}close(sock);return0;}
  1. 接收端 multicast_recv.c
#include<stdio.h>#include<string.h>#include<stdlib.h>#include<unistd.h>#include<arpa/inet.h>#include<sys/socket.h>#include<netinet/in.h>#defineMULTICAST_IP"224.1.1.1"#defineMULTICAST_PORT8888#defineBUF_SIZE1024intmain(void){intsock=socket(AF_INET,SOCK_DGRAM,0);if(sock<0){perror("socket");return-1;}intreuse=1;setsockopt(sock,SOL_SOCKET,SO_REUSEADDR,&reuse,sizeof(reuse));structsockaddr_inlocal={.sin_family=AF_INET,.sin_port=htons(MULTICAST_PORT),.sin_addr.s_addr=INADDR_ANY};if(bind(sock,(structsockaddr*)&local,sizeof(local))<0){perror("bind");return-1;}structip_mreqmreq={.imr_multiaddr.s_addr=inet_addr(MULTICAST_IP),.imr_interface.s_addr=INADDR_ANY};if(setsockopt(sock,IPPROTO_IP,IP_ADD_MEMBERSHIP,&mreq,sizeof(mreq))<0){perror("IP_ADD_MEMBERSHIP");return-1;}charbuf[BUF_SIZE];structsockaddr_insender;socklen_tsender_len=sizeof(sender);while(1){ssize_tn=recvfrom(sock,buf,sizeof(buf)-1,0,(structsockaddr*)&sender,&sender_len);if(n<0){perror("recvfrom");break;}buf[n]=0;printf("recv from %s:%d %s",inet_ntoa(sender.sin_addr),ntohs(sender.sin_port),buf);if(strncmp(buf,"quit",4)==0)break;}setsockopt(sock,IPPROTO_IP,IP_DROP_MEMBERSHIP,&mreq,sizeof(mreq));close(sock);return0;}
  1. 编译与运行
gcc multicast_send.c-osend gcc multicast_recv.c-orecv# 终端 1./recv# 终端 2./send

同一网段可启动多个recv实例,都能收到数据包。

五、setsockopt 函数原型速查

intsetsockopt(intsockfd,intlevel,/* SOL_SOCKET / IPPROTO_IP / IPPROTO_TCP … */intoptname,/* SO_REUSEADDR / SO_RCVBUF / TCP_NODELAY … */constvoid*optval,socklen_toptlen);

示例:一次把地址重用、接收缓冲区、TCP_NODELAY 全设好

intfd=socket(AF_INET,SOCK_STREAM,0);inton=1;setsockopt(fd,SOL_SOCKET,SO_REUSEADDR,&on,sizeof(on));intrcvbuf=64*1024;setsockopt(fd,SOL_SOCKET,SO_RCVBUF,&rcvbuf,sizeof(rcvbuf));setsockopt(fd,IPPROTO_TCP,TCP_NODELAY,&on,sizeof(on));

注意:TCP_NODELAY仅限 TCP,UDP 会返回-1

六、Linux 默认接收缓冲区到底多大?

查看当前系统值:

cat/proc/sys/net/core/rmem_default# 常见输出:212992 (约 208 KB)

不调用SO_RCVBUF时,新建套接字的初始接收缓冲区 =rmem_default

现代主流发行版均为 208 KB(不同内核可能略有差异)。


http://www.jsqmd.com/news/226195/

相关文章:

  • 3.2 FileStream
  • Infra十年演进(2015–2025)
  • 实战案例:汽车数字孪生车间的提示工程应用
  • 嘉环科技携手 TDengine,助力某水务公司构建一体化融合平台
  • ControlNet十年演进(2015–2025)
  • 吐血推荐专科生必用TOP8AI论文平台
  • 华为OD机试 - 整型数组按照个位数排序(Java 双机位C卷 100分)
  • 信创电话助手电话录音盒操作系统兼容性
  • 当模型“知道自己在作弊”:Scheming 与 Reward Hacking 的技术解剖
  • HY-MT1.5开源社区贡献指南:模型改进与反馈提交实战
  • 巴菲特的公司治理观点
  • 吐血推荐9个AI论文写作软件,研究生轻松搞定毕业论文!
  • springboot竞赛团队组建与管理系统的设计与实现
  • Triton十年演进(2015–2025)
  • 1.线性switch case语句逆向特征
  • 实时数据异常检测模块
  • HY-MT1.5-7B A/B测试:不同参数版本效果对比部署方案
  • 实时日志分析:ELK Stack深度优化指南
  • MoE(Mixture of Experts)架构十年演进(2015–2025)
  • HY-MT1.5如何接入现有系统?API接口调用实战教程
  • Fine-tuning十年演进(2015–2025)
  • 导师推荐!8款一键生成论文工具测评:本科生毕业论文高效写作指南
  • HY-MT1.5-1.8B模型微调教程:特定领域适应性训练步骤
  • 9个降AI率工具推荐!继续教育学员高效避坑指南
  • 提示工程架构师实战:Agentic AI可追溯性的技术实现
  • Agent十年演进(2015–2025)
  • HY-MT1.5-7B支持哪些民族语言?方言翻译实测与部署说明
  • LangChain十年演进(2015–2025)
  • HY-MT1.5如何保护隐私?完全离线翻译系统搭建
  • Llama十年演进(2015–2025)