Linux下C语言实现半双工的UDP通信

------------恢复内容开始------------

Linux下C语言实现半双工的UDP通信

1、单向通信:又称为单工通信,即只能有一个方向的通信而没有反方向的交互。无线电广播或有线电广播以及电视广播就属于这种类型。

单向通信只需要一条信道,而双向交替通信或双向同时通信则都需要两条信道(每个方向各一条)。显然,双向同时通信的传输效率最高。不过应当指出,虽然电信局为打电话的用户提供了双向同时通信的信道,但有效的电话交谈一般都还是双方交替通信。当双方发生争吵时往往就是采用双向同时通信的方式。

2、半双工通信,是指数据可以沿两个方向传送.但同一时刻一个信道只允许单方向传送,因此义被称为双向交替通信,如图中(b)所示。例如,无线对讲机就是一种半双工设备,在同一时间内只允许一方讲话。

3、全双工通信,是指同时发生在两个方向上的一种数据传输方式,如图中(c)所示。电话机就是一种全双工设备,其通话双方可以同时进行对话。计算机之间的高速数据通信也是这种方式。

双向交替通信又称为半双工通信,即通信的双方都可以发送信息,但不能双方同时发送(当然也就不能同时接收)。这种通信方式是一方发送另一方接收,过一段时间后再反过来。

server.c

#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
#include<errno.h>
#include<sys/types.h>
#include<sys/socket.h>
#include<netinet/in.h>
#include<string.h>
 
#define MYPORT 8887
 
 
#define ERR_EXIT(m) \
    do { \
    perror(m); \
    exit(EXIT_FAILURE); \
    } while (0)
 
void echo_ser(int sock)
{
    char recvbuf[1024] = {0};
    struct sockaddr_in peeraddr;
    char close_str[1024]={"q"};
    socklen_t peerlen;
    int n;
    
    while (1)
    {
        
        peerlen = sizeof(peeraddr);	//获得地址长度
        memset(recvbuf, 0, sizeof(recvbuf));	//清除数组中的数据
        //4.接收数据,recvfrom返回值为接受的字节数
        n = recvfrom(sock, recvbuf, sizeof(recvbuf), 0,
                     (struct sockaddr *)&peeraddr, &peerlen);	
        if (n <= 0)
        {
            
            if (errno == EINTR)
                continue;
            
            ERR_EXIT("recvfrom error");
        }
	//5,发送数据,sendto返回值为发送的字节数,出错则返回-1
        else if(n > 0)
        {
            printf("接收到的数据:%s\n",recvbuf);
	    if(memcmp(recvbuf,close_str,1)==0){
	    	printf("接受到关闭信号~\n");
		sendto(sock, "close ok~", 9, 0,
                   (struct sockaddr *)&peeraddr, peerlen);
		break;
            }
            sendto(sock, recvbuf, n, 0,
                   (struct sockaddr *)&peeraddr, peerlen);
            printf("回送的数据:%s\n",recvbuf);
        }
    }
    close(sock);
}
 
int main(void)
{
    //1.创建套接字
    int sock;
    if ((sock = socket(PF_INET, SOCK_DGRAM, 0)) < 0)
        ERR_EXIT("socket error");
    //2.创建服务器地址和端口号并设置
    struct sockaddr_in servaddr;
    memset(&servaddr, 0, sizeof(servaddr));
    servaddr.sin_family = AF_INET;	//设置地址类型为IP
    servaddr.sin_port = htons(MYPORT);	//设置端口号
    servaddr.sin_addr.s_addr = htonl(INADDR_ANY);	//自动获取地址
    //3.绑定套接字
    printf("监听%d端口\n",MYPORT);
    if (bind(sock, (struct sockaddr *)&servaddr, sizeof(servaddr)) < 0)//绑定套接字和地址变量
        ERR_EXIT("bind error");
    
    echo_ser(sock);
    
    return 0;
}

client.c

#include <unistd.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <stdlib.h>
#include <stdio.h>
#include <errno.h>
#include <string.h>
 
#define MYPORT 8887
char* SERVERIP = "127.0.0.1";
 
#define ERR_EXIT(m) \
    do \
{ \
    perror(m); \
    exit(EXIT_FAILURE); \
    } while(0)
 
void echo_cli(int sock)
{
    //2.设置远程服务器的地址信息和端口号
    struct sockaddr_in servaddr;
    memset(&servaddr, 0, sizeof(servaddr));
    servaddr.sin_family = AF_INET;
    servaddr.sin_port = htons(MYPORT);
    servaddr.sin_addr.s_addr = inet_addr(SERVERIP);
    
    int ret;
    char sendbuf[1024] = {0};
    char recvbuf[1024] = {0};
    //3.向服务器发送数据
    while (fgets(sendbuf, sizeof(sendbuf), stdin) != NULL)	//读入字符串
    {
        
        printf("向服务器发送:%s\n",sendbuf);
		//sendto返回值为发送字节的长度,出错返回-1
        sendto(sock, sendbuf, strlen(sendbuf), 0, (struct sockaddr *)&servaddr, sizeof(servaddr));
        	//recvfrom返回值为接受的字节数
        ret = recvfrom(sock, recvbuf, sizeof(recvbuf), 0, NULL, NULL);
        if (ret == -1)
        {
            if (errno == EINTR)
                continue;
            ERR_EXIT("recvfrom");
        }
        printf("从服务器接收:%s\n",recvbuf);
        	//清除数组中的数据
        memset(sendbuf, 0, sizeof(sendbuf));
        memset(recvbuf, 0, sizeof(recvbuf));
    }
    
    close(sock);
    
    
}
 
int main(void)
{
    //1.创建套接字
    int sock;
    if ((sock = socket(PF_INET, SOCK_DGRAM, 0)) < 0)
        ERR_EXIT("socket");
    
    echo_cli(sock);
    
    return 0;
}

运行结果:

Linux下C语言实现半双工的UDP通信
------------恢复内容结束------------

上一篇:使用WINSOCK的网络通信模拟程序 (客户机端)


下一篇:Tcp Daytime获取客户端