【Linux取经路】网络套接字编程——TCP篇

在这里插入图片描述

文章目录

  • 前言
  • 十、Tcp Server 端代码
    • 10.1 socket、bind
    • 10.1 listen——监听一个套接字
    • 10.2 accept——获取一个新连接
    • 10.3 read——从套接字中读取数据
    • 10.4 write——向套接字中进行写入
    • 10.5 Tcp Service 端完整代码(单进程版)
    • 10.6 Tcp Server 端代码(多进程版)
    • 10.7 Tcp Server 端代码(多线程版)
    • 10.8 Tcp Server 端完整代码(线程池版)——英汉字典
    • 10.9 setsockopt——设置套接字描述符的属性
  • 十一、Tcp Client 端代码
    • 11.1 connect——向服务端发起连接
    • 11.2 write、read——向服务器发送数据、从服务器接收数据
    • 11.3 Tcp Client 端完整代码
  • 十二、基于 TCP 协议的客户端/服务器程序的一般流程
  • 十三、结语

前言

本篇文章接上一篇【Linux取经路】网络套接字编程——UDP篇,所以目录号从十开始。

十、Tcp Server 端代码

10.1 socket、bind

listensockfd_ = socket(AF_INET, SOCK_STREAM, 0);
if (listensockfd_ < 0)
{
    lg(Fatal, "socket create error, errno: %d, error message: %s", errno, strerror(errno));
    exit(SOCKET_ERR);
}
lg(Info, "socket creat successful, listensockfd_: %d", listensockfd_);

// 2. bind
struct sockaddr_in local;
memset(&local, 0, sizeof(local));
local.sin_family = AF_INET;
local.sin_port = htons(port_);
// inet_aton(ip_.c_str(), &(local.sin_addr)); // 将 ip 地址转化成网络序列
local.sin_addr.s_addr = INADDR_ANY;

socklen_t len = sizeof(local);
int ret = bind(listensockfd_, (struct sockaddr *)&local, len);

if (ret < 0)
{
    lg(Fatal, "bind error, errno: %d, error message: %s", errno, strerror(errno));
    exit(BIND_ERR);
}
lg(Info, "socket bind successful");

TCP 这里的创建套接字、绑定代码和 UDP 中的一样。

10.1 listen——监听一个套接字

#include <sys/types.h>          /* See NOTES */
#include <sys/socket.h>

int listen(int sockfd, int backlog);

绑定完之后需要将监听套接字设为监听状态。

// 3. listen
// TCP 是面向连接的,服务器一般是比较“被动的”,在没有客户端访问的时候就只能干等,不能退出,服务器需要一直处于一种,一直在等待连接到来的状态
int n = listen(listensockfd_, backlog);
if (n < 0)
{
    lg(Fatal, "bind error, errno: %d, error message: %s", errno, strerror(errno));
    exit(LISTEN_ERR);
}
lg(Info, "socket listen successful");

10.2 accept——获取一个新连接

#include <sys/types.h>          /* See NOTES */
#include <sys/socket.h>

int accept(int sockfd, struct sockaddr *addr, socklen_t *addrlen);

该函数参数和 recvfrom 函数的参数一样,后面俩输出型参数用来获取 client 端的套接字信息。该函数的返回值是一个文件描述符,它和 socket 函数返回的文件描述符有什么区别呢?socket 函数返回的这个套接字用来绑定、监听和获取新的连接,一般把这个套接字叫做监听套接字,它的主要任务就是用来获取新连接,可以把它看成拉客的。accept 函数返回的套接字才是后面进行网络通信时使用的套接字,一个服务器进程,它的监听套接字一般就只有一个,而用来通信的套接字可能会存在多个。

for (;;)
{
    // 4. 获取新连接
    lg(Info, "server is running...");
    struct sockaddr_in client;
    socklen_t len;
    int sockfd = accept(listensockfd_, (struct sockaddr *)&client, &len);
    if (sockfd < 0)
    {
        lg(Warning, "bind error, errno: %d, error message: %s", errno, strerror(errno));
        continue;
    }
    uint16_t client_port = ntohs(client.sin_port);
    char client_ip[32];
    inet_ntop(AF_INET, &(client.sin_addr), client_ip, sizeof(client_ip));
    lg(Info, "get a new link, sockfd: %d, clientip: %s, clientport: %d", sockfd, client_ip, client_port);

    // 5. 根据新连接来进行通信
    // version 1
    Server(sockfd, client_ip, client_port);

    sleep(1);
}

可以在本本机使用 telnet 127.0.0.1 9999 (其中 9999 是端口号,是自定义的)指令来验证服务器是否能成功获取一个连接。

10.3 read——从套接字中读取数据

因为 TCP 是面向字节流的,所以可以直接使用 read 系统调用去读取数据。

void Server(int sockfd, const std::string &client_ip, const uint16_t &client_port)
{
    char buffer[num];
    while(true)
    {
        memset(buffer, 0, sizeof(buffer));
        ssize_t n = read(sockfd, buffer, sizeof(buffer));
        if(n > 0)
        {
            buffer[n] = 0;
            std::cout << "client asy# <b>" << buffer << std::endl;
            std::string echo_string = "tcpserver echo@ ";
            echo_string += buffer;

            int n = write(sockfd_, echo_string.c_str(), echo_string.size());
            if (n < 0)
            {
                lg(Warning, "write error, errno: %d, strerror: %d", errno, strerror(errno));
            }
        }
        else if(n == 0)
        {
            lg(Info, "%s-%d Client quit... sockfd: %d", client_ip.c_str(), client_port, sockfd);
            break;
        }
        else
        {
            lg(Warning, "read err:, errno: %d, errstr: %s", errno, strerror(errno));
            break;
        }
    }
}

