C语言终篇--基于epoll ET模式 的 非阻塞IO服务器模型

使用技术:

1 epoll事件驱动机制:使用epoll作为IO多路复用的技术,以高效地管理多个socket上的事件。

2 边缘触发(Edge Triggered, ET)模式:epoll事件以边缘触发模式运行,这要求代码必须负责消费所有可用的数据,直到收到EAGAIN或EWOULDBLOCK错误。

3 非阻塞IO:通过设置socket为非阻塞模式,确保IO操作不会导致服务器线程挂起等待。

4 信号处理:捕捉SIGINT信号,以允许服务器通过中断信号优雅地关闭和清理资源。

5 地址重用:设置SO_REUSEADDR套接字选项,允许服务器快速重启而不必等待TIME_WAIT状态的连接自然超时。

6 调整接收缓冲区为1字节,将接收的数据存在内存的容器中,用于模拟大数据来袭,无法全部接收的情景

7 可动态扩容的兴趣事件列表

8 使用whlie()循环,分别包裹ET模式下的accpet(),recv(),send(),确保读尽,写尽,及合适的时机退出

注意事项:

运行环境:

unix-like gun_c 因为涉及到信号需独立终端 不能再IDE环境下运行

地址端口 按需调整 

赠送:

epoll LT模式 服务端 对比用双循环阻塞服务端 测试用客户端

运行效果:

epoll ET模式 服务端:

#define _GNU_SOURCE
#include <stdio.h>
#include <sys/socket.h>
#include <sys/un.h>
#include <fcntl.h>
#include <arpa/inet.h>
#include <netinet/in.h>
#include <sys/epoll.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <signal.h>
#include <errno.h>

#define SERVER_IP "192.168.142.132"
#define SERVER_PORT 50001
// 单线程访问count,count表示感兴趣的fd的总数
int count = 0;
// 用于清理函数
void *global_r_events;
char *global_buf_p;
int global_server_sockfd;
// 注册清理函数
void clean_up()
{
    free(global_r_events);
    free(global_buf_p);
    close(global_server_sockfd);
    printf("clean_up\n");
}
// 如果按了CTRL+C则退出并执行清理函数
void sig_handler(int sig)
{
    exit(EXIT_SUCCESS);
}
// 此函数功能将socket设为非阻塞
void set_non_blocking(int fd)
{
    int flags = fcntl(fd, F_GETFL);
    if (flags == -1)
    {
        perror("fcntl F_GETFL");
    }
    flags |= O_NONBLOCK;
    if (fcntl(fd, F_SETFL, flags) == -1)
    {
        perror("fcntl F_SETFL");
    }
}
// 此函数功能 调用设置非阻塞函数+将sockfd加入兴趣列表+将sockfd设为ET
void add_ins_events(int epoll_fd, int fd)
{
    // 设成非阻塞
    set_non_blocking(fd);
    struct epoll_event ins_event = {0};
    // 关注读+边缘触发
    ins_event.events = EPOLLIN | EPOLLET;
    ins_event.data.fd = fd;
    if (epoll_ctl(epoll_fd, EPOLL_CTL_ADD, fd, &ins_event) == -1)
    {
        perror("epoll_ctl");
    }
    // count表示感兴趣的fd的总数
    count++;
}

