zoukankan      html  css  js  c++  java
  • 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
    ",recvbuf);
    	    if(memcmp(recvbuf,close_str,1)==0){
    	    	printf("接受到关闭信号~
    ");
    		sendto(sock, "close ok~", 9, 0,
                       (struct sockaddr *)&peeraddr, peerlen);
    		break;
                }
                sendto(sock, recvbuf, n, 0,
                       (struct sockaddr *)&peeraddr, peerlen);
                printf("回送的数据:%s
    ",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端口
    ",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
    ",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
    ",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;
    }
    
    

    运行结果:

    img
    ------------恢复内容结束------------

  • 相关阅读:
    前端脚手架的那些事儿
    CSS重置默认样式reset.css代码模板
    Web 3.0 前瞻:基于区块链的下一代浏览器
    关键词定位是网站推广的基础
    6年架构师针对web前端小白,作出的职业规划建议
    和程序员约会的优点和缺点
    如何在软件开发中避免出现漏洞
    Linux下修改时区
    前端需要掌握的Nginx知识
    Nginx入门指南
  • 原文地址:https://www.cnblogs.com/lightice/p/12726717.html
Copyright © 2011-2022 走看看