【Linux编程】TcpServer 类的设计与实现:构建高性能的 TCP 服务器(二)

news2025/1/1 11:34:00

TcpServer 类的设计与实现:构建高性能的 TCP 服务器

在现代网络编程中,构建一个高效、稳定的 TCP 服务器是许多网络应用的基础。本文将详细介绍一个基于 C++ 的 TcpServer 类的设计与实现,该类提供了创建 TCP 服务器、处理客户端连接、数据传输和接收等功能。通过这个类,我们可以更容易地理解和实现 TCP 通信的细节。

1. TcpServer 类概述

TcpServer 类是一个用于创建和管理 TCP 服务器的类。它封装了套接字创建、绑定、监听、客户端连接处理、数据发送和接收等操作,使得网络通信更加简洁和易于管理。

2. 类构造与析构
  • 构造函数 TcpServer::TcpServer(int _port)TcpServer::TcpServer(std::string _host, int _port) 初始化服务器的主机地址和端口,并创建套接字。
TcpServer::TcpServer(int _port) : TcpServer("0.0.0.0", _port) {
}

TcpServer::TcpServer(std::string _host, int _port) : host(_host), port(_port) {
    std::cout << "create tcp server start." << std::endl;
    socket_fd = socket(AF_INET, SOCK_STREAM, 0);
    if (socket_fd == -1) {
        std::cout << "socket create error!";
        return;
    }
    int ret = set_epoll_mode(socket_fd, O_NONBLOCK);
    if (ret < 0) {
        std::cout << "epoll_mode failed:" << ret << std::endl;
        close(socket_fd);
        return;
    }
    server_addr.sin_family = AF_INET;
    inet_pton(AF_INET, host.c_str(), &server_addr.sin_addr);
    server_addr.sin_port = htons(port);
    int opt = 1;
    setsockopt(socket_fd, SOL_SOCKET, SO_REUSEADDR, &opt, sizeof(opt));
    isCreate = true;
    std::cout << "create tcp server ok." << std::endl;
}
  • 析构函数 TcpServer::~TcpServer() 虚析构函数,确保派生类的析构函数被正确调用。
TcpServer::~TcpServer() {
}
3. 服务器启动与停止
  • 启动服务器 TcpServer::Start() 绑定套接字到指定端口,并开始监听。
void TcpServer::Start() {
    if (running || !isCreate) {
        std::cout << "TcpServer start failed!" << "running=" << running << ", port=" << port << std::endl;
        return;
    }
    auto ret = bind(socket_fd, (struct sockaddr *)&server_addr, sizeof(server_addr));
    if (ret == -1) {
        std::cout << "bind faild:" << ret << std::endl;
        close(socket_fd);
        return;
    }
    ret = listen(socket_fd, SOMAXCONN);
    if (ret == -1) {
        std::cout << "Listen failed:" << ret << std::endl;
        close(socket_fd);
        return;
    }
    std::cout << "server open: " << host << ":" << port << std::endl;
    epoll_start();
}
  • 停止服务器 TcpServer::Stop() 关闭服务器并释放资源。
void TcpServer::Stop() {
    this->Close();
}
4. 资源管理
  • 关闭连接 TcpServer::Close() 关闭套接字和 epoll 文件描述符,释放资源。
void TcpServer::Close() {
    isCreate = false;
    running = false;
    socket_event.data.fd = socket_fd;
    int ret = epoll_ctl(epoll_fd, EPOLL_CTL_DEL, socket_fd, &socket_event);
    ret = close(socket_fd);
    std::cout << "socket_fd已关闭:" << ret << std::endl;
    for (TcpClient *client : clients)
        client->Close();
    ret = close(epoll_fd);
    std::cout << "TcpServer epoll_fd已关闭:" << ret << std::endl;
}
5. 非阻塞模式设置
  • 设置非阻塞模式 TcpServer::set_epoll_mode() 设置套接字为非阻塞模式。
