Linux C应用编程-6-网络通信

489 篇文章 14 订阅
464 篇文章 13 订阅

1.TCP通信

1)TCP进程服务器

/* server */
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
//socket 所需的头文件
#include <sys/socket.h>
#include <netinet/in.h>

#include <arpa/inet.h>
#include <ctype.h>

//进程需要使用的头文件
#include <sys/types.h>
#include <unistd.h>
//waitpid函数需要的头文件
#include <sys/types.h>
#include <sys/wait.h>

#define MAXLINE 80
#define SERV_PORT   8000


int main(void)
{
    struct sockaddr_in servaddr, cliaddr;   //socket 地址结构体 服务端和客户端地址
    socklen_t cliaddr_len;
    int listenfd, connfd;//文件描述符
    char buf[MAXLINE];
    char str[INET_ADDRSTRLEN];
    int i,n;
    int status;
    
    pid_t pid;
    struct sigaction act_new;
    
    //创建流式套接字,即TCP socket
    listenfd = socket(AF_INET, SOCK_STREAM, 0);
    if (listenfd == -1) {
        fprintf(stderr, "create socket\n");
        exit(1);
    }
    
    //初始化socket地址
    bzero(&servaddr, sizeof(servaddr));
    servaddr.sin_family = AF_INET;//地址类型
    servaddr.sin_addr.s_addr = htonl(INADDR_ANY);//可监听任意ip地址
    servaddr.sin_port = htons(SERV_PORT);//可监听的网络端口
    //绑定地址与端口到套接字
    status = bind(listenfd, (struct sockaddr *)&servaddr, sizeof(servaddr) );
    if (status == -1) {
        fprintf(stderr, "bind\n");
        exit(1);
    }
    
    //设置socket为监听模式,最多允许有20个客户端处于连接待状态
    listen(listenfd, 20);
    if (status == -1) {
        fprintf(stderr, "listen\n");
        exit(1);
    }
    
    printf("Accepting connections ...\n");
    
    while (1) {
        cliaddr_len = sizeof(cliaddr);
        //服务端阻塞等待客户端连接
        connfd = accept(listenfd, (struct sockaddr *)&cliaddr, &cliaddr_len);
        
        //当有新的客户端连接时,fork出一个新进程来管理接收
        pid = fork();
        if (pid == -1) {
            perror("call to fork");
            exit(1);
        } else if (pid == 0) {
            //child
            close(listenfd);
            
            printf("connect to %s at PORT %d\n", inet_ntop(AF_INET, &cliaddr.sin_addr, str, sizeof(str)), ntohs(cliaddr.sin_port));
            
            while (1) {
                //通过文件描述符读取数据  阻塞等待
                n = read(connfd, buf, MAXLINE);
                if (n == 0) {
                    printf("disconnect to %s at PORT %d\n", inet_ntop(AF_INET, &cliaddr.sin_addr, str, sizeof(str)), ntohs(cliaddr.sin_port));
                    exit(0);
                }
                //客户端的端口是随机分配的
                printf("received from %s at PORT %d\n", inet_ntop(AF_INET, &cliaddr.sin_addr, str, sizeof(str)), ntohs(cliaddr.sin_port));
                
                for (i = 0; i < n; i++) {
                    buf[i] = toupper(buf[i]);//将小写变大写
                }
                //通过文件描述符写入数据
                write(connfd, buf, n);
            }
            close(connfd);
            
        } else {
            //parent
            
            close(connfd);
        }

    }
    
    close(listenfd);
}

2)TCP客户端

/* client */
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
//socket 所需的头文件
#include <sys/socket.h>
#include <netinet/in.h>

#include <arpa/inet.h>
#include <ctype.h>

#define MAXLINE 80
#define SERV_PORT   8000

//将client改为交互式输入

int main(int argc, char *argv[])
{
    struct sockaddr_in servaddr;
    char buf[MAXLINE];
    int sockfd; //文件描述符
    int n;
    
    //创建流式套接字,即TCP socket
    sockfd = socket(AF_INET, SOCK_STREAM, 0);
    if (sockfd == -1) {
        fprintf(stderr, "create socket\n");
        exit(1);
    }
    
    //初始化socket地址
    bzero(&servaddr, sizeof(servaddr));
    servaddr.sin_family = AF_INET;
    inet_pton(AF_INET, "127.0.0.1", &servaddr.sin_addr);
    servaddr.sin_port = htons(SERV_PORT);
    
    //客户端请求连接服务器
    connect(sockfd, (struct sockaddr *)&servaddr, sizeof(servaddr));
    
    while (1) {
        if (fgets(buf, MAXLINE, stdin) != NULL) {
            //通过文件描述符写入数据 
            write(sockfd, buf, strlen(buf));
            
            //通过文件描述符读取数据  阻塞等待
            n = read(sockfd, buf, MAXLINE);
            if (n == 0) {
                printf("the other side has been closed.\n");
            } else {
                printf("Response from server:\n");
                write(STDOUT_FILENO, buf, n);
            }
        }
    }

    close(sockfd);
    
    return 0;
}

