linux基础 ---TCP《十》

本文介绍网络应用程序设计模式,涵盖数据链路层以太网帧协议、网络层IP协议、传输层TCP/UDP协议。详细阐述TCP协议面向连接的特点,包括三次握手、数据确认与重传,还介绍了UDP协议面向无连接的特性。此外,讲解了socket编程、TCP四次挥手、滑动窗口及多进程、多线程并发服务器等内容。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

网络应用程序设计模式:
在这里插入图片描述
在这里插入图片描述
协议格式:
数据链路层协议=>以太网帧协议
在这里插入图片描述
以太网帧协议需要知道对方的mac地址才能给对方发送数据。
不知道对方的mac地址的时候填ff:ff:ff:ff:ff:ff
在这里插入图片描述
在这里插入图片描述
IP协议=》网络层
在这里插入图片描述
DNS - 服务器
在这里插入图片描述
TCP/UDP=》传输层协议
在这里插入图片描述
在这里插入图片描述TCP协议:

  • 面向连接的安全的流式传输协议
  • 连接的时候,进行三次握手
  • 数据发送的时候,会进行数据的确认
  • 数据丢失的之后,会进行数据的重传
    在这里插入图片描述
    UDP协议
  • 面向无连接的不安全的报式传输
  • 连接的时候不会握手
  • 数据发送出去之后就不管了
    在这里插入图片描述

数据的发送和接收:
在这里插入图片描述
socket编程:
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
发送数据的原理:先写到本机的写缓冲区然后操作系统自动发送
在这里插入图片描述
默认是阻塞的:阻塞指的是文件描述符对应的设备文件的属性
服务器端的步骤:
在这里插入图片描述
在这里插入图片描述
htonl()小端转大端 =》主机字节序转网络字节序
INADDR_ANY会自动适配本机的IP地址

客户端的步骤:
客户端的端口是隐式绑定的,占用空闲的端口号就可以
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
TCP的三次握手:
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
TCP四次挥手的过程:
在这里插入图片描述
第四次挥手:
客户端:

  • 发送:ACK+确认编号

服务器端:

  • 数据检测

TCP连接、传输和断开过程
在这里插入图片描述
TCP-滑动窗口
在这里插入图片描述
TCP多进程并发服务器:
在这里插入图片描述
在这里插入图片描述
TCP通信多进程代码:
在这里插入图片描述
waitpid()会暂时停止目前进程的执行,直到有信号来到或子进程结束。
WNOHANG 如果没有任何已经结束的子进程则马上返回,不予以等待。
使用while来轮询
一直判断子进程是否退出,如果没有退出就会一直循环。
在这里插入图片描述

/*server.c*/
#include <stdio.h>
#include <string.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <signal.h>
#include <sys/wait.h>
#include <ctype.h>
#include <unistd.h>
#include <stdlib.h>
#include "wrap.h"
#include <string.h>
#include<errno.h>
#define MAXLINE 8192
#define SERV_PORT 8000

void do_sigchild(int num)
{
    while (waitpid(0, NULL, WNOHANG) > 0);
}

int main(int argc, char *argv[])
{
    struct sockaddr_in servaddr, cliaddr;
    socklen_t cliaddr_len;
    int listenfd, connfd;
    char buf[MAXLINE];
    char str[INET_ADDRSTRLEN];
    int i, n;
    pid_t pid;

    //临时屏蔽sigchld信号
    sigset_t myset;
    sigemptyset(&myset);
    sigaddset(&myset, SIGCHLD);
    // 自定义信号集 -》 内核阻塞信号集
    sigprocmask(SIG_BLOCK, &myset, NULL);


    listenfd = Socket(AF_INET, SOCK_STREAM, 0);

    int opt = 1;
    // 设置端口复用
    setsockopt(listenfd, SOL_SOCKET, SO_REUSEADDR, &opt, sizeof(opt));

    bzero(&servaddr, sizeof(servaddr));
    servaddr.sin_family = AF_INET;
    servaddr.sin_addr.s_addr = htonl(INADDR_ANY);
    int port =atoi(argv[1]);
    servaddr.sin_port = htons(port);

    Bind(listenfd, (struct sockaddr *)&servaddr, sizeof(servaddr));

    Listen(listenfd, 20);

    printf("Accepting connections ...\n");
    while (1)
    {
        cliaddr_len = sizeof(cliaddr);
        connfd = Accept(listenfd, (struct sockaddr *)&cliaddr, &cliaddr_len);
        while(connfd == -1 && errno == EINTR)
        {
            connfd = Accept(listenfd, (struct sockaddr *)&cliaddr, &cliaddr_len);
        }
        // 有新的连接则创建一个进程
        pid = fork();
        if (pid == 0)
        {
            Close(listenfd);
            while (1)
            {
                n = Read(connfd, buf, MAXLINE);
                if (n == 0)
                {
                    printf("the other side has been closed.\n");
                    break;
                }
                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(STDOUT_FILENO, buf, n);
                Write(connfd, buf, n);
            }
            Close(connfd);
            return 0;
        }
        else if (pid > 0)
        {
            struct sigaction act;
            act.sa_flags = 0;
            act.sa_handler = do_sigchild;
            sigemptyset(&act.sa_mask);
            sigaction(SIGCHLD, &act, NULL);
            // 解除对sigchld信号的屏蔽
            sigprocmask(SIG_UNBLOCK, &myset, NULL);

            Close(connfd);
        }
        else
        {
            perr_exit("fork");
        }
    }
    return 0;
}

/* client.c */
#include <stdio.h>
#include <string.h>
#include <unistd.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <stdlib.h>
#include "wrap.h"

#define MAXLINE 8192
#define SERV_PORT 8000

int main(int argc, char *argv[])
{
    struct sockaddr_in servaddr;
    char buf[MAXLINE];
    int sockfd, n;

    sockfd = Socket(AF_INET, SOCK_STREAM, 0);

    bzero(&servaddr, sizeof(servaddr));
    servaddr.sin_family = AF_INET;
    inet_pton(AF_INET, "127.0.0.1", &servaddr.sin_addr);
    int port =atoi(argv[1]);
    servaddr.sin_port = htons(port);

    Connect(sockfd, (struct sockaddr *)&servaddr, sizeof(servaddr));

    while (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");
            break;
        }
        else
            Write(STDOUT_FILENO, buf, n);
    }

    Close(sockfd);

    return 0;
}

TCP多线程并发服务器:
在这里插入图片描述

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值