int TcpServer::set_epoll_mode(int sock_fd, int mode) {
    int flags = fcntl(sock_fd, F_GETFL, 0);
    if (flags == -1) {
        std::cout << "epoll_mode failed:" << sock_fd << std::endl;
        return -1;
    }
    return fcntl(sock_fd, F_SETFL, flags | mode);
}
6. 客户端连接处理
  • 客户端接受线程 TcpServer::client_accept_thread() 处理客户端连接和数据事件。
void TcpServer::client_accept_thread() {
    struct epoll_event client_events[1024];
    while (running) {
        int ret = epoll_wait(epoll_fd, client_events, 1024, -1);
        if (ret < 0) {
            if (errno == EAGAIN || errno == EWOULDBLOCK)
                continue;
            else {
                std::cerr << "epoll_wait failed: " << ret << " : " << errno << " : " << strerror(errno) << std::endl;
                break;
            }
        }
        for (int n = 0; n < ret; ++n) {
            if (client_events[n].data.fd == socket_fd) {
                client_connect();
            } else {
                struct epoll_event client_event = client_events[n];
                auto client = std::find_if(clients.begin(), clients.end(), [&client_event](const TcpClient *_client)
                                           { return (_client->client_fd == client_event.data.fd); });
                if (client == clients.end())
                    continue;
                int ret = (*client)->data_receive(*client);
                if (ret == 0) {
                    clients.erase(client);
                    delete *client;
                }
            }
        }
    }
    std::cout << "服务已关闭,不再提供任何服务!" << std::endl;
    isDispose = true;
}
  • 客户端连接 TcpServer::client_connect() 接受客户端连接并添加到 epoll 监控。
void TcpServer::client_connect() {
    struct sockaddr_in client_addr;
    socklen_t addr_len = sizeof(client_addr);
    int client_fd = accept(socket_fd, (struct sockaddr *)&client_addr, &addr_len);
    std::cout << "accept:" << client_fd << " : " << strerror(errno) << std::endl;
    if (client_fd <= 0)
        return;

    TcpClient *client = new TcpClient;
    client->running = true;
    client->connected = true;
    client->client_fd = client_fd;
    client->local_addr = client_addr;
    int ret = set_epoll_mode(client_fd, O_NONBLOCK);
    if (ret == -1) {
        std::cout << "服务器接受客户端-set_epoll_mode failed:" << strerror(errno) << std::endl;
        client->Close();
        return;
    }
    client->add_epoll_event(client_fd, epoll_fd, EPOLLIN | EPOLLET);

    client->DataReceived = [this](void *sender, DataReceiveEventArgs *e)
    {
        DataReceived.Invoke(this, e);
    };

    client->recv_data = new char[client->recv_data_length];

    clients.push_back(client);
    std::cout << "新的客户端已接入:" << inet_ntoa(client_addr.sin_addr) << ":" << htons(client_addr.sin_port) << std::endl;
}
7. epoll 事件处理
  • 启动 epoll TcpServer::epoll_start() 创建 epoll 实例并添加监听套接字。
void TcpServer::epoll_start() {
    epoll_fd = epoll_create1(0);
    if (epoll_fd == -1) {
        std::cout << "poll_create1 failed:" << epoll_fd << std::endl;
        close(socket_fd);
        return;
    }

    socket_event.events = EPOLLIN;
    socket_event.data.fd = socket_fd;
    int ret = epoll_ctl(epoll_fd, EPOLL_CTL_ADD, socket_fd, &socket_event);
    if (ret == -1) {
        std::cout << "epoll_ctl failed:" << ret << std::endl;
        close(epoll_fd);
        return;
    }

    running = true;
    std::thread th = std::thread(&TcpServer::client_accept_thread, this);
    th.detach();
}

完整的代码:
TcpServer.h 头文件

#pragma once