如果客户端退出了,那么 read 会读到0,此时需要把之前 accept 返回的 sockfd 关闭,防止误操作,造成意想不到的结果。

10.4 write——向套接字中进行写入

同理,向套接字中进行写入时,直接使用 write 系统调用即可。服务端在收到客户端的数据后,先进行加工处理,然后再进行写入,上面 if(n > 0) 后面就是写入的代码。唯一需要注意的就是,如果在写入前或者正在写入的过程中,client 端退出了,此时客户端与服务器之间的连接就断了,此时客户端如果进行写入操作可能会导致整个服务端崩掉。这和管道类似,读端关闭,写端继续写,操作系统会给写端发送 13 号信号,将写端 kill 调,为了避免这种情况,我们需要在服务器启动的时候将 13 号新号进行捕捉。

10.5 Tcp Service 端完整代码(单进程版)

#pragma once
#include <sys/types.h>
#include <sys/socket.h>
#include <string.h>
#include <iostream>
#include <arpa/inet.h>
#include <netinet/in.h>
#include "log.hpp"
#include <string>
#include <unistd.h>

extern Log lg;

#define num 1024

const int defaultfd = -1;
const std::string defaultip = "0.0.0.0";
const int backlog = 10; // 一般不要设置的太大

enum
{
    SOCKET_ERR = 1,
    BIND_ERR,
    LISTEN_ERR
};

class TcpServer
{
private:
    void Server(int sockfd, const std::string &client_ip, const uint16_t &client_port)
    {
        char buffer[num];
        while(true)
        {
            memset(buffer, 0, sizeof(buffer));
            ssize_t n = read(sockfd, buffer, sizeof(buffer));
            if(n > 0)
            {
                buffer[n] = 0;
                std::cout << "client asy# <b>" << buffer << std::endl;
                std::string echo_string = "tcpserver echo@ ";
                echo_string += buffer;

                write(sockfd, echo_string.c_str(), echo_string.size()); 
            }
            else if(n == 0)
            {
                lg(Info, "%s-%d Client quit... sockfd: %d", client_ip.c_str(), client_port, sockfd);
                break;
            }
            else
            {
                lg(Warning, "read err:, errno: %d, errstr: %s", errno, strerror(errno));
                break;
            }
        }
    }

public:
    TcpServer(const uint16_t &port, const std::string &ip = defaultip)
        : listensockfd_(defaultfd), port_(port), ip_(ip)
    {
    }

    void Init()
    {
        // 1. 创建套接字
        listensockfd_ = socket(AF_INET, SOCK_STREAM, 0);
        if (listensockfd_ < 0)
        {
            lg(Fatal, "socket create error, errno: %d, error message: %s", errno, strerror(errno));
            exit(SOCKET_ERR);
        }
        lg(Info, "socket creat successful, listensockfd_: %d", listensockfd_);

        // 2. bind
        struct sockaddr_in local;
        memset(&local, 0, sizeof(local));
        local.sin_family = AF_INET;
        local.sin_port = htons(port_);
        // inet_aton(ip_.c_str(), &(local.sin_addr)); // 将 ip 地址转化成网络序列
        local.sin_addr.s_addr = INADDR_ANY;

        socklen_t len = sizeof(local);
        int ret = bind(listensockfd_, (struct sockaddr *)&local, len);

        if (ret < 0)
        {
            lg(Fatal, "bind error, errno: %d, error message: %s", errno, strerror(errno));
            exit(BIND_ERR);
        }
        lg(Info, "socket bind successful");

        // 3. listen
        // TCP 是面向连接的,服务器一般是比较“被动的”,在没有客户端访问的时候就只能干等,不能退出,服务器需要一直处于一种,一直在等待连接到来的状态
        int n = listen(listensockfd_, backlog);
        if (n < 0)
        {
            lg(Fatal, "bind error, errno: %d, error message: %s", errno, strerror(errno));
            exit(LISTEN_ERR);
        }
        lg(Info, "socket listen successful");
    }

    void Start()
    {
        for (;;)
        {
            // 4. 获取新连接
            lg(Info, "server is running...");
            struct sockaddr_in client;
            socklen_t len;
            int sockfd = accept(listensockfd_, (struct sockaddr *)&client, &len);
            if (sockfd < 0)
            {
                lg(Warning, "bind error, errno: %d, error message: %s", errno, strerror(errno));
                continue;
            }
            uint16_t client_port = ntohs(client.sin_port);
            char client_ip[32];
            inet_ntop(AF_INET, &(client.sin_addr), client_ip, sizeof(client_ip));
            lg(Info, "get a new link, sockfd: %d, clientip: %s, clientport: %d", sockfd, client_ip, client_port);

            // 5. 根据新连接来进行通信
            // version 1
            Server(sockfd, client_ip, client_port);
            close(sockfd);
            sleep(1);
        }
    }

    ~TcpServer()
    {
        close(listensockfd_);
    }

private:
    int listensockfd_; //
    uint16_t port_;    // 端口号
    std::string ip_;   // ip
};

因为服务端是单进程版,所以一但一个 client 与 服务端建立了连接,那么其它的客户端在前一个 client 退出之前就无法再与客户端建立连接,这意味着,服务端同时只能服务一个 client 端。这显然是不符合实际应用场景的。

单进程版 TcpServer