【学习地址】:FFmpeg/WebRTC/RTMP/NDK/Android音视频流媒体高级开发
【文章福利】:免费领取更多音视频学习资料包、大厂面试题、技术视频和学习路线图,资料包括(C/C++,Linux,FFmpeg webRTC rtmp hls rtsp ffplay srs 等等)有需要的可以点击1079654574加群领取哦~

  

2.UDP通信

1)UDP服务器

/* server */
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
//socket 所需的头文件
#include <sys/socket.h>
#include <netinet/in.h>

#include <arpa/inet.h>
#include <ctype.h>

#define MAXLINE 80
#define SERV_PORT 8000

int main(void)
{
    struct sockaddr_in servaddr, cliaddr;   //socket 地址结构体 服务端和客户端地址
    socklen_t cliaddr_len;
    int sockfd;//文件描述符
    char buf[MAXLINE];
    char str[INET_ADDRSTRLEN];
    int i,n;
    int status;
    
    //创建块式套接字,即UDP socket
    sockfd = socket(AF_INET, SOCK_DGRAM, 0);
    if (sockfd == -1) {
        fprintf(stderr, "create socket\n");
        exit(1);
    }
    
    //初始化socket地址
    bzero(&servaddr, sizeof(servaddr));
    servaddr.sin_family = AF_INET;//地址类型
    servaddr.sin_addr.s_addr = htonl(INADDR_ANY);//可监听任意ip地址
    servaddr.sin_port = htons(SERV_PORT);//可监听的网络端口
    //绑定地址与端口到套接字
    status = bind(sockfd, (struct sockaddr *)&servaddr, sizeof(servaddr) );
    if (status == -1) {
        fprintf(stderr, "bind\n");
        exit(1);
    }
    
    printf("Accepting connections ...\n");
    
    while (1) {
        cliaddr_len = sizeof(cliaddr);
        //阻塞等待客户端数据到来
        n = recvfrom(sockfd, buf, MAXLINE, 0, (struct sockaddr *)&cliaddr, &cliaddr_len);
        if (n == -1) {
            fprintf(stderr, "recvfrom error\n");
            exit(1);
        }
        printf("received from %s at PORT %d\n", inet_ntop(AF_INET, &cliaddr.sin_addr, str, sizeof(str)), ntohs(cliaddr.sin_port));
            
        for (i = 0; i < n; i++) {
            buf[i] = toupper(buf[i]);//小写字母转大写
        }
        
        n = sendto(sockfd, buf, n, 0, (struct sockaddr *)&cliaddr, sizeof(cliaddr));
        if (n == -1) {
            fprintf(stderr, "sendto error\n");
            exit(1);
        }
    }
    
}

2)UDP客户端

/* client */
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
//socket 所需的头文件
#include <sys/socket.h>
#include <netinet/in.h>

#include <arpa/inet.h>
#include <ctype.h>

#define MAXLINE 80
#define SERV_PORT 8000

int main(int argc, char *argv[])
{
    struct sockaddr_in servaddr;
    int sockfd, n;
    char buf[MAXLINE];
    char str[INET_ADDRSTRLEN];
    socklen_t servaddr_len;
    
    //创建UDP socket
    sockfd = socket(AF_INET, SOCK_DGRAM, 0);

    bzero(&servaddr, sizeof(servaddr));
    servaddr.sin_family = AF_INET;
    inet_pton(AF_INET, "127.0.0.1", &servaddr.sin_addr);
    servaddr.sin_port = htons(SERV_PORT);
    
    while (1) {
        if (fgets(buf, MAXLINE, stdin) != NULL) {
            //通过文件描述符发送数据
            n = sendto(sockfd, buf, strlen(buf), 0, (struct sockaddr *)&servaddr, sizeof(servaddr));
            if (n == -1) {
                fprintf(stderr, "sendto error\n");
                exit(1);
            }
            
            //通过文件描述符接收数据
            n = recvfrom(sockfd, buf, MAXLINE, 0, NULL, 0);
            if (n == -1) {
                fprintf(stderr, "recvfrom error\n");
                exit(1);
            }
            
            write(STDOUT_FILENO, buf, n);

        }
    }
    close(sockfd);
    
    return 0;
}
  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值