2022.11.22---TCP,UDP

本文提供了一个基于TCP和UDP的服务器与客户端通信的代码示例。通过详细的C语言实现,展示了如何创建服务器和客户端,进行消息收发,并处理网络通信中的常见错误。

创建基于TCP的服务器和客户端

//服务器
#include <stdio.h>
#include <sys/socket.h>
#include <sys/types.h>
#include <arpa/inet.h>
#include <netinet/in.h>
#include <string.h>
#include <unistd.h>

#define SER_PORT 7777
#define SER_IP "192.168.8.205"
#define ERR_MSG(msg) do{\
	fprintf(stderr,"%d--",__LINE__);\
	perror(msg);\
}while(0)
int main(int argc, const char *argv[])
{
	//创建套接字
	int sfd=socket(AF_INET,SOCK_STREAM,0);
	if(sfd<0)
	{
		ERR_MSG("socket");
		return -1;
	}
	//如果不加这段代码,则服务器异常退出后,会导致端口号在30s~3min内释放不出来
	//当程序结束后,想要让其他进程能够快速复用端口号,就可以使用下面的代码
	//允许端口快速被重用
	int reuse = 1;
	if(setsockopt(sfd, SOL_SOCKET, SO_REUSEADDR, &reuse ,sizeof(reuse)) < 0)
	{
		ERR_MSG("setsockopt");
		return -1;
	}
	printf("允许端口快速重用\n");
	//绑定服务器和客户端
	struct sockaddr_in sin;
	sin.sin_family=AF_INET;
	sin.sin_port=htons(SER_PORT);
	sin.sin_addr.s_addr=inet_addr(SER_IP);
	if(bind(sfd,(struct sockaddr*)&sin,sizeof(sin))<0)
	{
		ERR_MSG("bind");
		return -1;
	}
	//将套接字设置为被动监听状态listen
	if(listen(sfd,128)==-1)
	{
		ERR_MSG("listen");
		return -1;
	}
	//获取新的文件描述符
	struct sockaddr_in cin;
	socklen_t addr=sizeof(cin);
	int nfd=accept(sfd,(struct sockaddr*)&cin,&addr);
	if(nfd<0)
	{
		ERR_MSG("accept");
		return -1;
	}
	printf("%d号客户端连接成功\nIP:%s 端口号:%d\n",nfd,\
			inet_ntoa(cin.sin_addr),ntohs(cin.sin_port));
	ssize_t res=0;
	char buf[128]="";
	while(1)
	{
		//接受客户端的消息
		bzero(buf,sizeof(buf));
		res=recv(nfd,buf,sizeof(buf),0);
		if(res<0)
		{
			ERR_MSG("recv");
			return -1;
		}
		else if(res==0)
		{
			printf("%d号客户端退出\n",nfd);
			break;
		}
		printf("[%s:%d] %s\n",inet_ntoa(cin.sin_addr),\
				ntohs(cin.sin_port),buf);
		//发送消息给客户端
		strcat(buf,"^_^");
		if(send(nfd,buf,sizeof(buf),0)<0)
		{
			ERR_MSG("send");
			return -1;
		}
	}
	//关闭文件描述符
	close(sfd);
	close(nfd);
	return 0;
}
//客户端
#include <stdio.h>
#include <sys/socket.h>
#include <sys/types.h>
#include <arpa/inet.h>
#include <netinet/in.h>
#include <string.h>
#include <unistd.h>