10.6 Tcp Server 端代码(多进程版)

void Start()
{
    for (;;)
    {
        // 4. 获取新连接
        lg(Info, "server is running...");
        struct sockaddr_in client;
        socklen_t len;
        int sockfd = accept(listensockfd_, (struct sockaddr *)&client, &len);
        if (sockfd < 0)
        {
            lg(Warning, "bind error, errno: %d, error message: %s", errno, strerror(errno));
            continue;
        }
        uint16_t client_port = ntohs(client.sin_port);
        char client_ip[32];
        inet_ntop(AF_INET, &(client.sin_addr), client_ip, sizeof(client_ip));
        lg(Info, "get a new link, sockfd: %d, clientip: %s, clientport: %d", sockfd, client_ip, client_port);

        // 5. 根据新连接来进行通信
        // version 1——单进程版
        // Server(sockfd, client_ip, client_port);
        // close(sockfd);
        // sleep(1);

        // version 2——多进程版
        pid_t id = fork();
        if (id == 0)
        {
            // 子进程
            close(listensockfd_); // 子进程不关心这个 socket
            if(fork() > 0) exit(0); // 子进程再创建子进程,然后让子进程直接退出,由孙子进程执行后续代码
            Server(sockfd, client_ip, client_port);
            close(sockfd); // 子进程处理完后将套接字信息关闭
            exit(0); // 因为子进程退出了,所以孙子进程就会被系统领养,最终执行完毕,会被系统直接回收
        }
        close(sockfd); // 子进程已经把当前父进程中 accept 获得的文件描述符继承下去了,所以父进程已经不关心该文件描述符了,可以直接关闭掉,不然到时候父进程中存在大量被打开的文件

        // 父进程
        int rid =  waitpid(id, nullptr, 0);

    }
}

相较于单进程版,多进程的变化是在 Start 中,来一个 client 连接请求,通过创建子进程,让该子进程去为这个 client 端进行服务(获取该 client 端的数据、对数据加工处理、将处理完的数据发送给 client 端),父进程继续去 accept 获取新连接。这里需要注意:子进程是会继承父进程中的所有文件描述符,对子进程来说 listensockfd_ 是没有用的,因为子进程不去拉客(获取新连接),因此子进程代码中第一件事就是将 listensockfd_ 这个文件描述符进行关闭。对父进程来说,获取到一个新连接后会为该连接创建一个新的文件描述符,这个文件描述符被子进程继承下去后父进程就不在关心了,因为父进程要去继续拉客(获取新连接),而不是为当前获取到的连接提供服务。其次父进程需要等待子进程,如果是 waitpid(id, nullptr, 0); 这种等待,那和单进程版的,没有任何区别,因为这是阻塞等待,只有等子进程退出了父进程才能继续执行。解决该问题的办法有很多:上面代码中,采用的是让子进程继续创建孙子进程,然后让孙子进程去提供服务,子进程直接退出,这时,父进程就立刻能将子进程回收,孙子进程会被操作系统领养,最终执行完毕,会被操作系统直接回收。这里再提供两种解决方案:

  • 用一个数据将创建的子进程的 pid 管理起来,然后 waitpid(id, nullptr, WNOHANG); 或者直接 waitpid(-1, nullptr, WNOHANG);非阻塞等待任意子进程。
  • 父进程中对 SIGCHLD 进行捕捉,在捕捉方法里面对退出的子进程进行回收;或者父进程中直接对 SIFCHLD 信号进行 SIG_IGN,此时子进程在退出时会自动被清理掉,不会产生僵尸进程。

多进程版 TcpServer

10.7 Tcp Server 端代码(多线程版)

创建一个新进程相较于创建一个新线程是很“重”的,所以下面引入多线程版。一个进程打开的文件描述符是被所有子线程共享的,子线程可以根据文件描述符去对文件进行读写操作。

// 线程函数的参数
class ThreadData
{
public:
    ThreadData(int sockfd, const std::string &client_ip, const uint16_t &client_port, TcpServer *tcpserver)
        : sockfd_(sockfd), client_ip_(client_ip), client_port_(client_port), tcpserver_(tcpserver)
    {
    }

    ~ThreadData()
    {
    }

public:
    int sockfd_;            // 文件描述符
    std::string client_ip_; // 客户端 ip
    uint16_t client_port_;  // 客户端 port
    TcpServer *tcpserver_;  // 一个 tcpserver 进程
};
static void *Rountine(void *args)
{
    pthread_detach(pthread_self()); // 让子线程分离,这样在主线程中不用进行 join 了
    ThreadData *td = static_cast<ThreadData *>(args);

    td->tcpserver_->Server(td->sockfd_, td->client_ip_, td->client_port_);
    close(td->sockfd_);
}

void Start()
{
    for (;;)
    {
        // 4. 获取新连接
        lg(Info, "server is running...");
        struct sockaddr_in client;
        socklen_t len;
        int sockfd = accept(listensockfd_, (struct sockaddr *)&client, &len);
        if (sockfd < 0)
        {
            lg(Warning, "bind error, errno: %d, error message: %s", errno, strerror(errno));
            continue;
        }
        uint16_t client_port = ntohs(client.sin_port);
        char client_ip[32];
        inet_ntop(AF_INET, &(client.sin_addr), client_ip, sizeof(client_ip));
        lg(Info, "get a new link, sockfd: %d, clientip: %s, clientport: %d", sockfd, client_ip, client_port);

        // 5. 根据新连接来进行通信

        // version 3——多线程版
        ThreadData *td = new ThreadData(sockfd, client_ip, client_port, this);
        pthread_t tid;
        pthread_create(&tid, nullptr, Rountine, td);
    }
}