int main()
{
    int server_sockfd, client_sockfd;
    struct sockaddr_in server_sockaddr, client_sockaddr;
    memset(&server_sockaddr, 0, sizeof(server_sockaddr));
    memset(&client_sockaddr, 0, sizeof(client_sockaddr));
    socklen_t client_sockaddr_len = sizeof(client_sockaddr);
    socklen_t server_sockaddr_len = sizeof(server_sockaddr);
    ssize_t send_bytes, recv_bytes;
    // 预留
    char send_buf[1024] = {0};
    // 调整接收缓冲区为1字节,用于模拟大数据来袭,无法全部接收的情景
    char recv_buf[1] = {0};
    // 一个用于添加兴趣事件的结构体
    struct epoll_event ins_event = {0};

    // i用于循环,optval套接字属性用,size是用户空间就绪事件列表的长度
    // 用户空间兴趣事件的计数是count,遍历的上限也是count,就绪事件的个数一定小于count,size只作为容器大小存在
    int i = 0, optval = 1, size = 1024;
    // 需要发送的总字节数,已发送字节数,已接收字节总数
    int send_total, sent, recv_total;
    // 注册信号
    if (signal(SIGINT, sig_handler) == SIG_ERR)
    {
        perror("signal");
    }

    // 注册退出清理函数
    atexit(clean_up);
    // 创建socket ipv4 tcp
    server_sockfd = socket(AF_INET, SOCK_STREAM, 0);
    if (server_sockfd == -1)
    {
        perror("socket");
    }
    // 向全局变量传递值,便于关闭
    global_server_sockfd = server_sockfd;
    // 地址端口复用
    if (setsockopt(server_sockfd, SOL_SOCKET, SO_REUSEADDR, &optval, sizeof(optval)) == -1)
    {
        perror("setsockopt");
    }
    // 绑定
    server_sockaddr.sin_family = AF_INET;
    server_sockaddr.sin_port = htons(SERVER_PORT);
    if (inet_pton(AF_INET, SERVER_IP, &server_sockaddr.sin_addr.s_addr) == -1)
    {
        perror("inet_pton");
    }
    if (bind(server_sockfd, (struct sockaddr *)&server_sockaddr, server_sockaddr_len) == -1)
    {
        perror("bind");
    }
    // 监听
    if (listen(server_sockfd, 1024) == -1)
    {
        perror("listen");
    }
    // 创建epoll实例
    int epoll_fd = epoll_create1(0);
    if (epoll_fd == -1)
    {
        perror("epoll_create1");
    }
    // 将server_sockfd设成非阻塞 对读感兴趣 ET模式 count++
    add_ins_events(epoll_fd, server_sockfd);
    // 这是一个在堆上分配的就绪事件容器,epoll_wait将填充这个容器的一部分
    struct epoll_event *r_events = (struct epoll_event *)calloc(size, sizeof(struct epoll_event));
    if (r_events == NULL)
    {
        perror("calloc");
    }
    else
    {
        // 向全局变量传递值,便于释放
        global_r_events = r_events;
    }
    // 这是一个足够大的内存缓冲区 用于储存接收的数据
    // 因为每次接收被限流为1字节 模拟大数据到来时缓冲区的渺小
    // 每次接收的数据会用指针算数,拼在已接收数据的后面
    char *buf_p = (char *)calloc(1024, sizeof(char));
    if (buf_p == NULL)
    {
        perror("calloc");
    }
    else
    {
        // 向全局变量传递值,便于释放
        global_buf_p = buf_p;
    }

    printf("server start ...\n");
    // 服务器主循环 里面有3个 whlie(1)循环,分别用于包裹ET模式下的accpet,recv,send
    while (1)
    {
        // 清零就绪事件的容器
        memset(r_events, 0, size * sizeof(struct epoll_event));
        // count表示已添加兴趣列表事件的最大值,size是容器大小,因为就绪事件的大小永远小于等于兴趣事件,所以用count
        // -1表示没有就绪事件则一直阻塞,为除了算力问题外,整个架构设计的唯一IO阻塞点
        int r = epoll_wait(epoll_fd, r_events, count, -1);
        if (r > 0)
        {
            // r>0,r为就绪事件数量,所以遍历r
            for (i = 0; i < r; i++)
            {
                // 如果读就绪且fd是server_sockfd
                if ((r_events[i].events & EPOLLIN) && (r_events[i].data.fd == server_sockfd))
                {
                    // ET模式下包裹accept的循环
                    while (1)
                    {
                        // 接收连接
                        client_sockfd = accept(server_sockfd, (struct sockaddr *)&client_sockaddr, &client_sockaddr_len);
                        if (client_sockfd < 0)
                        {
                            if (errno == EAGAIN || errno == EWOULDBLOCK)
                            {
                                // 读尽,跳出循环
                                break;
                            }
                            else
                            {
                                // 这就是出错了
                                perror("accept");
                            }
                        }
                        else
                        {
                            // 打印日志
                            printf("[+] %u: connected\n", ntohs(client_sockaddr.sin_port));
                            // 将client_sockfd设成非阻塞 对读感兴趣 ET模式 count++
                            add_ins_events(epoll_fd, client_sockfd);
                            // 如果count的值将要接近容器容量,则容器容量翻倍
                            if ((size - count) < 100)
                            {
                                size += size;
                                r_events = (struct epoll_event *)realloc(r_events, size * sizeof(struct epoll_event));
                                if (r_events == NULL)
                                {
                                    perror("realloc");
                                }
                                // 赋值全局变量,方便释放
                                global_r_events = r_events;
                            }
                        }
                    }
                }
                // 如果fd读就绪且不是server_sockfd,此时处理读写
                if ((r_events[i].events & EPOLLIN) && (r_events[i].data.fd != server_sockfd))
                {
                    // 获取peer端口号,打印日志用
                    getpeername(r_events[i].data.fd, (struct sockaddr *)&client_sockaddr, &client_sockaddr_len);
                    // 打印日志
                    printf("[+] %u: recv ready\n", ntohs(client_sockaddr.sin_port));
                    // 已接收的字节总数,用于在buf_p(一个足够大的内存缓冲区)中定位
                    recv_total = 0;
                    // ET模式下 读尽策略 包裹recv 的循环
                    while (1)
                    {
                        // 读缓冲区被限流到1字节
                        recv_bytes = recv(r_events[i].data.fd, recv_buf, sizeof(recv_buf), 0);
                        // 如果读到了数据
                        if (recv_bytes > 0)
                        {
                            // 将数据追加在buf_p的最后
                            memcpy(buf_p + recv_total, recv_buf, recv_bytes);
                            // 计数
                            recv_total += recv_bytes;
                        }
                        if (recv_bytes < 0)
                        {
                            // 读尽
                            if (errno == EAGAIN || errno == EWOULDBLOCK)
                            {
                                // 打印buf_p(一个足够大的内存缓冲区)
                                printf("[+] %u: %s\n", ntohs(client_sockaddr.sin_port), buf_p);
                                /* send */
                                // 每次发送的字节
                                send_bytes = 0;
                                // 需要发送的总字节
                                send_total = strlen(buf_p);
                                // 已发送总字节
                                sent = 0;

                                // ET模式下 包裹send的循环
                                while (1)
                                {
                                    // buf_p + sent是指针运算表示发送位置,send_total - sent是剩余发送量
                                    send_bytes = send(r_events[i].data.fd, buf_p + sent, send_total - sent, 0);
                                    if (send_bytes == -1)
                                    {
                                        // 表示发送缓冲区已满
                                        if (errno == EAGAIN || errno == EWOULDBLOCK)
                                        {
                                            // 添加对写感兴趣
                                            ins_event.events = EPOLLET | EPOLLIN | EPOLLOUT;
                                            ins_event.data.fd = r_events[i].data.fd;
                                            epoll_ctl(epoll_fd, EPOLL_CTL_MOD, r_events[i].data.fd, &ins_event);
                                            // 直接跳出循环,如果写就绪epoll_wait会通知
                                            break;
                                        }
                                        // 对方异常关闭
                                        if (errno == ECONNRESET)
                                        {
                                            // 将fd移除兴趣事件列表
                                            epoll_ctl(epoll_fd, EPOLL_CTL_DEL, r_events[i].data.fd, NULL);
                                            // 关闭fd
                                            close(r_events[i].data.fd);
                                            // 兴趣列表最大值-1
                                            count--;
                                            // 跳出send循环
                                            break;
                                        }
                                    }
                                    // 如果发送了一些数据
                                    if (send_bytes > 0)
                                    {
                                        // 累加到已发送字节
                                        sent += send_bytes;
                                        // 如果 已发送字节等于发送总字节 则取消对于写的兴趣
                                        // 如果 之前fd没有关注写 则不变
                                        if (sent == send_total)
                                        {
                                            ins_event.events = EPOLLET | EPOLLIN;
                                            ins_event.data.fd = r_events[i].data.fd;
                                            epoll_ctl(epoll_fd, EPOLL_CTL_MOD, r_events[i].data.fd, &ins_event);
                                            break;
                                        }
                                    }
                                }
                                /* send end */
                                // 如果 不发送的话打印完接收数据就可以直接 跳出循环
                                break;
                            }
                            // recv 对方异常终止
                            else if (errno == ECONNRESET)
                            {
                                // 打印日志
                                char a[32] = {0};
                                snprintf(a, sizeof(a), "[-] %u", ntohs(client_sockaddr.sin_port));
                                perror(a);
                                // 从感兴趣列表删除
                                epoll_ctl(epoll_fd, EPOLL_CTL_DEL, r_events[i].data.fd, NULL);
                                // 关闭fd
                                close(r_events[i].data.fd);
                                // 兴趣列表最大值-1
                                count--;
                                break;
                            }
                            else
                            {
                                perror("recv");
                            }
                        }
                        // 如果读到0,表示peer正常关闭了连接
                        if (recv_bytes == 0)
                        {
                            getpeername(r_events[i].data.fd, (struct sockaddr *)&client_sockaddr, &client_sockaddr_len);
                            printf("[-] %u: closed\n", ntohs(client_sockaddr.sin_port));
                            // 从兴趣列表删除这个fd
                            epoll_ctl(epoll_fd, EPOLL_CTL_DEL, r_events[i].data.fd, NULL);
                            // 关闭fd
                            close(r_events[i].data.fd);
                            // 兴趣列表最大值-1
                            count--;
                            // 跳出读循环
                            break;
                        }
                    }
                }
            }
        }
        if (r == -1)
        {
            perror("epoll_wait");
        }
    }
    return 0;
}