#include <iostream>
#include <vector>
#include <list>
#include <string>
#include <sys/socket.h>
#include <netinet/in.h>
#include <sys/epoll.h>
#include <sys/fcntl.h>
#include <arpa/inet.h>
#include <algorithm>
#include <unistd.h>
#include <thread>
#include "DataReceiveEventArgs.h"
#include "TcpClient.h"

class TcpServer
{
public:
    TcpServer(int _port);
    TcpServer(std::string _host, int _port);
    ~TcpServer();

public:
    EventHandler<DataReceiveEventArgs> DataReceived;

public:
    void Start();
    void Stop();
    void Close();
    bool IsDispose();

private:
    void epoll_start();                        // epoll初始化(创建epoll)
    int set_epoll_mode(int sock_fd, int mode); // epoll模式--创建socket时,为非阻塞模式
    void client_accept_thread();
    void client_connect();

    sockaddr_in get_remote_addr(int sock);

private:
    int socket_fd;
    int epoll_fd;
    std::string host = "0.0.0.0";
    int port = 0;
    bool running = false;
    bool isCreate = false;
    int send_buff_size = 1024 * 1024;
    int recv_buff_size = 1024 * 1024;
    struct sockaddr_in server_addr;
    std::list<TcpClient *> clients;
    // 将监听套接字添加到 epoll 中,监控 EPOLLIN (表示有数据可读)事件
    struct epoll_event socket_event;
    bool isDispose = false;
};

TcpServer.cpp

#include "TcpServer.h"

TcpServer::TcpServer(int _port) : TcpServer("0.0.0.0", _port)
{
}

TcpServer::TcpServer(std::string _host, int _port) : host(_host), port(_port)
{
    std::cout << "create tcp server start." << std::endl;
    // AF_INET 表示使用 IPv4 协议
    // SOCK_STREAM 表示套接字的类型,表示 面向连接的流式套接字
    socket_fd = socket(AF_INET, SOCK_STREAM, 0);
    // std::cout << "create tcp server socket_fd:" << socket_fd << std::endl;
    if (socket_fd == -1)
    {
        std::cout << "socket ceate error!";
        return;
    }
    // 文件描述符为非阻塞模式
    int ret = set_epoll_mode(socket_fd, O_NONBLOCK);
    if (ret < 0)
    {
        std::cout << "epoll_mode failed:" << ret << std::endl;
        close(socket_fd);
        return;
    }

    server_addr.sin_family = AF_INET;

    // ip字符串转int
    inet_pton(AF_INET, host.c_str(), &server_addr.sin_addr);
    //server_addr.sin_addr.s_addr = ntohl(server_addr.sin_addr.s_addr);
    server_addr.sin_port = htons(port);

    // 以下设置表示当调用close关闭客户端时,立即释放端口,不等待
    // 在TCP服务端,客户端调用 close(client_fd) 关闭连接后,如果你尝试重新连接时出现端口没有完全释放的情况,通常是由于 TCP 连接的 TIME_WAIT 状态没有及时清理。这是 TCP 协议的正常行为。
    // 在TCP连接关闭后,端口会进入 TIME_WAIT 状态。这个状态的目的是确保最后的数据包能够正确到达。如果新的连接尝试在该端口上进行,而该端口仍然处于 TIME_WAIT 状态,就会出现端口被占用的情况。
    // TIME_WAIT 状态通常会持续一段时间(默认是4分钟,即240秒),这可以通过操作系统的内核参数来修改。

    // struct linger linger_opt;
    // linger_opt.l_onoff = 1;  // 启用
    // linger_opt.l_linger = 1; // 立即关闭
    // // 此方法亲测无效
    // // setsockopt(socket_fd, SOL_SOCKET, SO_LINGER, &linger_opt, sizeof(linger_opt));

    int opt = 1;
    setsockopt(socket_fd, SOL_SOCKET, SO_REUSEADDR, &opt, sizeof(opt));
    isCreate = true;
    std::cout << "create tcp server ok." << std::endl;
}