#define SER_PORT 7777
#define SER_IP "192.168.8.205"
#define ERR_MSG(msg) do{\
	fprintf(stderr,"%d--",__LINE__);\
	perror(msg);\
}while(0)
int main(int argc, const char *argv[])
{
	//创建套接字
	int sfd=socket(AF_INET,SOCK_STREAM,0);
	if(sfd<0)
	{
		ERR_MSG("socket");
		return -1;
	}
	//连接服务器
	struct sockaddr_in sin;
	sin.sin_family=AF_INET;
	sin.sin_port=htons(SER_PORT);
	sin.sin_addr.s_addr=inet_addr(SER_IP);
	if(connect(sfd,(struct sockaddr*)&sin,sizeof(sin))<0)
	{
		ERR_MSG("connect");
		return -1;
	}
	printf("服务器连接成功\n");
	ssize_t res=0;
	char buf[128]="";
	while(1)
	{
		//发送消息给服务器
		bzero(buf,sizeof(buf));
		printf("input:");
		fgets(buf,sizeof(buf),stdin);
		buf[strlen(buf)-1]=0;
		if(send(sfd,buf,sizeof(buf),0)<0)
		{
			ERR_MSG("send");
			return -1;
		}
		//接收服务器的消息
		res=recv(sfd,buf,sizeof(buf),0);
		if(res<0)
		{
			ERR_MSG("recv");
			return -1;
		}
		else if(res==0)
		{
			break;
		}
		printf("%s\n",buf);
	}
	//关闭文件描述符
	close(sfd);
	return 0;
}

创建基于UDP的服务器和客户端

//服务器
#include <stdio.h>
#include <string.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <netinet/in.h>
#include <unistd.h>
#define ERR(msg) do{fprintf(stderr,"%d--",__LINE__);\
	perror(msg);}while(0)
#define SER_PORT 7777
#define SER_IP "192.168.8.205"
int main(int argc, const char *argv[])
{
	//创建套接字
	int sfd=socket(AF_INET,SOCK_DGRAM,0);
	if(sfd<0)
	{
		ERR("socket");
		return -1;
	}
	//绑定服务器的IP和端口号
	struct sockaddr_in sin={AF_INET,\
		htons(SER_PORT),inet_addr(SER_IP)};
	if(bind(sfd,(struct sockaddr*)&sin,sizeof(sin))<0)
	{
		ERR("bind");
		return -1;
	}
	printf("服务器已就绪\n");
	char buf[128]="";
	struct sockaddr_in cin;
	socklen_t addr=sizeof(cin);
	while(1)
	{
		//接收客户端的消息
		bzero(buf,sizeof(buf));
		if(recvfrom(sfd,buf,sizeof(buf),0,(struct sockaddr*)&cin,&addr)<0)
		{
			ERR("recvfrom");
			return -1;
		}
		printf("[%s:%d] %s\n",inet_ntoa(cin.sin_addr),ntohs(cin.sin_port),buf);
		//发送消息给客户端
		strcat(buf,"^_^");
		if(sendto(sfd,buf,sizeof(buf),0,(struct sockaddr*)&cin,sizeof(cin))<0)
		{
			ERR("sendto");
			return -1;
		}
	}
	//关闭文件描述符
	close(sfd);
	return 0;
}
//客户端
#include <stdio.h>
#include <string.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <netinet/in.h>
#include <unistd.h>
#define ERR(msg) do{fprintf(stderr,"%d--",__LINE__);\
	perror(msg);}while(0)
#define SER_PORT 7777
#define SER_IP "192.168.8.205"
int main(int argc, const char *argv[])
{
	//创建套接字
	int sfd=socket(AF_INET,SOCK_DGRAM,0);
	if(sfd<0)
	{
		ERR("socket");
		return -1;
	}
	//获取服务器的IP和端口号
	struct sockaddr_in sin={AF_INET,\
		htons(SER_PORT),inet_addr(SER_IP)};
	char buf[128]="";
	socklen_t addr=sizeof(sin);
	while(1)
	{
		//发送消息给服务器
		bzero(buf,sizeof(buf));
		printf("input:");
		fgets(buf,sizeof(buf),stdin);
		buf[strlen(buf)-1]=0;
		if(sendto(sfd,buf,sizeof(buf),0,(struct sockaddr*)&sin,sizeof(sin))<0)
		{
			ERR("sendto");
			return -1;
		}
		//接收服务器的消息
		if(recvfrom(sfd,buf,sizeof(buf),0,(struct sockaddr*)&sin,&addr)<0)
		{
			ERR("recvfrom");
			return -1;
		}
		printf("%s\n",buf);
	}
	//关闭文件描述符
	close(sfd);
	return 0;
}