epoll LT模式 服务端 :

#define _GNU_SOURCE
#include <stdio.h>
#include <sys/socket.h>
#include <sys/un.h>
#include <sys/ioctl.h>
#include <fcntl.h>
#include <netdb.h>
#include <arpa/inet.h>
#include <netinet/in.h>
#include <sys/select.h>
#include <sys/poll.h>
#include <sys/epoll.h>
#include <sys/types.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <aio.h>
#include <signal.h>

#define SERVER_IP "192.168.142.132"
#define SERVER_PORT 50001

int count = 0;
void *global_r_events;
int global_server_sockfd;
void clean_up()
{
    free(global_r_events);
    close(global_server_sockfd);
    printf("clean_up\n");
}
void sig_handler(int sig)
{
    exit(EXIT_SUCCESS);
}

void set_non_blocking(int fd)
{
    int flags = fcntl(fd, F_GETFL);
    if (flags == -1)
    {
        perror("fcntl F_GETFL");
    }
    flags |= O_NONBLOCK;
    if (fcntl(fd, F_SETFL, flags) == -1)
    {
        perror("fcntl F_SETFL");
    }
}

void add_ins_events(int epoll_fd, int fd)
{
    set_non_blocking(fd);
    struct epoll_event ins_event = {0};
    ins_event.events = EPOLLIN;
    ins_event.data.fd = fd;
    if (epoll_ctl(epoll_fd, EPOLL_CTL_ADD, fd, &ins_event) == -1)
    {
        perror("epoll_ctl");
    }
    count++;
}

