客户端和服务器实现全双工通信(基于线程)

http://blog.sina.com.cn/s/blog_87766e4a0100yp7a.html

1、 基于多线程实现一个服务器和一个客户端实现全双工通信

服务器端创建两个线程:一个用于接收客户端发送过来的信息;一个用来给客户端发送信息。

客户端也创建两个线程:一个用于接收服务器端发送过来的信息;一个用来给服务器端发送信息。

 

2、 示例代码

服务器端的代码client.c

#include <sys/types.h>

#include <sys/socket.h>

#include <stdio.h>

#include <stdlib.h>

#include <errno.h>

#include <string.h>

#include <unistd.h>

#include <netinet/in.h>

#define SERVPORT 3333

#define BACKLOG 10                 //请求队列中的最大请求数

#define MAX_CONNECTED_NO 10  //最大是128台电脑连接

#define MAXDATASIZE 100

 

void pthread_recv(void *arg);

void Pthread_send(void * arg);

 

int main()

{

         struct sockaddr_in server_sockaddr,client_sockaddr;

         int sockfd,client_fd,sin_size,ret;

         pthread_t id1,id2;

         sin_size =sizeof(struct sockaddr);

         //struct sockaddr_in server_sockaddr,client_sockaddr;

         //建立一个socket连接

     //目前都是AF_INET(ipv4);SOCK_STREAM(TCP),如果是UDP,则SOCK_DGRAM

         if((sockfd = socket(AF_INET,SOCK_STREAM,0))==-1){

                   perror("socket");

                   exit(1);

         }

         printf("socket success!,sockfd=%d\n",sockfd);

         //bind用于本地IP地址和端口的绑定

         server_sockaddr.sin_family=AF_INET;

         server_sockaddr.sin_port=htons(SERVPORT);

         server_sockaddr.sin_addr.s_addr=INADDR_ANY;         //htonl(INADDR_ANY)

     bzero(&(server_sockaddr.sin_zero),8);

         //把本端的信息绑定

         if(bind(sockfd,(struct sockaddr *)&server_sockaddr,sizeof(struct sockaddr))==-1)

      {              

           perror("bind");

               exit(1);

         }

         printf("bind success!\n");

         if(listen(sockfd,BACKLOG)==-1)

     {

                   perror("listen");

                   exit(1);

         }

         printf("listening....\n");

     //接受并保存客户端的信息

         if((client_fd=accept(sockfd,(struct sockaddr *)&client_sockaddr,&sin_size))==-1)

     {                

perror("accept");

                exit(1);

         }

         printf("client_fd=%d",client_fd);

         pthread_create(&id1,NULL,(void *) pthread_recv,(void*)&client_fd);

         if(ret != 0)

                   perror("pthread_recv creat");

         ret=pthread_create(&id2,NULL,(void *)Pthread_send,(void*)&client_fd);

         if(ret != 0)

                   perror("Pthread_send creat");

         pthread_join(id1,NULL);

         pthread_join(id2,NULL);

         close(sockfd);

}

 

//服务器发送线程

void Pthread_send(void * arg)

{

         int buf_len, client_fd,sendbytes;

         char buf[MAXDATASIZE];

         client_fd = *((int *)arg);

         printf("server send: ");

         while(1)

         {       

                   fgets(buf,MAXDATASIZE,stdin);

                   buf_len =sizeof(buf);

                   if((sendbytes = send(client_fd ,buf,buf_len,0))==-1)

                   {

                            perror("send");

                            exit(1);

                   }

                   if(!strncmp(buf,"end",3))         //只比较字符串的前三个字符

                            break;

                   sleep(2);

         }

                           

}

 

//服务器接收数据线程

void pthread_recv(void *arg)

{

         int buf_len, client_fd,recvbytes;

         char buf[MAXDATASIZE];

         client_fd = *((int *)arg);

     //接收服务端发来的信息

         while(1)

         {

                   if((recvbytes = recv(client_fd ,buf,MAXDATASIZE,0))==-1)

                   {

                            perror("recv");

                            exit(1);

                   }

                   else

                   {

                            buf[recvbytes]=0;

                            printf("server received data from server is :%s\n",buf);

                            if(!strncmp(buf,"end",3))         //只比较字符串前三个字符

                                     break;

                   }

                   sleep(1);

         }

}

 

客户端的代码:

#include <stdio.h>

#include <stdlib.h>