注意要把线程函数设置成 static

多线程版 TcpServer

10.8 Tcp Server 端完整代码(线程池版)——英汉字典

上面三个版本的服务端都提供的是常服务,什么事常服务呢?就是只要一个 client 发起了连接,服务端这边就一直 while 循环式的为该客户端提供服务,直到客户端退出,这样实际也是不合理的,正确做法是,服务端在收到一个连接请求后,一次处理完后就应该返回了。所以这里我们引入线程池,将一次客户端的连接当做一个任务,将该任务放到线程池里面去,让线程池里面的线程为这一次连接服务(获取客户端发来的数据、对数据加工处理、将结果返回给客户端)。

// TcpServer.hpp——服务端
#pragma once
#include <sys/types.h>
#include <sys/socket.h>
#include <string.h>
#include <iostream>
#include <arpa/inet.h>
#include <netinet/in.h>
#include "log.hpp"
#include <string>
#include <unistd.h>
#include <sys/wait.h>
#include <pthread.h>
#include "ThreadPool.hpp"
#include "Task.h"

extern Log lg;

#define num 1024

const int defaultfd = -1;
const std::string defaultip = "0.0.0.0";
const int backlog = 10; // 一般不要设置的太大

enum
{
    SOCKET_ERR = 1,
    BIND_ERR,
    LISTEN_ERR
};

class TcpServer;

class ThreadData
{
public:
    ThreadData(int sockfd, const std::string &client_ip, const uint16_t &client_port, TcpServer *tcpserver)
        : sockfd_(sockfd), client_ip_(client_ip), client_port_(client_port), tcpserver_(tcpserver)
    {
    }

    ~ThreadData()
    {
    }

public:
    int sockfd_;            // 文件描述符
    std::string client_ip_; // 客户端 ip
    uint16_t client_port_;  // 客户端 port
    TcpServer *tcpserver_;  // 一个 tcpserver 进程
};

class TcpServer
{
public:
    TcpServer(const uint16_t &port, const std::string &ip = defaultip)
        : listensockfd_(defaultfd), port_(port), ip_(ip)
    {
    }

    void Init()
    {
        // 1. 创建套接字
        listensockfd_ = socket(AF_INET, SOCK_STREAM, 0);
        if (listensockfd_ < 0)
        {
            lg(Fatal, "socket create error, errno: %d, error message: %s", errno, strerror(errno));
            exit(SOCKET_ERR);
        }
        lg(Info, "socket creat successful, listensockfd_: %d", listensockfd_);

        // 2. bind
        struct sockaddr_in local;
        memset(&local, 0, sizeof(local));
        local.sin_family = AF_INET;
        local.sin_port = htons(port_);
        // inet_aton(ip_.c_str(), &(local.sin_addr)); // 将 ip 地址转化成网络序列
        local.sin_addr.s_addr = INADDR_ANY;

        socklen_t len = sizeof(local);
        int ret = bind(listensockfd_, (struct sockaddr *)&local, len);

        if (ret < 0)
        {
            lg(Fatal, "bind error, errno: %d, error message: %s", errno, strerror(errno));
            exit(BIND_ERR);
        }
        lg(Info, "socket bind successful");

        // 3. listen
        // TCP 是面向连接的,服务器一般是比较“被动的”,在没有客户端访问的时候就只能干等,不能退出,服务器需要一直处于一种,一直在等待连接到来的状态
        int n = listen(listensockfd_, backlog);
        if (n < 0)
        {
            lg(Fatal, "bind error, errno: %d, error message: %s", errno, strerror(errno));
            exit(LISTEN_ERR);
        }
        lg(Info, "socket listen successful");
    }

    void Start()
    {
        signal(SIGPIPE, SIG_IGN);
        ThreadPool<Task>::GetInstance()->start();
        for (;;)
        {
            // 4. 获取新连接
            lg(Info, "server is running...");
            struct sockaddr_in client;
            socklen_t len;
            int sockfd = accept(listensockfd_, (struct sockaddr *)&client, &len);
            if (sockfd < 0)
            {
                lg(Warning, "bind error, errno: %d, error message: %s", errno, strerror(errno));
                continue;
            }
            uint16_t client_port = ntohs(client.sin_port);
            char client_ip[32];
            inet_ntop(AF_INET, &(client.sin_addr), client_ip, sizeof(client_ip));
            lg(Info, "get a new link, sockfd: %d, clientip: %s, clientport: %d", sockfd, client_ip, client_port);

            // 5. 根据新连接来进行通信
            // version 4——线程池版本
            Task t(sockfd, client_ip, client_port);
            ThreadPool<Task>::GetInstance()->push(t);
        }
    }

    ~TcpServer()
    {
        close(listensockfd_);
    }

private:
    int listensockfd_; //
    uint16_t port_;    // 端口号
    std::string ip_;   // ip
};
// ThreadPool.hpp——线程池
#pragma once

#include <pthread.h>
#include <vector>
#include <string>
#include <queue>
#include <unistd.h>
#include <unordered_map>

struct ThreadInfo
{
    pthread_t tid_;    // 线程 ID
    std::string name_; // 线程的名字
};

template <class T>
class ThreadPool
{
    static const int defaultnum = 5;

public:
    void Lock()
    {
        pthread_mutex_lock(&mutex_);
    }

    void Unlock()
    {
        pthread_mutex_unlock(&mutex_);
    }