int main()
{
    int server_sockfd, client_sockfd;
    struct sockaddr_in server_sockaddr, client_sockaddr;
    memset(&server_sockaddr, 0, sizeof(server_sockaddr));
    memset(&client_sockaddr, 0, sizeof(client_sockaddr));
    socklen_t client_sockaddr_len = sizeof(client_sockaddr);
    socklen_t server_sockaddr_len = sizeof(server_sockaddr);
    ssize_t send_bytes, recv_bytes;
    char send_buf[1024] = "How can I help you today ?";
    char recv_buf[1024] = {0};
    int i = 0, optval = 1, size = 1024;

    signal(SIGINT, sig_handler);

    atexit(clean_up);

    server_sockfd = socket(AF_INET, SOCK_STREAM, 0);
    if (server_sockfd == -1)
    {
        perror("socket");
    }
    global_server_sockfd = server_sockfd;

    if (setsockopt(server_sockfd, SOL_SOCKET, SO_REUSEADDR, &optval, sizeof(optval)) == -1)
    {
        perror("setsockopt");
    }

    server_sockaddr.sin_family = AF_INET;
    server_sockaddr.sin_port = htons(SERVER_PORT);
    if (inet_pton(AF_INET, SERVER_IP, &server_sockaddr.sin_addr.s_addr) == -1)
    {
        perror("inet_pton");
    }
    if (bind(server_sockfd, (struct sockaddr *)&server_sockaddr, server_sockaddr_len) == -1)
    {
        perror("bind");
    }

    if (listen(server_sockfd, 1024) == -1)
    {
        perror("listen");
    }

    int epoll_fd = epoll_create1(0);
    if (epoll_fd == -1)
    {
        perror("epoll_create1");
    }

    add_ins_events(epoll_fd, server_sockfd);

    struct epoll_event *r_events = (struct epoll_event *)calloc(size, sizeof(struct epoll_event));
    if (r_events == NULL)
    {
        perror("calloc");
    }
    global_r_events = r_events;

    printf("server start ...\n");

    while (1)
    {
        memset(r_events, 0, size * sizeof(struct epoll_event));
        // count用于有效遍历,而size则是数组的实际大小
        int r = epoll_wait(epoll_fd, r_events, count, -1);
        if (r > 0)
        {
            for (i = 0; i < r; i++)
            {
                if ((r_events[i].events & EPOLLIN) && (r_events[i].data.fd == server_sockfd))
                {
                    printf("accept ready\n");
                    client_sockfd = accept(server_sockfd, NULL, NULL);
                    add_ins_events(epoll_fd, client_sockfd);
                    if ((size - count) < 10)
                    {
                        size += size;
                        r_events = (struct epoll_event *)realloc(r_events, size * sizeof(struct epoll_event));
                        if (r_events == NULL)
                        {
                            perror("realloc");
                        }
                        global_r_events = r_events;
                    }
                }
                if ((r_events[i].events & EPOLLIN) && (r_events[i].data.fd != server_sockfd))
                {
                    printf("recv ready\n");
                    recv_bytes = recv(r_events[i].data.fd, recv_buf, sizeof(recv_buf), 0);
                    if (recv_bytes < 0)
                    {
                        perror("recv");
                    }
                    if (recv_bytes == 0)
                    {
                        printf("close by peer\n");
                        close(r_events[i].data.fd);
                    }
                    if (recv_bytes > 0)
                    {
                        printf("%s\n", recv_buf);
                        send_bytes = send(r_events[i].data.fd, send_buf, strlen(send_buf), 0);
                        if (send_bytes == -1)
                        {
                            perror("send");
                        }
                    }
                }
            }
        }
    }

    return 0;
}