<think>嗯,用户遇到了Docker命令的错误,提示“docker: invalid reference format”,需要帮他们解决这个问题。首先,我得仔细看看他们执行的命令。用户输入的是: docker run --restart always -d -it --name srs-stack -v $HoME/data:/data (-p 2022:2022 -p2442:2443 -p 1935:1935 -p 8008:8000/udp -p 10080:10080/udp registry.cn-hangzhou.aliyuncs.com/ossrs/srs-stack:5 首先,检查命令结构是否有问题。Docker命令的选项通常应该在镜像名称之前,而用户可能在端口映射部分用了括号,这在命令行中是不允许的,因为shell会误解括号。此外,变量$HoME可能拼写错误,正确应该是$HOME,注意大小写,Linux的环境变量是区分大小写的。 接下来,看看镜像标签是否正确。用户使用的标签是“5”,可能应该是具体的版本号,比如“5”可能不完整,正确的可能是类似“5”或更详细的标签,比如“v5”或者“latest”。 然后,检查端口映射的格式是否正确。用户有的端口参数写成了-p2442:2443,缺少了空格,正确应该是-p 2442:2443。另外,UDP端口的部分是否正确,比如8008:8000/udp,这部分语法是对的,但需要确认是否需要同时映射TCPUDP,或者是否有其他配置问题。 还有,注意到命令末尾用户输入了“docker: invalid reference format”,这通常是因为镜像名称或标签不正确,或者命令中存在语法错误导致Docker无法正确解析镜像名称。例如,如果前面的参数有错误,Docker可能把后面的参数误认为是镜像的一部分。 总结可能的错误点: 1. 括号的使用:在-p参数周围用了括号,导致shell解析错误。 2. 变量名$HoME大小写错误,应为$HOME。 3. 端口映射参数缺少空格,如-p2442应改为-p 2442。 4. 镜像标签可能不完整,比如“5”可能需要更具体的标签。 5. 命令结构问题,比如选项的位置是否正确,是否在镜像名前。 需要逐步修正这些错误,然后重新运行命令。可能还需要检查是否存在其他拼写错误,比如--name后面的名称是否有特殊字符,或者卷挂载路径是否存在。 另外,用户可能在输入命令时不小心包含了其他字符,比如在命令末尾的“docker: invalid reference format”可能是错误信息的一部分,而用户可能复制粘贴时包含了错误输出到命令中,导致再次执行时出错。需要确保命令的纯净,没有混入其他文本。 最后,建议用户使用正确的命令结构,修正变量名和端口参数,移除括号,检查镜像标签,重新运行命令。</think>您遇到的错误 `docker: invalid reference format` 主要源于命令格式问题。以下是逐步分析和修复方案: --- ### **错误原因分析** 1. **环境变量错误**:`$HoME` 应为 `$HOME`(Linux 环境变量区分大小写)。 2. **括号误用**:`(-p 2022:2022 ...` 中的括号 `()` 是非法字符,Docker 命令不支持括号。 3. **端口映射格式问题**:`-p2442:2443` 缺少空格,应改为 `-p 2442:2443`。 4. **镜像标签不完整**:`registry.cn-hangzhou.aliyuncs.com/ossrs/srs-stack:5` 中的标签 `:5` 可能不完整,需确认是否存在该版本。 --- ### **修复后的命令** ```bash docker run --restart always -d -it \ --name srs-stack \ -v $HOME/data:/data \ -p 2022:2022 \ -p 2442:2443 \ -p 1935:1935 \ -p 8008:8000/udp \ -p 10080:10080/udp \ registry.cn-hangzhou.aliyuncs.com/ossrs/srs-stack:5 ``` --- ### **关键修改说明** 1. **移除非法括号**:Docker 命令中无需使用 `()`,直接列出参数即可。 2. **修正环境变量**:`$HOME` 是标准环境变量,指向用户家目录。 3. **规范端口映射**: - 格式必须为 `-p 宿主机端口:容器端口`,注意 `-p` 后需加空格。 - UDP 端口
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值