    void Weakup()
    {
        pthread_cond_signal(&cond_);
    }

    void Sleep()
    {
        pthread_cond_wait(&cond_, &mutex_);
    }

    bool IsTaskQueueEmpty()
    {
        return tasks_.empty();
    }

    T PopTasks()
    {
        T task = tasks_.front();
        tasks_.pop();
        return task;
    }

    const std::string &GetThreadName(pthread_t tid)
    {
        return um_[tid];
    }

public:
    static void *Routine(void *args)
    {
        ThreadPool *tp = static_cast<ThreadPool *>(args);
        std::string name = tp->GetThreadName(pthread_self());
        while (true)
        {
            tp->Lock();
            while (tp->IsTaskQueueEmpty())
            {
                tp->Sleep();
            }

            T task = tp->PopTasks();
            tp->Unlock();
            task(); // 此时这个任务已经属于该线程私有的了,所以对任务的处理工作可以在解锁之后进行
            // printf("%s is running----%s\n", name.c_str(), task.result_to_string().c_str());
        }
    }

    void start()
    {
        for (int i = 0; i < thread_num_; i++)
        {
            threads_[i].name_ = "Thread-" + std::to_string(i);
            pthread_create(&(threads_[i].tid_), nullptr, Routine, this);
            um_[threads_[i].tid_] = threads_[i].name_;
        }
    }

    void push(const T &task)
    {
        Lock();
        tasks_.push(task);
        Weakup();
        Unlock();
    }

    static ThreadPool<T> *GetInstance() // 指正通过该接口获取一个单例对象
    {
        if (ptp_ == nullptr)
        {
            pthread_mutex_lock(&smutex_);
            if (ptp_ == nullptr)
            {
                lg(Info, "singleton creat done first!");
                ptp_ = new ThreadPool<T>();
            }
            pthread_mutex_unlock(&smutex_);
        }
        return ptp_;
    }

private:
    ThreadPool(int thread_num = defaultnum)
        : threads_(thread_num), thread_num_(thread_num)
    {
        pthread_mutex_init(&mutex_, nullptr);
        pthread_cond_init(&cond_, nullptr);
    }
    ~ThreadPool()
    {
        pthread_mutex_destroy(&mutex_);
        pthread_cond_destroy(&cond_);
    }
    ThreadPool(const ThreadPool<T> &tp) = delete;
    const ThreadPool<T> &operator=(const ThreadPool<T> &tp) = delete;

private:
    std::vector<ThreadInfo> threads_;               // 一批线程
    int thread_num_;                                // 线程池中的线程的数量
    std::queue<T> tasks_;                           // 任务
    pthread_mutex_t mutex_;                         // 定义一把锁,让所有线程保持互斥
    pthread_cond_t cond_;                           // 定义一个条件变量,让线程同步
    std::unordered_map<pthread_t, std::string> um_; // 一个 map 用来快速查找一个线程的name

    static ThreadPool<T> *ptp_;     // 单例
    static pthread_mutex_t smutex_; // 定义一把静态的锁
};

template <class T>
ThreadPool<T> *ThreadPool<T>::ptp_ = nullptr;

template <class T>
pthread_mutex_t ThreadPool<T>::smutex_ = PTHREAD_MUTEX_INITIALIZER;
// Task.hpp——任务
#include <iostream>
#include <string>
#include "log.hpp"  
#include "ECDictionary.hpp"

ECDictionary ecd;

extern Log lg;

#define num 1024



class Task
{
public:
    Task(int sockfd, const std::string &client_ip, const uint16_t &client_port)
        : sockfd_(sockfd), client_ip_(client_ip), client_port_(client_port)
    {
    }

    void run()
    {
        char buffer[num];
        memset(buffer, 0, sizeof(buffer));
        ssize_t n = read(sockfd_, buffer, sizeof(buffer));
        if (n > 0)
        {
            buffer[n] = 0;
            std::cout << "client asy# <b>" << buffer << std::endl;
            std::string echo_string = "tcpserver echo@ ";
            echo_string += ecd.GetTranslation(buffer);

            write(sockfd_, echo_string.c_str(), echo_string.size());
        }
        else if (n == 0)
        {
            lg(Info, "%s-%d Client quit... sockfd: %d", client_ip_.c_str(), client_port_, sockfd_);
        }
        else
        {
            lg(Warning, "read err:, errno: %d, errstr: %s", errno, strerror(errno));
        }

        close(sockfd_);
    }

    void operator()()
    {
        run();
    }
private:
    int sockfd_;
    std::string client_ip_;
    uint16_t client_port_;
};
// ECDictionary.hpp——字典
#include <fstream>
#include <string>
#include <string.h>
#include <unordered_map>
#include "log.hpp"

#define SEP ":"

extern Log lg;

const std::string path = "./dictionary.txt";

class ECDictionary
{
private:
    static void Split(const std::string &line, std::string *english, std::string *chinese)
    {
        size_t pos = line.find(SEP);
        if(pos == std::string::npos) return;
        *english = line.substr(0, pos);
        *chinese = line.substr(pos+1);
        return;
    }

public:
    ECDictionary()
    {
        std::ifstream di(path.c_str());
        if(!di.is_open())
        {
            lg(Fatal, "open %s Dictionary errror, errno: %d, errstr: %s", path.c_str(), errno, strerror(errno));
            exit(1);
        }

        std::string line;
        while(std::getline(di, line))
        {
            std::string english, chinese;
            Split(line, &english, &chinese);
            dir_.insert({english, chinese});
        }
        di.close();
    }