对比用双循环阻塞服务端:

#define _GNU_SOURCE
#include <stdio.h>
#include <sys/socket.h>
#include <sys/un.h>
#include <arpa/inet.h>
#include <netinet/in.h>
#include <string.h>
#include <unistd.h>

#define SERVER_IP "192.168.142.132"
#define SERVER_PORT 50001

int main()
{
    int server_sockfd, client_sockfd;
    struct sockaddr_in server_sockaddr, client_sockaddr;
    memset(&server_sockaddr, 0, sizeof(server_sockaddr));
    memset(&client_sockaddr, 0, sizeof(client_sockaddr));
    socklen_t client_sockaddr_len = sizeof(client_sockaddr);
    ssize_t send_bytes, recv_bytes;
    char send_buf[1024] = "How can I help you today ?";
    char recv_buf[1024] = {0};

    server_sockfd = socket(AF_INET, SOCK_STREAM, 0);
    if (server_sockfd == -1)
    {
        perror("socket");
    }

    int optval = 1;
    setsockopt(server_sockfd, SOL_SOCKET, SO_REUSEADDR, &optval, sizeof(optval));

    server_sockaddr.sin_family = AF_INET;
    inet_pton(AF_INET, SERVER_IP, &server_sockaddr.sin_addr.s_addr);
    server_sockaddr.sin_port = htons(SERVER_PORT);
    if (bind(server_sockfd, (struct sockaddr *)&server_sockaddr, sizeof(server_sockaddr)) == -1)
    {
        perror("bind");
    }

    if (listen(server_sockfd, 16) == -1)
    {
        perror("listen");
    }

    printf("server start...\n");

    while (1)
    {
        client_sockfd = accept(server_sockfd, (struct sockaddr *)&client_sockaddr, &client_sockaddr_len);
        if (client_sockfd == -1)
        {
            perror("accept");
        }
        while (1)
        {

            recv_bytes = recv(client_sockfd, recv_buf, sizeof(recv_buf), 0);
            if (recv_bytes == -1)
            {
                perror("recv");
            }
            else if (recv_bytes == 0)
            {
                printf("closed by peer\n");
                break;
            }
            else
            {
                printf("%s\n", recv_buf);
            }
            send_bytes = send(client_sockfd, send_buf, strlen(send_buf), 0);
            if (send_bytes == -1)
            {
                perror("send");
            }
        }
    }
    close(server_sockfd);
    return 0;
}

