产品开发01-socket编程

实验目的

  1. 根据提供的TCP协议代码完成具有相关功能的UDP代码的编写,并成功实现功能
  2. 客户端要求:
    (1) 客户端可以多次向服务器发送数据;
    (2) 客户端能够接收服务器端发送数据,并回显正确;
  3. 服务器端要求:
    (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并将消息发送给客户端,支持多个客户端同时发送

posted @   Orey-May  阅读(7)  评论(0编辑  收藏  举报
相关博文:
阅读排行:
· 分享一个免费、快速、无限量使用的满血 DeepSeek R1 模型,支持深度思考和联网搜索!
· 使用C#创建一个MCP客户端
· 基于 Docker 搭建 FRP 内网穿透开源项目(很简单哒)
· ollama系列1:轻松3步本地部署deepseek,普通电脑可用
· 按钮权限的设计及实现
点击右上角即可分享
微信分享提示