    std::string GetTranslation(const std::string &english)
    {
        auto it = dir_.find(english);
        if(it == dir_.end()) return "There is no such word";

        return it->second;
    }

private:
    std::unordered_map<std::string, std::string> dir_;
};

线程池版 TcpServer

由于我们当前是短连接,即对于一个客户端发起的一次连接请求,服务器只处理一次,就把相应的文件描述符进行了关闭。所以一个客户端启动后只能翻译一次。

10.9 setsockopt——设置套接字描述符的属性

#include <sys/types.h>          /* See NOTES */
#include <sys/socket.h>
int setsockopt(int sockfd, int level, int optname, const void *optval, socklen_t optlen);
int opt = 1;
setsockopt(listensockfd_, SOL_SOCKET, SO_REUSEADDR|SO_REUSEPORT, &opt, sizeof(opt)); // 防止偶发性的服务器无法进行立即重启

十一、Tcp Client 端代码

和 Udp Client 端一样,Tcp Client 在创建完套接字之后,不需要我们自己编写代码去显示的绑定 ip 地址和端口号,而是在 connect 的时候,由操作系统随机的为我们绑定端口号。

11.1 connect——向服务端发起连接

#include <sys/types.h>          /* See NOTES */
#include <sys/socket.h>

int connect(int sockfd, const struct sockaddr *addr, socklen_t addrlen);

后两个参数用来标识要连接服务端的套接字信息。返回值:只要绑定成功并且获取连接成功,0被返回;失败-1被返回。

11.2 write、read——向服务器发送数据、从服务器接收数据

和服务端一样,客户端也是通过 writeread 接口来发送数据和读取数据。

11.3 Tcp Client 端完整代码

因为服务端对一个连接只会处理一次,然后就会把服务端对应的文件描述符给关闭了,此时相当于连接就断了,所以客户端需要循环去发起连接请求,并且每连接一次,得到服务器端的结果后需要把创建的套接字进行关闭,因为服务端已经关了,你不关还想干啥??所以客户端需要每次重新去创建套接字,发起连接。

#include <iostream>
#include <sys/types.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <netinet/in.h>
#include <string.h>
#include <unistd.h>
#include <string>

using namespace std;

void Usage(const char *command)
{
    std::cout << "\n\tUsage: " << command << " port[1024+]" << std::endl;
}

int main(int argc, char *argv[])
{
    if (argc != 3)
    {
        Usage(argv[0]);
        exit(1);
    }

    string server_ip = argv[1];
    uint16_t server_port = stoi(argv[2]);

    sockaddr_in server;
    memset(&server, 0, sizeof(server));
    server.sin_family = AF_INET;
    server.sin_port = htons(server_port);
    inet_pton(AF_INET, server_ip.c_str(), &(server.sin_addr));
    socklen_t len = sizeof(server);

    while (true)
    {
        int cnt = 5; // 重连次数
        bool needreconnect = true;
        // 1. 创建套接字
        int sockfd = socket(AF_INET, SOCK_STREAM, 0);
        if (sockfd < 0)
        {
            perror("sockfd");
            exit(2);
        }

        do // 重连模块
        {
            // 2. bind —— 客户端不需要我们显示去 bind,是由操作系统帮我们进行绑定的

            // 3. 发起连接

            int result = connect(sockfd, (struct sockaddr *)&server, len);
            if (result < 0)
            {
                needreconnect = true;
                std::cerr << "The client is reconnecting cnt: " << cnt << std::endl;
                cnt--;
                sleep(2);
            }
            else
            {
                needreconnect = false; // 连接成功,不需要重连
                break;
            }
        } while (cnt && needreconnect);

        if (cnt == 0)
        {
            std::cerr << "The user is disconnected..." << std::endl;
            break;
        }

        // 4. 向server发送消息
        string message;
        std::cout << "Please Enter: ";
        getline(cin, message);

        ssize_t n = write(sockfd, message.c_str(), message.size());
        if (n < 0)
        {
            std::cerr << "write err, errno: " << errno << ", strerror: " << strerror(errno) << std::endl;
            continue;
        }

        char buffer[4096];
        int ret = read(sockfd, buffer, sizeof(buffer));
        if (ret > 0)
        {
            buffer[ret] = 0;
            printf("%s\n", buffer);
        }
        close(sockfd);
    }
    return 0;
}

image-20240320161648846

image-20240320162932709

十二、基于 TCP 协议的客户端/服务器程序的一般流程

image-20240321114434136

服务器初始化

  • 调用 socket ,创建文件描述符

  • 调用 bind ,将当前的文件描述符和 ip、port 绑定在一起;如果端口号被其他进程占用就会 bind 失败。

  • 调用 listen ,声明当前这个文件描述符作为一个服务器的文件描述符,为后面的 accept 做好准备

  • 调用 accept,并阻塞,等待客户端连接过来。

建立连接的过程

  • 调用 socket ,创建文件描述符

  • 调用 connect,向服务器发起连接请求

  • connect 会发出 SYN 段并阻塞,等待服务器应答(第一次)

  • 服务器收到客户端的 SYN ,会应答一个 SYN-ACK 段表示“同意建立连接”(第二次)

  • 客户端收到 SYN-ACK后会从 connect() 返回,同时应答一个 ACK 段(第三次)

这个建立连接的过程,通常成为三次握手