测试用客户端:

#define _GNU_SOURCE
#include <stdio.h>
#include <sys/socket.h>
#include <sys/un.h>
#include <arpa/inet.h>
#include <netinet/in.h>
#include <string.h>
#include <unistd.h>
#include <stdlib.h>
#include <time.h>
#include <fcntl.h>
#define SERVER_IP "192.168.142.132"
#define SERVER_PORT 50001
int set_non_blocking(int sockfd)
{
    int flags = fcntl(sockfd, F_GETFL, 0);
    if (flags == -1)
    {
        perror("fcntl(F_GETFL)");
    }

    flags |= O_NONBLOCK;

    if (fcntl(sockfd, F_SETFL, flags) == -1)
    {
        perror("fcntl(F_SETFL)");
    }

    return 0;
}
int main()
{
    int client_sockfd;
    struct sockaddr_in server_sockaddr, client_sockaddr;
    memset(&server_sockaddr, 0, sizeof(server_sockaddr));
    memset(&client_sockaddr, 0, sizeof(client_sockaddr));
    socklen_t client_sockaddr_len = sizeof(client_sockaddr);
    ssize_t send_bytes, recv_bytes;
    char send_buf[1024] = "he$$$llo ser$$$ver !!!";
    char recv_buf[1024] = {0};
    client_sockfd = socket(AF_INET, SOCK_STREAM, 0);
    if (client_sockfd == -1)
    {
        perror("socket");
    }
    inet_pton(AF_INET, SERVER_IP, &server_sockaddr.sin_addr.s_addr);
    server_sockaddr.sin_port = htons(SERVER_PORT);
    server_sockaddr.sin_family = AF_INET;
    if (connect(client_sockfd, (struct sockaddr *)&server_sockaddr, sizeof(server_sockaddr)) == -1)
    {
        perror("connect");
    }
    while (1)
    {
        send_bytes = send(client_sockfd, send_buf, strlen(send_buf), 0);
        if (send_bytes == -1)
        {
            perror("send");
        }
        recv_bytes = recv(client_sockfd, recv_buf, sizeof(recv_buf), 0);
        if (recv_bytes == -1)
        {
            perror("recv");
        }
        printf("%s\n", recv_buf);
        sleep(3);
    }
    close(client_sockfd);
    return 0;
}

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

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

相关文章

LM4890S 音频功率放大器 2.2-5.5V 防倒充电路

LM4890S是一款音频功率放大器&#xff0c;适用于蓝牙耳机和其他便携式设备。它的作业电压为2.2V至5.5V&#xff0c;输出功率为1W x 1 8Ω。该产品具有恒定电流/恒定电压线性操控&#xff0c;热反应可对充电电流进行自动调理&#xff0c;以限制芯片温度。此外&#xff0c;LM489…

