news 2026/5/14 22:51:51

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

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
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(不同内核可能略有差异)。


版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/5/14 14:22:10

3.2 FileStream

1.FileStream1.FileStream FileStream操作的是字节数组, 读写文本需配合Encoding(如: UTF8)转换1).构造函数这是使用FileStream的第一步, 用于打开/创建文件并指定操作模式// 参数&#xff1a;文件路径、文件模式、文件访问权限、文件共享方式 FileStream fs new FileStream(&…

作者头像 李华
网站建设 2026/5/1 8:47:35

实战案例:汽车数字孪生车间的提示工程应用

从物理车间到数字孪生&#xff1a;提示工程如何让汽车制造“会思考”&#xff1f; 关键词 数字孪生车间、提示工程、AI智能制造、故障预测、生产优化、自然语言交互、数据闭环 摘要 当汽车制造车间从“物理实体”进化为“数字孪生”&#xff0c;如何让这个“虚拟双胞胎”不仅能…

作者头像 李华
网站建设 2026/5/12 17:22:53

基于SpringBoot的“有光”摄影分享网站系统毕业设计

博主介绍&#xff1a;✌ 专注于Java,python,✌关注✌私信我✌具体的问题&#xff0c;我会尽力帮助你。 一、研究目的 本研究旨在设计并实现一个基于SpringBoot框架的“有光”摄影分享网站系统。该系统旨在为摄影爱好者提供一个便捷、高效、互动的在线摄影作品分享平台。具体研…

作者头像 李华
网站建设 2026/5/12 10:20:02

实时数据异常检测模块

污水监控 环境监测 云平台半夜三点&#xff0c;我盯着电脑屏幕上的折线图突然开始剧烈波动——污水厂进水口的COD浓度在十分钟内从200mg/L飙升到1200mg/L。这不是普通的污染事件&#xff0c;系统自动触发警报的同时&#xff0c;Python脚本已经开始抓取周边企业排水数据。 def d…

作者头像 李华
网站建设 2026/5/13 10:25:21

MoE(Mixture of Experts)架构十年演进(2015–2025)

MoE&#xff08;Mixture of Experts&#xff09;架构十年演进&#xff08;2015–2025&#xff09; 一句话总论&#xff1a; 2015年MoE还是“理论复苏小规模手工专家路由”的学术时代&#xff0c;2025年已进化成“万亿级多模态VLA动态MoE意图级自适应专家量子加速自进化全域具身…

作者头像 李华
网站建设 2026/5/13 15:41:39

Infra十年演进(2015–2025)

Infra十年演进&#xff08;2015–2025&#xff09; 一句话总论&#xff1a; 2015年Infra还是“单机多卡手工脚本本地集群”的原始时代&#xff0c;2025年已进化成“万亿级多模态VLA大模型云原生训练量子加速自进化全域弹性无服务器秒级自愈分布式”的终极AI基础设施&#xff0c…

作者头像 李华