#include <errno.h>

#include <string.h>

#include <netdb.h>

#include <sys/types.h>

#include <netinet/in.h>

#include <sys/socket.h>

 

#define SERVPORT 3333

#define MAXDATASIZE 100

 

void pthread_recv(void *arg);

void Pthread_send(void * arg);

 

main(int argc,char *argv[]){

         int sockfd,sin_size,ret;

         struct hostent *host;

         pthread_t id1,id2;

         struct sockaddr_in serv_addr;         //套接字的地址结构

         struct sockaddr_in server_sockaddr;

        

         if(argc < 2)

    {

                   fprintf(stderr,"Please enter the server's hostname!\n");

                   exit(1);

         }       

     //AF_INET:使用的是IPV4

     //SOCK_STREAM:流式套接字

         if((sockfd=socket(AF_INET,SOCK_STREAM,0))==-1)

     {

          perror("socket");

                 exit(1);

         }

         printf("sockfd=%d\n",sockfd);

         serv_addr.sin_family=AF_INET;

         serv_addr.sin_port=htons(SERVPORT);

     //serv_addr.sin_addr=*((struct in_addr *)host->h_addr);

     //host->h_addr强制类型转换;建议直接使用下面的方法

         serv_addr.sin_addr.s_addr=inet_addr(argv[1]);   //"127.0.0.1");       Internet地址

         bzero(&(serv_addr.sin_zero),8);

     //连接

         if(connect(sockfd,(struct sockaddr *)&serv_addr,sizeof(struct sockaddr))==-1)

         {

                   perror("connect");

                   exit(1);

         }

         ret=pthread_create(&id1,NULL,(void *)Pthread_send,(void*)&sockfd);

         if(ret != 0)

                   perror("Pthread_send creat");

         pthread_create(&id2,NULL,(void *) pthread_recv,(void*)&sockfd);

         if(ret != 0)

                   perror("pthread_recv creat");

         pthread_join(id1,NULL);

         pthread_join(id2,NULL);

         close(sockfd);

}

 

//客户端发送数据线程

void Pthread_send(void * arg)

{

         int buf_len, sockfd,sendbytes;

         char buf[MAXDATASIZE];

         sockfd = *((int *)arg);

         printf("client send: ");

         while(1)

         {       

                   fgets(buf,MAXDATASIZE,stdin);

                   buf_len =sizeof(buf);

                   if((sendbytes = send(sockfd,buf,buf_len,0))==-1)

                   {

                            perror("send");

                            exit(1);

                   }

                   if(!strncmp(buf,"end",3))         //只比较字符串的前三个字符

                            break;

                   sleep(2);

         }                   

}

 

//客户端接收数据线程

void pthread_recv(void *arg)

{

         int buf_len, sockfd,recvbytes;

         char buf[MAXDATASIZE];

         sockfd = *((int *)arg);

     //接收服务端发来的信息

         while(1)

         {

                   if((recvbytes = recv(sockfd,buf,MAXDATASIZE,0))==-1)

                   {

                            perror("recv");

                            exit(1);

                   }

                   else

                   {

                            buf[recvbytes]=0;

                           printf("client received data from server is :%s\n",buf);

                            if(!strncmp(buf,"end",3))         //只比较字符串的前三个字符

                                     break;

                   }

                   sleep(1);

         }

}