【C++入门】初识C++

&#x1f49e;&#x1f49e; 前言 hello hello~ &#xff0c;这里是大耳朵土土垚~&#x1f496;&#x1f496; &#xff0c;欢迎大家点赞&#x1f973;&#x1f973;关注&#x1f4a5;&#x1f4a5;收藏&#x1f339;&#x1f339;&#x1f339; &#x1f4a5;个人主页&#x…

5.动态规划

1.背包问题 (1)0/1背包问题 01背包问题即每个物品只能选1个 考虑第i件物品&#xff0c;当j<w[i]时&#xff0c;f[i][j]f[i-1][j]&#xff0c;当j>w[i]时&#xff0c;此时有两种选择&#xff0c;选择第i件物品和不选第i件物品。此时f[i][j]max(f[i-1][j],f[i-1][j-w[i]]v…

硬件工程师职责与核心技能有哪些?

作为一个优秀的硬件工程师&#xff0c;必须要具备优秀的职业技能。那么&#xff0c;有些刚入行的工程师及在校的学生经常会问到&#xff1a;硬件工程师需要哪些核心技能&#xff1f;要回答这个问题&#xff0c;首先要明白硬件工程师的职责&#xff0c;然后才能知道核心技能要求…

c语言游戏实战(7):扫雷(下)

前言&#xff1a; 扫雷是一款经典的单人益智游戏&#xff0c;它的目标是在一个方格矩阵中找出所有的地雷&#xff0c;而不触碰到任何一颗地雷。在计算机编程领域&#xff0c;扫雷也是一个非常受欢迎的项目&#xff0c;因为它涉及到许多重要的编程概念&#xff0c;如数组、循环…

E-魔法猫咪(遇到过的题,做个笔记)

题解&#xff1a; 来自学长们思路&#xff1a; 其中一种正解是写单调队列。限制队列内的数单调递增&#xff0c;方法为每当新来的数据比当前队尾数据小时队 尾出列&#xff0c;直到能够插入当前值&#xff0c;这保证了队头永远是最小值。因此总体思路是队尾不断插入新值的同时 …

C++函数匹配机制

函数匹配 在大多数情况下&#xff0c;我们容易确定某次调用应该选用哪个重载函数。 然而&#xff0c;当几个重载函数的形参数量相等以及某些形参的类型可以由其他类型转换得来时&#xff0c;这项工作就不那么容易了。 以下面这组函数及其调用为例&#xff1a; void f(); vo…

【介绍什么是DDOS】

&#x1f308;个人主页:程序员不想敲代码啊 &#x1f3c6;CSDN优质创作者&#xff0c;CSDN实力新星&#xff0c;CSDN博客专家 &#x1f44d;点赞⭐评论⭐收藏 &#x1f91d;希望本文对您有所裨益&#xff0c;如有不足之处&#xff0c;欢迎在评论区提出指正&#xff0c;让我们共…

ShareX 截图工具详细使用心得

一、软件介绍 ShareX是一款免费的开源程序。不仅可以截图&#xff0c;还可以录屏&#xff0c;自动添加水印和阴影&#xff0c;除此之外&#xff0c;还有很多很多&#xff0c;比如OCR识别、屏幕录制、颜色拾取、哈希检查、修改DNS、尺子功能、显示器测试等等。 二、下载安装 …

C++——list类及其模拟实现

前言&#xff1a;这篇文章我们继续进行C容器类的分享——list&#xff0c;也就是数据结构中的链表&#xff0c;而且是带头双向循环链表。 一.基本框架 namespace Mylist {template<class T>//定义节点struct ListNode{ListNode<T>* _next;ListNode<T>* _pre…

Stable Diffusion扩散模型推导公式的基础知识

文章目录 1、独立事件的条件概率2、贝叶斯公式、先验概率、后验概率、似然、证据3、马尔可夫链4、正态分布 / 高斯分布5、重参数化技巧6、期望7、KL散度 、高斯分布的KL散度8、极大似然估计9、ELBO :Evidence Lower Bound10、一元二次方程 1、独立事件的条件概率 A 和 B 是两个…

