产品开发01-socket编程
实验目的:
- 根据提供的TCP协议代码完成具有相关功能的UDP代码的编写,并成功实现功能
- 客户端要求:
(1) 客户端可以多次向服务器发送数据;
(2) 客户端能够接收服务器端发送数据,并回显正确; - 服务器端要求:
(1) 运行端口可配置
(2) 将客户端发送来的消息正确显示,并将该消息发送给客户端;
(3) 支持多个客户端同时发送
TCP代码如下:
server:
#include<stdlib.h>
#include<stdio.h>
#include<sys/socket.h>
#include<netdb.h>
#include<string.h>
#include<unistd.h>
#include<netinet/in.h>
#include<arpa/inet.h>
#define PORT 8900
int main(int argc,char** argv)
{
struct sockaddr_in server;
struct sockaddr_in client;
int len;
int port;
int listend;
int connectd;
int sendnum;
int opt;
int recvnum;
char send_buf[2048];
char recv_buf[2048];
//if (2==argc)
port= PORT;
memset(send_buf,0,2048);
memset(recv_buf,0,2048);
opt = SO_REUSEADDR;
if (-1==(listend=socket(AF_INET,SOCK_STREAM,0)))
{
perror("create listen socket error\n");
exit(1);
}
setsockopt(listend,SOL_SOCKET,SO_REUSEADDR,&opt,sizeof(opt));
#ifdef DEBUG
printf("the listen id is %d\n",listend);
#endif
memset(&server,0,sizeof(struct sockaddr_in));
server.sin_family = AF_INET;
server.sin_addr.s_addr = htonl(INADDR_ANY);
server.sin_port = htons(port);
if (-1==bind(listend,(struct sockaddr *)&server,sizeof(struct sockaddr)))
{
perror("bind error\n");
exit(1);
}
if (-1==listen(listend,5))
{
perror("listen error\n");
exit(1);
}
while (1)
{
if (-1==(connectd=accept(listend,(struct sockaddr*)&client,&len)))
{
perror("create connect socket error\n");
continue;
}
#ifdef DEBUG
printf("the connect id is %d",connect);
printf("the client ip addr is %s",inet_ntoa(client.sin_addr));
#endif
sendnum = sprintf(send_buf,"hello,the guest from %s\n",inet_ntoa(client.sin_addr));
if ( 0 >send(connectd,send_buf,sendnum,0))
{
perror("send error\n");
close(connectd);
continue;
}
#ifdef DEBUG
printf("the send num is %d",sendnum);
printf("the client ip addr is %s",inet_ntoa(client.sin_addr));
#endif
if (0>(recvnum = recv(connectd,recv_buf,sizeof(recv_buf),0)))
{
perror("recv error\n");
close(connectd);
continue;
}
recv_buf[recvnum]='\0';
printf ("the message from the client is: %s\n",recv_buf);
if (0==strcmp(recv_buf,"quit"))
{
perror("the client break the server process\n");
close(connectd);
break;
}
sendnum = sprintf(send_buf,"byebye,the guest from %s\n",inet_ntoa(client.sin_addr));
send(connectd,send_buf,sendnum,0);
close(connectd);
continue;
}
close(listend);
return 0;
}
client:
点击查看代码
#include<stdlib.h>
#include<stdio.h>
#include<sys/socket.h>
#include<netdb.h>
#include<string.h>
#include<unistd.h>
#include<netinet/in.h>
#include<arpa/inet.h>
#define PORT 8900
void print_usage(char * cmd)
{
fprintf(stderr," %s usage:\n",cmd);
fprintf(stderr,"%s IP_Addr [port]\n",cmd);
}
int main(int argc,char** argv)
{
struct sockaddr_in server;
int ret;
int len;
int port;
int sockfd;
int sendnum;
int recvnum;
char send_buf[2048];
char recv_buf[2048];
if ((2>argc)|| (argc >3))
{
print_usage(argv[0]);
exit(1);
}
if (3==argc)
{
port = atoi(argv[2]);
}
if (-1==(sockfd=socket(AF_INET,SOCK_STREAM,0)))
{
perror("can not create socket\n");
exit(1);
}
memset(&server,0,sizeof(struct sockaddr_in));
server.sin_family = AF_INET;
server.sin_addr.s_addr = inet_addr(argv[1]);
server.sin_port = htons(port);
if (0>(ret=connect(sockfd,(struct sockaddr*)&server,sizeof(struct sockaddr))))
{
perror("connect error");
close(sockfd);
exit(1);
}
//memset(send_buf,0,2048);
//memset(recv_buf,0,2048);
printf("what words do you want to tell to server:\n");
//gets(send_buf);
fgets(send_buf,2048,stdin);
#ifdef DEBUG
printf("%s\n",send_buf);
#endif
//sprintf(send_buf,"i am lg,thank for your servering\n");
if (0>(len=send(sockfd,send_buf,strlen(send_buf),0)))
{
perror("send data error\n");
close(sockfd);
exit(1);
}
if (0>(len=recv(sockfd,recv_buf,2048,0)))
{
perror("recv data error\n");
close(sockfd);
exit(1);
}
recv_buf[len]='\0';
printf("the message from the server is:%s\n",recv_buf);
close(sockfd);
}
相关代码已上传:https://github.com/doooraaAA/-
实验步骤:
该实验过程中,我们可以将已有的TCP代码与PPT进行结合阅读可看出每一部分具体的作用:
创建套接字;绑定套接字;进入监听模式;一直循环:等待请求,接收请求建立连接,读写数据,终止连接
同样,再根据提示对UDP代码进行修改:
再搜索得到sendto函数与recvfrom函数的具体使用方法,对listen函数与accept函数进行删除并修改剩余部分的代码:
如在UDPserver中:recvfrom函数与sendto函数都需要加入两个额外的参数
可实现显示客户发送来的消息并将消息发送给客户
使用#define PORT 8900可以让使用者直接更改端口号,易于维护,实现运行端口可配置的功能
而客户端中:
使用while(1)可以实现客户端重复发送信息的功能,并且能够接收来自服务端的信息,并显示
实验结果:
客户端可多次发送数据:
同时支持多个客户端:
客户端2:
并且可以看到两者都能看到服务端消息
服务器端:
可以接收到多个客户端信息,并显示正确,并且显示客户端的ip。
客户端可实现多次发送信息,接收服务端数据并回显正确
服务端可实现运行端口可配置,显示客户端消息、ip并将消息发送给客户端,支持多个客户端同时发送
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 分享一个免费、快速、无限量使用的满血 DeepSeek R1 模型,支持深度思考和联网搜索!
· 使用C#创建一个MCP客户端
· 基于 Docker 搭建 FRP 内网穿透开源项目(很简单哒)
· ollama系列1:轻松3步本地部署deepseek,普通电脑可用
· 按钮权限的设计及实现