TcpServer::~TcpServer()
{
}

void TcpServer::Start()
{
    if (running || !isCreate)
    {
        std::cout << "TcpServer start failed!" << "running=" << running << ", port=" << port << std::endl;
        // close(socket_fd);
        return;
    }
    // 绑定套接字到指定端口
    auto ret = bind(socket_fd, (struct sockaddr *)&server_addr, sizeof(server_addr));
    if (ret == -1)
    {
        std::cout << "bind faild:" << ret << std::endl;
        close(socket_fd);
        return;
    }

    // 开始监听
    ret = listen(socket_fd, SOMAXCONN);
    if (ret == -1)
    {
        std::cout << "Listen failed:" << ret << std::endl;
        close(socket_fd);
        return;
    }
    std::cout << "server open: " << host << ":" << port << std::endl;

    epoll_start();
}

void TcpServer::Stop()
{
    this->Close();
}

void TcpServer::Close()
{
    isCreate = false;
    running = false;

    // 1. 删除socket_fd的epoll事件
    socket_event.data.fd = socket_fd;
    int ret = epoll_ctl(epoll_fd, EPOLL_CTL_DEL, socket_fd, &socket_event);

    // 2. 关闭监听套接字
    ret = close(socket_fd);
    std::cout << "socket_fd已关闭:" << ret << std::endl;

    // 3. 从 epoll 中移除所有连接的客户端文件描述符
    for (TcpClient *client : clients)
        client->Close();
    // 4. 关闭 epoll 文件描述符
    ret = close(epoll_fd);
    std::cout << "TcpServer epoll_fd已关闭:" << ret << std::endl;
}

bool TcpServer::IsDispose()
{
    return isDispose;
}

int TcpServer::set_epoll_mode(int sock_fd, int mode)
{
    /*
    O_NONBLOCK(非阻塞模式):如果设置了这个标志,表示该套接字(或文件)是非阻塞的,执行读写操作时不会阻塞调用进程或线程。
    套接字在没有数据可读或可写时不会让程序等待,而是立即返回。
    O_RDWR、O_WRONLY、O_RDONLY(访问模式):表示套接字的打开方式。
    O_APPEND(追加模式):指示文件或套接字在写操作时会追加数据。
    */
    int flags = fcntl(sock_fd, F_GETFL, 0); // 获取当前套接字的文件状态标志
    if (flags == -1)
    {
        std::cout << "epoll_mode failed:" << sock_fd << std::endl;
        return -1;
    }
    // 设置套接字为非阻塞模式
    return fcntl(sock_fd, F_SETFL, flags | mode);
}

void TcpServer::client_accept_thread()
{
    struct epoll_event client_events[1024];
    while (running)
    {
        // 阻塞等待事件
        int ret = epoll_wait(epoll_fd, client_events, 1024, -1);
        if (ret < 0)
        {
            if (errno == EAGAIN || errno == EWOULDBLOCK)
                continue;
            else
            {
                std::cerr << "epoll_wait failed: " << ret << " : " << errno << " : " << strerror(errno) << std::endl;
                break;
            }
        }
        // 处理返回的事件
        for (int n = 0; n < ret; ++n)
        {
            if (client_events[n].data.fd == socket_fd)
            {
                // 如果是监听套接字的事件,有新的客户端连接
                client_connect();
            }
            else
            {
                struct epoll_event client_event = client_events[n];
                auto client = std::find_if(clients.begin(), clients.end(), [&client_event](const TcpClient *_client)
                                           { return (_client->client_fd == client_event.data.fd); });
                if (client == clients.end())
                    continue;
                // 客户端有数据
                int ret = (*client)->data_receive(*client);
                if (ret == 0)
                {
                    clients.erase(client);
                    delete *client;
                }
            }
        }
    }
    std::cout << "服务已关闭,不再提供任何服务!" << std::endl;
    isDispose = true;
}