Rredis缓存常见面试题

文章目录 1.什么是缓存穿透&#xff0c;怎么解决2.什么是缓存击穿&#xff0c;怎么解决3.什么是缓存雪崩&#xff0c;怎么解决4.双写一致性问题5.redisson添加的排他锁是如何保证读写、读读互斥的6.为什么不使用延迟双删7.redis做为缓存&#xff0c;数据的持久化是怎么做的8.re…

【热门话题】文言一心与ChatGPT-4:一场跨时代智能对话系统的深度比较

&#x1f308;个人主页: 鑫宝Code &#x1f525;热门专栏: 闲话杂谈&#xff5c; 炫酷HTML | JavaScript基础 ​&#x1f4ab;个人格言: "如无必要&#xff0c;勿增实体" 文章目录 文言一心与ChatGPT-4&#xff1a;一场跨时代智能对话系统的深度比较一、技术背景…

成绩管理系统|基于springboot成绩管理系统的设计与实现(附项目源码+论文)

基于springboot成绩管理系统的设计与实现 一、摘要 传统办法管理信息首先需要花费的时间比较多&#xff0c;其次数据出错率比较高&#xff0c;而且对错误的数据进行更改也比较困难&#xff0c;最后&#xff0c;检索数据费事费力。因此&#xff0c;在计算机上安装毕业设计成绩管…

HarmonyOS应用开发ArkUI(TS)电商项目实战

项目介绍 本项目基于 HarmonyOS 的ArkUI框架TS扩展的声明式开发范式&#xff0c;关于语法和概念直接看官网官方文档地址&#xff1a;基于TS扩展的声明式开发范式&#xff0c; 工具版本&#xff1a; DevEco Studio 3.1 Canary1 SDK版本&#xff1a; 3.1.9.7&#xff08;API V…

海外媒体软文发稿:带动海外宣发新潮流,迈向国际舞台

引言 随着全球化的发展&#xff0c;越来越多的中国企业希望在国际舞台上展示自己的实力。而海外媒体软文发稿作为一种全新的海外宣传方式&#xff0c;正逐渐成为带动海外宣发新潮流的有力工具。本文将探讨海外媒体软文发稿的优势和如何迈向国际舞台。 海外媒体软文发稿的优势…

代码随想录阅读笔记-二叉树【最大二叉树】

题目 给定一个不含重复元素的整数数组。一个以此数组构建的最大二叉树定义如下&#xff1a; 二叉树的根是数组中的最大元素。左子树是通过数组中最大值左边部分构造出的最大二叉树。右子树是通过数组中最大值右边部分构造出的最大二叉树。 通过给定的数组构建最大二叉树&#x…

linux系统负载对系统的意义

负载平均值的含义 负载平均值是通过uptime和top命令显示的三个数字&#xff0c;分别代表不同时间段的平均负载&#xff08;1分钟、5分钟和15分钟的平均值&#xff09;。这三个数字越低越好&#xff0c;较高的数字意味着系统可能存在问题或过载。然而&#xff0c;并没有一个固定…

男生穿什么裤子最帅?必备的男生裤子推荐

每个人都想拥有很多条好看质量又好的裤子。不过市面上有太多服装品牌&#xff0c;甚至还有不少劣质的衣裤&#xff0c;穿洗两遍之后就出现松垮、变形的情况。为了能够让大家可以选到合适的衣裤&#xff0c;我自费购买了多个品牌的裤子&#xff0c;并给出大家测评结果。 购买到质…

网站访问502,网站服务器崩溃,比较常见几个的原因

其实&#xff0c;配置再好的服务器也难免在使用过程中出现一些故障&#xff0c;造成宕机。 服务器一旦出现故障&#xff0c;影响到用户实时访问网站&#xff0c;造成用户流失&#xff0c;如果在企业的销售高峰期&#xff0c;则将直接影响到商业利润&#xff0c;而且不仅影响外…