数据传输的过程

  • 建立连接后,TCP 协议提供全双工的通信服务;所谓全双工,就是在同一条连接中,同一时刻,通信双方可以同时写数据。当创建一个 TCP 套接字的时候,我们虽然只获得了一个文件描述符,但是在底层操作系统会为给文件描述符创建两个缓冲区,一个是发送缓冲区,另一个接收缓冲区;相对的概念是半双工,同一条连接在同一时刻,只能由一方来写数据。

  • 服务器从 accept 返回后,立刻调用 read()socket 就像读管道一样,如果没有数据到达就阻塞等待。

  • 这时客户端调用 write() 发送请求给服务器,服务器收到后从 read() 返回,对客户端的请求进行处理,在此期间客户端调用 read() 阻塞等待服务器的应答。

  • 服务器调用 write() 将处理结果发回给客户端,再次调用 read() 阻塞等待下一条请求。

  • 客户端收到后从 read() 返回,发送下一条请求,如此循环下去。

断开连接的过程

  • 如果客户端没有更多的请求了,就调用 close() 关闭连接,客户端会向服务器发送 FIN段(第一次)

  • 此时服务器收到 FIN 后,会回应一个 ACK ,同时 read() 会返回 0(第二次)

  • read() 返回之后,服务器就知道客户端关闭了连接,也调用 close() 关闭连接,这个时候服务器会向客户端发送一个 FIN (第三次)

  • 客户端收到 FIN,再返回一个 ACK 给服务器(第四次)

image-20240322172927274

同一时刻可能会有多个客户端来连接服务器,所以在服务端一定会存在多个连接,服务端是需要把这多个连接管理起来的,管理一定是通过先描述再组织的方式。我们调用 write 接口本质上是将数据写入到 TCP 的发送缓冲区,至于写入的数据什么时候发送到对端的接收缓冲区,以及发送多少,出错了怎么办,完全是由 TCP 协议自主控制的, 所以 TCP 也叫做传输控制协议。我们使用的这些接口 writereadsendtorecvfrom 本质上都是在用户和内核之间进行数据拷贝。用户把数据交给操作系统,本质上就是把数据交给操作系统。

image-20240322180614425

对于读方来说,可能 TCP 将发送方多次 write 的数据一次打包发了过来,接收方通过read 一次就多上来了多份数据,也可能 TCP 将发送方一次 write 的数据,分了好几次发送,read 每次读取到的都是部分数据。所以对接收方来说,它读上来的数据就存在很大的不确定性,它不清楚自己当前读到的数据是否是一个完整数据,也不清楚自己是否读到了多份数据。为了解决这个问题,就需要引入协议的概念,接收方可以根据协议来确定自己是否读取到了一份完整的数据。

十三、结语

今天的分享到这里就结束啦!如果觉得文章还不错的话,可以三连支持一下,春人的主页还有很多有趣的文章,欢迎小伙伴们前去点评,您的支持就是春人前进的动力!

在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/685555.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

【ZYNQ】CPU 私有定时器

Zynq 的每个 Cortex-A9 处理器都有自己的专用 32 位定时器和 32 位看门狗定时器&#xff0c;两个处理器共享一个全局 64 位定时器&#xff0c;这些计时器的时钟频率始终为 CPU 频率的 1/2。本文主要介绍 Zynq 芯片 CPU 私有定时器的工作特性&#xff0c;以及私有定时器的基本使…

(面试官问我微服务与naocs的使用我回答了如下,面试官让我回去等通知)微服务拆分与nacos的配置使用

微服务架构 正常的小项目就是所有的功能集成在一个模块中&#xff0c;这样代码之间不仅非常耦合&#xff0c;而且修改处理的时候也非常的麻烦&#xff0c;应对高并发时也不好处理&#xff0c;所以 我们可以使用微服务架构&#xff0c;对项目进行模块之间的拆分&#xff0c;每一…

精选网络安全书单:打造数字世界的钢铁长城!

目录 1.前言 2.书单推荐 2.1. 《内网渗透实战攻略》 2.2. 《Kali Linux高级渗透测试&#xff08;原书第4版&#xff09;》 2.3. 《CTF那些事儿》 2.4. 《权限提升技术&#xff1a;攻防实战与技巧》 2.5. 《数字政府网络安全合规性建设指南&#xff1a;密码应用与数据安全…

C++:红黑树

红黑树的概念 红黑树是一种二叉搜索树&#xff0c;但在每个结点上增加一个存储位表示结点的颜色&#xff0c;可以是Red或 Black。 通过对任何一条从根到叶子的路径上各个结点着色方式的限制&#xff0c;红黑树确保没有一条路 径会比其他路径长出两倍&#xff0c;因而是接近平衡…

ArrayList——简单洗牌算法

特殊语法介绍&#xff1a; List<List<E>> 该语法情况比较特殊&#xff0c;相当于一个“二维数组”存着一个个线性表的结构&#xff0c;如图&#xff1a; 该语法的灵活性强&#xff0c;可适用于多种类型和多种情况。接下来就使用该语法来实现一个简单的洗牌操作。…

halo进阶-主题插件使用

开始捣鼓捣鼓halo&#xff0c;换换主题&#xff0c;加个页面 可参考&#xff1a;Halo 文档 安装/更新主题 主题如同壁纸&#xff0c;萝卜青菜各有所爱&#xff0c;大家按需更换即可&#xff1b; Halo好在一键更换主题&#xff0c;炒鸡方便。 安装/更新插件 此插件还扩展了插件…

CR80通用清洁卡:证卡打印机、ATM机、POS机、读卡器等卡片设备清洁维护的好助手!