void TcpServer::client_connect()
{
    struct sockaddr_in client_addr;
    socklen_t addr_len = sizeof(client_addr);
    int client_fd = accept(socket_fd, (struct sockaddr *)&client_addr, &addr_len); // 接受连接
    std::cout << "accept:" << client_fd << " : " << strerror(errno) << std::endl;
    if (client_fd <= 0)
        return;

    TcpClient *client = new TcpClient;
    client->running = true;
    client->connected = true;
    client->client_fd = client_fd;
    client->local_addr = client_addr;
    // 设置客户端套接字为非阻塞模式
    int ret = set_epoll_mode(client_fd, O_NONBLOCK);
    if (ret == -1)
    {
        std::cout << "服务器接受客户端-set_epoll_mode failed:" << strerror(errno) << std::endl;
        client->Close();
        return;
    }
    // client.SetSendBuffSize(send_buff_size);
    // client.SetRecvBuffSize(recv_buff_size);
    //  将新客户端套接字添加到 epoll 中,监听可读事件

    // client->create_epoll();
    client->add_epoll_event(client_fd, epoll_fd, EPOLLIN | EPOLLET);

    client->DataReceived = [this](void *sender, DataReceiveEventArgs *e)
    {
        DataReceived.Invoke(this, e);
    };

    // client->start_receive();
    client->recv_data = new char[client->recv_data_length];

    clients.push_back(client);
    std::cout << "新的客户端已接入:" << inet_ntoa(client_addr.sin_addr) << ":" << htons(client_addr.sin_port) << std::endl;
}

void TcpServer::epoll_start()
{
    // 创建epoll
    epoll_fd = epoll_create1(0);
    if (epoll_fd == -1)
    {
        std::cout << "poll_create1 failed:" << epoll_fd << std::endl;
        close(socket_fd);
        return;
    }

    // 监听可读事件
    socket_event.events = EPOLLIN;
    // 将监听套接字的文件描述符传给 epoll
    socket_event.data.fd = socket_fd;
    // 将监听套接字添加到 epoll 中,监控 EPOLLIN 事件(表示有数据可读)
    int ret = epoll_ctl(epoll_fd, EPOLL_CTL_ADD, socket_fd, &socket_event);
    if (ret == -1)
    {
        std::cout << "epoll_ctl failed:" << ret << std::endl;
        close(epoll_fd);
        return;
    }

    running = true;
    std::thread th = std::thread(&TcpServer::client_accept_thread, this);
    th.detach();
}

8. 总结

本文详细介绍了 TcpServer 类的设计与实现,包括构造与析构、服务器启动与停止、资源管理、非阻塞模式设置、客户端连接处理以及 epoll 事件处理。通过这个类,我们可以更容易地理解和实现 TCP 通信的细节。这个类提供了一个简洁的接口来管理 TCP 服务器,使得网络编程更加高效和易于维护。

TcpServer 类的设计注重模块化和可扩展性,允许开发者根据具体需求进行定制和扩展。通过使用 epoll 事件模型,TcpServer 类能够支持高并发的客户端连接,适用于需要处理大量并发连接的网络应用。此外,类中的非阻塞模式设置和资源管理机制确保了服务器的稳定性和高效性。

总的来说,TcpServer 类为构建高性能的 TCP 服务器提供了一个强大的基础。通过这个类,开发者可以快速构建和部署 TCP 服务器,满足各种网络应用的需求。

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

AEO海关认证的注意事项

AEO海关认证的注意事项繁多且至关重要&#xff0c;企业需细致准备&#xff0c;确保万无一失。 首先&#xff0c;企业需深入研读相关政策文件&#xff0c;如《中华人民共和国海关注册登记和备案企业信用管理办法》及《海关高级认证企业标准》&#xff0c;以政策为指引&#xff0…

MySQL如何只取根据某列连续重复行的第一条记录