转载于:https://www.cnblogs.com/feng9exe/p/7612543.html

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 要使用UDP实现全双工聊天,你需要创建两个程序:一个用于发送消息,另一个用于接收消息。以下是一个简单的示例程序: 1.发送消息程序: ```c #include <stdio.h> #include <stdlib.h> #include <string.h> #include <sys/socket.h> #include <netinet/in.h> #include <arpa/inet.h> #include <unistd.h> #define PORT 8888 #define BUF_SIZE 1024 int main(int argc, char *argv[]) { int sockfd; struct sockaddr_in server_addr; char buf[BUF_SIZE]; if (argc != 2) { printf("Usage: %s <IP Address>\n", argv[0]); exit(1); } sockfd = socket(AF_INET, SOCK_DGRAM, 0); if (sockfd < 0) { perror("socket"); exit(1); } memset(&server_addr, 0, sizeof(server_addr)); server_addr.sin_family = AF_INET; server_addr.sin_port = htons(PORT); server_addr.sin_addr.s_addr = inet_addr(argv[1]); while (1) { printf("Please enter message: "); fgets(buf, BUF_SIZE, stdin); sendto(sockfd, buf, strlen(buf), 0, (struct sockaddr *)&server_addr, sizeof(server_addr)); } close(sockfd); return 0; } ``` 2.接收消息程序: ```c #include <stdio.h> #include <stdlib.h> #include <string.h> #include <sys/socket.h> #include <netinet/in.h> #include <arpa/inet.h> #include <unistd.h> #define PORT 8888 #define BUF_SIZE 1024 int main() { int sockfd; struct sockaddr_in server_addr, client_addr; socklen_t addr_len; char buf[BUF_SIZE]; sockfd = socket(AF_INET, SOCK_DGRAM, 0); if (sockfd < 0) { perror("socket"); exit(1); } memset(&server_addr, 0, sizeof(server_addr)); server_addr.sin_family = AF_INET; server_addr.sin_port = htons(PORT); server_addr.sin_addr.s_addr = htonl(INADDR_ANY); if (bind(sockfd, (struct sockaddr *)&server_addr, sizeof(server_addr)) < 0) { perror("bind"); exit(1); } while (1) { addr_len = sizeof(client_addr); memset(buf, 0, BUF_SIZE); recvfrom(sockfd, buf, BUF_SIZE, 0, (struct sockaddr *)&client_addr, &addr_len); printf("Received message from %s:%d: %s", inet_ntoa(client_addr.sin_addr), ntohs(client_addr.sin_port), buf); } close(sockfd); return 0; } ``` 在其中一个终中运行接收消息程序,另一个终中运行发送消息程序。这样就可以实现基本的全双工聊天了。 ### 回答2: C语言使用UDP协议实现全双工聊天可以通过以下步骤完成: 1. 创建两个套接字,一个用于发送消息,一个用于接收消息。其中一个套接字作为客户,另一个作为服务器。 2. 客户首先获取目标主机的IP地址和口号,然后将其作为另一方的套接字地址。 3. 客户通过发送消息套接字,将用户输入的消息发送给服务器。 4. 服务器接收到消息后,将其打印出来,并通过发送消息套接字将响应消息发送给客户。 5. 客户接收服务器的响应消息,并将其打印出来。 6. 重复以上步骤,实现全双工聊天。客户服务器都可以同时发送和接收消息。 需要注意的是,UDP是一种无连接协议,因此在发送消息之前,需要确保发送方和接收方已经建立好通信连接。为了实现全双工聊天,可以使用多线程的方式,分别处理发送消息和接收消息的功能。 在发送消息时,需要使用sendto()函数将消息发送给目标地址。接收消息时,可以使用recvfrom()函数从套接字中获取消息。 综上所述,通过实现以上步骤,我们可以使用C语言和UDP协议实现全双工聊天。这种方式可以实现用户之间的实时交流,适用于一对一或者一对多的聊天场景。 ### 回答3: 全双工聊天是指两个用户之间可以同时发送和接收消息。C语言可以使用UDP协议来实现全双工聊天。 在UDP通信中,每个用户都可以作为服务器客户同时存在。用户A和用户B可以同时发送和接收消息。 首先,用户A和用户B都需要创建一个套接字(socket)。套接字用于连接用户与网络。接下来,用户A将套接字绑定到一个指定的IP地址和口上,并等待接收用户B的消息。 用户B同样需要创建一个套接字,并将其绑定到一个指定的IP地址和口上。然后,用户B可以使用套接字向用户A发送消息。 用户A和用户B都可以通过套接字来发送和接收消息。发送消息的函数可以使用sendto()函数来发送消息到指定的IP地址和口。接收消息的函数可以使用recvfrom()函数来接收消息。 通过以上步骤,用户A和用户B可以在UDP协议下实现全双工聊天。他们可以同时发送和接收消息,实现实时的双向沟通。 需要注意的是,UDP协议是一种不可靠的协议,数据包可能会丢失或乱序。为了保证可靠性,可以在发送方和接收方添加一些数据校验或重传机制。 对于使用C语言实现全双工聊天,可以使用sockaddr_in结构体来存储IP地址和口号。使用socket()函数创建套接字,并使用bind()函数将套接字绑定到指定的IP地址和口上。然后,可以使用sendto()函数发送消息,recvfrom()函数接收消息。 通过以上步骤,就可以使用C语言使用UDP实现全双工聊天。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值