随着科技的进步&#xff0c;ATM机、POS终端、门禁系统、证卡打印机、读卡器等卡片设备在我们的日常生活中扮演着越来越重要的角色&#xff0c;些设备在长时间使用和环境因素的影响下&#xff0c;容易积聚油脂、灰尘和其他污染物&#xff0c;从而对其性能和功能产生负面影响。 深…

Xcode 打包报错Command PhaseScriptExecution failed with a nonzero exit code

解决办法: 1、在Xcode项目中 Pods -> Targets Support Files -> Pods-项目名 -> Pods-项目名-frameworks 中(大约在第44行) 加上 -f 2、CocoaPods版本太旧了,可以尝试升级CocoaPods版本 使用sudo gem update cocoapods更新cocoapods&#xff0c;问题将在1.12.1版本已…

力扣2968.执行操作使频率分数最大

力扣2968.执行操作使频率分数最大 方法一&#xff1a;滑窗 前缀和 求前缀和数组s 求一个数组补齐到中位数的差值 枚举右端点 class Solution {public:int maxFrequencyScore(vector<int>& nums, long long k) {int res0,n nums.size();sort(nums.begin(),nums…

单元测试AIR原则:提升代码质量的秘密武器

文章目录 引言一、AIR原则1. Automatic&#xff08;自动化&#xff09;2. Independent&#xff08;独立性&#xff09;3. Repeatable&#xff08;可重复性&#xff09; 二、Automatic&#xff08;自动化&#xff09;三、Independent&#xff08;独立性&#xff09;四、Repeatab…

GIF录屏工具Gif123 v3.3.0单文件

软件介绍 GIF的优势是小、轻、快&#xff0c;适合时间短、画面小、需要嵌入其他页面&#xff0c;打开就自动循环播放的动画。Gif123可录制合成鼠标轨迹,可调整鼠标指针大小,可在设置中打开鼠标指针高亮光圈功能,高亮光圈可跟随鼠标移动以指示鼠标位置。软件极其简单&#xff0…

C语言实现教学计划编制问题,Dev C++编译器下可运行(240606最新更新)

背景&#xff1a; 问题描述 大学的每个专业都要编制教学计划。假设任何专业都有固定的学习年限&#xff0c;每学年含两学期&#xff0c; 每学期的时间长度和学分上限都相等。每个专业开设的课程都是确定的&#xff0c;而且课程的开设时间的安排必须满足先修关系。每个课程的先…

树形表/树形数据接口的开发

数据表格式 需要返回的json格式 点击查看json数据 [{"childrenTreeNodes" : [{"childrenTreeNodes" : null,"id" : "1-1-1","isLeaf" : null,"isShow" : null,"label" : "HTML/CSS","na…

Spark MLlib 机器学习详解

目录 &#x1f349;引言 &#x1f349;Spark MLlib 简介 &#x1f348; 主要特点 &#x1f348;常见应用场景 &#x1f349;安装与配置 &#x1f349;数据处理与准备 &#x1f348;加载数据 &#x1f348;数据预处理 &#x1f349;分类模型 &#x1f348;逻辑回归 &a…

【成品设计】基于NB模块智能烟感系统设计

《基于NB模块智能烟感系统》 整体功能&#xff1a; 所需器件&#xff1a; 选用STM32F103为主控&#xff0c;选用DS18B20温度传感器和MQ烟雾传感器作为数据采集点&#xff0c; 采用0.96寸屏显实时显示采集到的温度、烟雾等数据&#xff0c;用蜂鸣器作为报警装置。 通过有人物联…

OpenCV的“画笔”功能

类似于画图软件的自由笔刷功能&#xff0c;当按住鼠标左键&#xff0c;在屏幕上画出连续的线条。 定义函数&#xff1a; import cv2 import numpy as np# 初始化参数 drawing False # 鼠标左键按下时为True ix, iy -1, -1 # 鼠标初始位置# 鼠标回调函数 def mouse_paint(…

初学者如何对大模型进行微调?

粗略地说&#xff0c;大模型训练有四个主要阶段&#xff1a;预训练、有监督微调、奖励建模、强化学习。 预训练消耗的时间占据了整个训练pipeline的99%&#xff0c;其他三个阶段是微调阶段&#xff0c;更多地遵循少量 GPU 和数小时或数天的路线。预训练对于算力和数据的要求非…

冯喜运:6.7今日黄金原油行情分析及独家操作策略

【黄金消息面分析】&#xff1a;周三&#xff08;6月5日&#xff09;&#xff0c;金价回升逾1.2%&#xff0c;收盘报每盎司2,355.49美元&#xff0c;全面收复前一交易日的跌幅。周三当天前公布的美国民间就业数据弱于预期&#xff0c;增强了美联储将在今年晚些时候降息的预期&a…

06- 数组的基础知识详细讲解

06- 数组的基础知识详细讲解 一、基本概念 一次性定义多个相同类型的变量&#xff0c;并且给它们分配一片连续的内存。 int arr[5];1.1 初始化 只有在定义的时候赋值&#xff0c;才可以称为初始化。数组只有在初始化的时候才可以统一赋值。 以下是一些示例规则&#xff1a; …

AI全自动批量剪辑软件,一天剪辑3000条原创视频不是梦【剪辑软件+全套教程】

创建一个AI全自动批量剪辑软件的简易程序涉及较为复杂的视频处理和机器学习技术&#xff0c;而且由于这是一个相当高级的任务&#xff0c;通常需要大量的代码以及深度学习框架支持。不过&#xff0c;我可以为您提供一个非常基础版本的程序示例&#xff0c;它会用Python的moviep…