前言 MySQL如何只取根据某列连续重复行的第一条记录&#xff0c;条件&#xff1a;某列、连续、验重 建表准备 DROP TABLE IF EXISTS test; CREATE TABLE test (id bigint NOT NULL,time datetime NULL DEFAULT NULL,price int NULL DEFAULT NULL,PRIMARY KEY (id) USING BT…

c++编译过程初识

编译过程 预处理&#xff1a;主要是执行一些预处理指令&#xff0c;主要是#开头的代码&#xff0c;如#include 的头文件、#define 定义的宏常量、#ifdef #ifndef #endif等条件编译的代码&#xff0c;具体包括查找头文件、进行宏替换、根据条件编译等操作。 g -E example.cpp -…

JS中的闭包和上下文

变量提升 和 函数提升 这里要提到一个提升的概念&#xff0c;即在JS中&#xff0c;在解析代码之前还有一个预处理的过程&#xff0c;这个过程中会把部分变量和函数声明提前到代码的最顶部&#xff0c; 会在其他所有代码之前执行。虽然当我们按照规范&#xff08;严格模式或者T…

GitLab 将停止为中国区用户提供服务,60天迁移期如何应对? | LeetTalk Daily

“LeetTalk Daily”&#xff0c;每日科技前沿&#xff0c;由LeetTools AI精心筛选&#xff0c;为您带来最新鲜、最具洞察力的科技新闻。 GitLab作为一个广受欢迎的开源代码托管平台&#xff0c;近期宣布将停止服务中国大陆、澳门和香港地区的用户提供服务。根据官方通知&#x…

【2024年最新】BilibiliB站视频动态评论爬虫

废话不多说&#xff0c;直接先放git仓库&#xff1a;GitHub - linyuye/Bilibili_crawler: bilibili爬虫&#xff0c;基于selenium获取oid与cookie&#xff0c;request获取api内容 〇&#xff1a;概念简述 oid&#xff1a;视频/动态的uuid&#xff0c;b站对于发布内容的通用唯…

汽车IVI中控开发入门及进阶(46):FFmpeg

概述: FFmpeg 是领先的多媒体框架,能够解码、编码、 转码、复用、解复用、流、过滤和播放 几乎所有人类和机器创建的东西。它支持最模糊的古老格式,直到最前沿。无论它们是由某个标准委员会、社区还是公司设计的。它还具有高度的可移植性:FFmpeg 在各种构建环境、机器架构…

计算属性 简写和 完整写法

计算属性渲染不加上括号 methods方法和computed属性区别&#xff1a; computed只计算一次&#xff0c;然后缓存&#xff0c;后续直接拿出来使用&#xff0c;而methods每次使用每次计算&#xff0c;不会缓存 计算属性完整写法&#xff1a; 既获取又设置 slice 截取 成绩案例 …

WebRTC Simulcast 大小流介绍与优化实践

Simulcast 是 WebRTC 中的一种标准化技术 &#xff0c;简称大小流。通过 Simulcast&#xff0c;客户端可以同时发送同一视频的多个版本。每个版本都以不同的分辨率和帧率独立编码&#xff0c;带宽较多的拉流端可以接收较高质量的视频流&#xff0c;带宽有限的拉流端则可以接收较…

kong网关使用pre-function插件,改写接口的返回数据

一、背景 kong作为api网关&#xff0c;除了反向代理后端服务外&#xff0c;还可对接口进行预处理。 比如本文提及的一个小功能&#xff0c;根据http header某个字段的值&#xff0c;等于多少的时候&#xff0c;返回一个固定的报文。 使用到的kong插件是pre-function。 除了上…

轮播图带详情插件、uniApp插件

超级好用的轮播图 介绍访问地址参数介绍使用方法&#xff08;简单使用&#xff0c;参数结构点击链接查看详情&#xff09;图片展示 介绍 带有底部物品介绍以及价格的轮播图组件&#xff0c;持续维护&#xff0c;uniApp插件&#xff0c;直接下载填充数据就可以在项目里面使用 …

Vite内网ip访问,两种配置方式和修改端口号教程

目录 问题 两种解决方式 结果 总结 preview.host preview.port 问题 使用vite运行项目的时候&#xff0c;控制台会只出现127.0.0.1&#xff08;localhost&#xff09;本地地址访问项目。不可以通过公司内网ip访问&#xff0c;其他团队成员无法访问&#xff0c;这是因为没…

老旧小区用电安全保护装置#限流式防火保护器参数介绍#

摘要 随着居民住宅区用电负荷的增加&#xff0c;用电安全问题日益突出&#xff0c;火灾隐患频繁发生。防火限流式保护器作为一种新型电气安全设备&#xff0c;能够有效预防因电气故障引发的火灾事故。本文介绍了防火限流式保护器的工作原理、技术特点及其在居民住宅区用电系统…

Ftrans数据摆渡系统 搭建安全便捷跨网文件传输通道

一、专业数据摆渡系统对企业的意义 专业的数据摆渡系统对企业具有重要意义&#xff0c;主要体现在以下几个方面‌&#xff1a; 1、‌数据安全性‌&#xff1a;数据摆渡系统通过加密传输、访问控制和审计日志等功能&#xff0c;确保数据在传输和存储过程中的安全性。 2、‌高…

LabVIEW生物医学信号虚拟实验平台

介绍了一款基于LabVIEW的多功能生物医学信号处理实验平台的设计和实现。平台通过实践活动加强学生对理论的理解和应用能力&#xff0c;特别是在心电图(ECG)和脑电图(EEG)的信号处理方面。实验平台包括信号的滤波、特征提取和频谱分析等功能&#xff0c;能直观体验和掌握生物医学…

大数据实验三

Python and anaconda 实验三数据预处理和轨迹聚类参考地址&#xff1a; https://www.hifleet.com/wp/communities/data/hangyundashujujishukechengshiyanzhinanshujuyuchulijiguijijuleichixugengxinzhong#post-2212https://www.hifleet.com/wp/communities/data/hangyundas…

【python因果库实战14】因果生存分析3

标准化生存分析 参见《因果推断》一书第17.5节&#xff08;“参数化的g公式”&#xff09;。 在参数化标准化中&#xff0c;也称为“参数化g公式”&#xff0c;时间步k处的生存率是对协变量X水平和处理分配a条件下的条件生存率的加权平均&#xff0c;权重为每个分层中个体的比…

云边端一体化架构

云边端一体化架构是一种将云计算、边缘计算和终端设备相结合的分布式计算模型。该架构旨在通过优化资源分配和数据处理流程&#xff0c;提供更高效、更低延迟的服务体验。 下面是对这个架构的简要说明&#xff1a; 01云计算&#xff08;Cloud Computing&#xff09; — 作为中心…

C/C++ 数据结构与算法【哈夫曼树】 哈夫曼树详细解析【日常学习,考研必备】带图+详细代码

哈夫曼树&#xff08;最优二叉树&#xff09; 1&#xff09;基础概念 **路径&#xff1a;**从树中一个结点到另一个结点之间的分支构成这两个结点间的路径。 **结点的路径长度&#xff1a;**两结点间路径上的分支数。 **树的路径长度&#xff1a;**从树根到每一个结点的路径…

2、C#基于.net framework的应用开发实战编程 - 设计(二、三) - 编程手把手系列文章...

二、设计&#xff1b; 二&#xff0e;三、构建数据库&#xff1b; 此例子使用的是SQLite数据库&#xff0c;所以数据库工具用的SQLiteStudio x64&#xff0c;这个是SQLite专用的数据库设计管理工具&#xff0c;其它的数据库管理工具比如DBeaver的使用请见实战工具系列文章。 1、…