linux socket select(tcp)

2023-05-16

1. 概述:
该demo主要实现了linux下通过select(tcp)方式的socket并发通讯,相关接口介绍可以参考<<UNIX环境高级编程>>
在这里插入图片描述在这里插入图片描述

2. 场景(服务端(一)<----(多)客户端):
1) 服务端:
输入:服务端监听的port;
接收到客户端发送的消息,打印到终端,并且回复客户端"ok"
2) 客户端:
输入:客户端的port,以及服务端ip和port;
输入消息内容,将消息发送给服务端,阻塞直到接收到服务端的返回信息,并打印到终端(超时未接收到时,报错);
当输入消息内容为"exit"时,选择退出客户端进程

3. 测试:
SELECT(TCP) 服务端
在这里插入图片描述
客户端1
在这里插入图片描述
客户端2
在这里插入图片描述

/*
demo_select_client.c
网络编程demo(select客户端)

*/
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <unistd.h>
#include <signal.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>

#include "lpm.h"

void signal_func() { 
    printf(RED BLINK"Server Close\n\n"NONE);
}

int main(int argc, char **argv){

    int ret;
    int client_fd = -1;
    char client_port[8];
    char server_ip[32];
    char server_port[8];
    char buf[MAX_BUF];

    printf(RED BLINK"SELECT 客户端\n"NONE);
    printf("请输入客户端 port\t");
    scanf("%s", client_port);
    printf("请输入服务端 ip  \t");
    scanf("%s", server_ip);
    printf("请输入服务端 port\t");
    scanf("%s", server_port);
    printf("\n");

    /*
    SIGPIPE : 在reader中止之后写Pipe的时候发送
    SIG_DFL : 默认信号处理程序
    SIG_IGN : 忽略信号的处理程序
    signal(SIGPIPE,SIG_IGN);
    对一个对端已经关闭的socket调用两次读/写, 第二次将会生成SIGPIPE信号, 该信号默认结束进程
    */
    if(SIG_ERR == signal(SIGPIPE,signal_func)){
        perror("signal:");
        return -1;
    }

    /*
    协议域 : AF_INET(IPV4) AF_INET6(IPV6) AF_LOCAL AF_ROUTE 等等
    socket类型 : SOCK_STREAM(TCP) SOCK_DGRAM(UDP) 等等
    0 : 表示协议域和socket类型选择默认的协议, 通常是0
    */
    client_fd = socket(AF_INET, SOCK_STREAM, 0);
    if(client_fd < 0){
        /*printf("socket:%s\n", strerror(errno));*/
        perror("socket:");
        return -1;
    }

    /*
    应用协议 : SOL_SOCKET(套接字) IPPROTO_TCP IPPROTO_IP...
    设置项 : SO_REUSEADDR(是否可以重用bind的地址)...
    opt : 一些设置项指的是开关...
    */
    int opt = 1;
    ret = setsockopt(client_fd, SOL_SOCKET, SO_REUSEADDR, (void *)&opt, sizeof(opt));
    if(ret < 0){
        perror("setsockopt");
        close(client_fd);
        return -1;
    }

    /*
    设置项 : SO_RCVTIMEO(为recv设置超时时间)...
    */
    struct timeval tv;
    tv.tv_sec = RECV_TIME_OUT;
    tv.tv_usec = 0;
    ret = setsockopt(client_fd, SOL_SOCKET, SO_RCVTIMEO, (void *)&tv, sizeof(tv));
    if(ret < 0){
        perror("setsockopt");
        close(client_fd);
        return -1;
    }

    struct sockaddr_in client_addr;
    memset(&client_addr, 0x0, sizeof(struct sockaddr_in));
    client_addr.sin_family = AF_INET;
    client_addr.sin_port = htons(atoi(client_port));
    client_addr.sin_addr.s_addr = htonl(INADDR_ANY);

    ret = bind(client_fd, (struct sockaddr*)&client_addr, sizeof(client_addr));
    if(ret < 0){
        perror("bind:");
        close(client_fd);
        return -1;
    }

    struct sockaddr_in server_addr;
    memset(&server_addr, 0x0, sizeof(struct sockaddr_in));
    server_addr.sin_family = AF_INET;
    server_addr.sin_port = htons(atoi(server_port));
    inet_pton(AF_INET, server_ip, (void *)&(server_addr.sin_addr.s_addr));

    ret = connect(client_fd, (struct sockaddr*)&server_addr, sizeof(server_addr));
    if(ret < 0){
        perror("connect:");
        close(client_fd);
        return -1;
    }

    memset(buf, 0x0, sizeof(buf));
    ret = recv(client_fd, buf, sizeof(buf), 0);
    if(ret > 0){
        printf("%s\n", buf);
        printf("\n");
    }
    else if(ret == 0){
        /* 无可用数据 或者 对等方已经按序结束 */
        printf(RED BLINK"Server Close\n\n"NONE);
        close(client_fd);
        return -1;
    }
    else if(ret < 0){
        perror("recv:");
        close(client_fd);
        return -1;
    }

    while(1){

        memset(buf, 0x0, sizeof(buf));
        printf("请输入消息\t");
        scanf("%s", buf);

        ret = send(client_fd, buf, strlen(buf), 0);
        if(ret < 0){
            perror("send:");
            continue;
        }

        if(strlen(buf) > 0 && strncmp(buf, EXIT, strlen(buf)) == 0){
            break;
        }

        memset(buf, 0x0, sizeof(buf));
        ret = recv(client_fd, buf, sizeof(buf), 0);
        if(ret > 0){
            printf("%s\n", buf);
            printf("\n");
        }
        else if(ret == 0){
            /* 无可用数据 或者 对等方已经按序结束 */
            printf(RED BLINK"Server Close\n\n"NONE);
            break;
        }
        else if(ret < 0){
            perror("recv:");
            continue;
        }
    }

    /*
    SHUT_RD(0) : 关闭sockfd上的读功能
    SHUT_WR(1) : 关闭sockfd的写功能
    SHUT_RDWR(2) : 关闭sockfd的读写功能
    */
    shutdown(client_fd, SHUT_RDWR);

    close(client_fd);

    return 0;
}
/*
demo_select_server.c
网络编程demo(select服务端)

cat /proc/sys/fs/file-max : 表示系统级别的能够打开的文件句柄的数量, 是对整个系统的限制, 并不是针对用户的
ulimit -n : 控制进程级别能够打开的文件句柄的数量

select : 单个进程可监视的fd数量有限制, 默认是1024
         维护一个用来存放大量fd的数据结构,使得用户空间和内核空间在传递该结构时复制开销大
         线性扫描, 即采用轮询的方法, I/O效率较低

*/
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <unistd.h>
#include <signal.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>

#include "lpm.h"

void signal_func() { 
    printf(RED BLINK"Client Close\n\n"NONE);
}

int main(int argc, char **argv){

    int ret;
    int i = 0;
    int server_fd = -1;
    char server_port[8];
    char buf[MAX_BUF];

    printf(RED BLINK"SELECT 服务端\n"NONE);
    printf("请输入服务端 port\t");
    scanf("%s", server_port);
    printf("\n");

    /*
    SIGPIPE : 在reader中止之后写Pipe的时候发送
    SIG_DFL : 默认信号处理程序
    SIG_IGN : 忽略信号的处理程序
    signal(SIGPIPE,SIG_IGN);
    对一个对端已经关闭的socket调用两次读/写, 第二次将会生成SIGPIPE信号, 该信号默认结束进程
    */
    if(SIG_ERR == signal(SIGPIPE,signal_func)){
        perror("signal:");
        return -1;
    }

    /*
    协议域 : AF_INET(IPV4) AF_INET6(IPV6) AF_LOCAL AF_ROUTE 等等
    socket类型 : SOCK_STREAM(TCP) SOCK_DGRAM(UDP) 等等
    0 : 表示协议域和socket类型选择默认的协议, 通常是0
    */
    server_fd = socket(AF_INET, SOCK_STREAM, 0);
    if(server_fd < 0){
        /*printf("socket:%s\n", strerror(errno));*/
        perror("socket:");
        return -1;
    }

    /*
    应用协议 : SOL_SOCKET(套接字) IPPROTO_TCP IPPROTO_IP...
    设置项 : SO_REUSEADDR(是否可以重用bind的地址)...
    opt : 一些设置项指的是开关...
    */
    int opt = 1;
    ret = setsockopt(server_fd, SOL_SOCKET, SO_REUSEADDR, (void *)&opt, sizeof(opt));
    if(ret < 0){
        perror("setsockopt");
        close(server_fd);
        return -1;
    }

    struct sockaddr_in server_addr;
    memset(&server_addr, 0x0, sizeof(struct sockaddr_in));
    server_addr.sin_family = AF_INET;
    server_addr.sin_port = htons(atoi(server_port));
    server_addr.sin_addr.s_addr = htonl(INADDR_ANY);

    ret = bind(server_fd, (struct sockaddr*)&server_addr, sizeof(server_addr));
    if(ret < 0){
        perror("bind:");
        close(server_fd);
        return -1;
    }

    ret = listen(server_fd, MAX_LISTEN);
    if(ret < 0){
        perror("listen:");
        close(server_fd);
        return -1;
    }

    char IP[32];
    int addrLen = sizeof(struct sockaddr);
    int client_fd[MAX_ACCEPT];
    struct sockaddr_in client_addr[MAX_ACCEPT];
    for(i = 0; i < MAX_ACCEPT; i ++){
        client_fd[i] = -1;
        memset(&client_addr[i], 0x0, sizeof(struct sockaddr_in));
    }

    /*
    用于记录超出并发的客户端
    */
    int OverClient_fd = -1;
    struct sockaddr_in OverClient_addr;

    /*
    NULL    : 永远等待
    0       : 根本不等待
    !0      : 等待指定时间
    */
    struct timeval t;
    fd_set fd_set;              //文件集
    int max_fd = server_fd;     //最大文件描述符

    while(1){

        /* select 调用之后需要重新赋值*/
        t.tv_sec = 60;
        t.tv_usec = 0;

        FD_ZERO(&fd_set);                       //清除文件集中相关fd的位
        FD_SET(server_fd, &fd_set);             //设置文件集中相关fd的位
        for(i = 0; i < MAX_ACCEPT; i ++){
            if(client_fd[i] > 0){
                FD_SET(client_fd[i], &fd_set);  //设置文件集中相关fd的位
            }
        }

        switch(select(max_fd+1, &fd_set, NULL, NULL, &t))
        {
            /*出错*/
            case -1:
                perror("select:");
                break;

            /*表示没有描述符准备好*/
            case 0:
                printf("time out!\n");
                break;

            /*表示已经准备好的描述符数*/
            default:
                {
                    if(FD_ISSET(server_fd, &fd_set)){           //测试文件集中相关fd的位是否为真
                        for(i = 0; i <= MAX_ACCEPT; i ++){
                            if(client_fd[i] < 0 && i < MAX_ACCEPT){
                                memset(&client_addr[i], 0x0, sizeof(struct sockaddr_in));
                                client_fd[i] = accept(server_fd, (struct sockaddr *)(&client_addr[i]), (socklen_t *)&addrLen);
                                if(client_fd[i] < 0){
                                    perror("accept:");
                                    break;
                                }

                                memset(IP, 0x0, sizeof(IP));
                                inet_ntop(AF_INET, (void *)&client_addr[i].sin_addr.s_addr, IP, sizeof(IP));
                                printf("客户端 IP  \t%s\n", IP);
                                printf("客户端 port\t%d\n", ntohs(client_addr[i].sin_port));
                                printf("\n");

                                ret = send(client_fd[i], OK, strlen(OK), 0);
                                if(ret < 0){
                                    perror("send:");
                                    close(client_fd[i]);
                                    client_fd[i] = -1;
                                    break;
                                }

                                FD_SET(client_fd[i], &fd_set);  //设置文件集中相关fd的位
                                max_fd = max_fd > client_fd[i] ? max_fd : client_fd[i];
                            }
                            else if(i == MAX_ACCEPT){
                                if(OverClient_fd < 0){
                                    memset(&OverClient_addr, 0x0, sizeof(struct sockaddr_in));
                                    OverClient_fd = accept(server_fd, (struct sockaddr *)(&OverClient_addr), (socklen_t *)&addrLen);
                                    if(OverClient_fd < 0){
                                        perror("accept:");
                                        break;
                                    }

                                    memset(IP, 0x0, sizeof(IP));
                                    inet_ntop(AF_INET, (void *)&OverClient_addr.sin_addr.s_addr, IP, sizeof(IP));
                                    printf("客户端 IP  \t%s\n", IP);
                                    printf("客户端 port\t%d\n", ntohs(OverClient_addr.sin_port));
                                    printf(RED BLINK"Unable to accept\n"NONE);
                                    printf("\n");

                                    ret = send(OverClient_fd, ERROR, strlen(ERROR), 0);
                                    if(ret < 0){
                                        perror("send:");
                                    }

                                    close(OverClient_fd);
                                    OverClient_fd = -1;
                                }
                            }
                        }
                    }
                    else {
                        for(i = 0; i < MAX_ACCEPT; i ++){
                            if(FD_ISSET(client_fd[i], &fd_set)){
                                memset(buf, 0x0, sizeof(buf));
                                memset(IP, 0x0, sizeof(IP));
                                inet_ntop(AF_INET, (void *)&client_addr[i].sin_addr.s_addr, IP, sizeof(IP));
                                printf("客户端 IP  \t%s\n", IP);
                                printf("客户端 port\t%d\n", ntohs(client_addr[i].sin_port));

                                ret = recv(client_fd[i], buf, sizeof(buf), 0);
                                if(ret > 0){
                                    printf("消息      \t%s\n", buf);
                                    printf("\n");
                                }
                                else if(ret == 0){
                                    /* 无可用数据 或者 对等方已经按序结束 */
                                    printf(RED BLINK"Client Close\n\n"NONE);
                                }
                                else if(ret < 0){
                                    perror("recv:");
                                    break;
                                }

                                if(ret == 0 || (strlen(buf) > 0 && strncmp(buf, EXIT, strlen(buf)) == 0)){
                                    FD_CLR(client_fd[i], &fd_set);  //清除文件集中相关fd的位
                                    close(client_fd[i]);
                                    client_fd[i] = -1;
                                    break;
                                }

                                ret = send(client_fd[i], OK, strlen(OK), 0);
                                if(ret < 0){
                                    perror("send:");
                                    break;
                                }
                            }
                        }
                    }
                }
                break;
        }
    }

    /*
    SHUT_RD(0) : 关闭sockfd上的读功能
    SHUT_WR(1) : 关闭sockfd的写功能
    SHUT_RDWR(2) : 关闭sockfd的读写功能
    */
    shutdown(server_fd, SHUT_RDWR);

    for(i = 0; i < MAX_ACCEPT; i ++){
        if(client_fd[i] > 0){
            close(client_fd[i]);
        }
    }
    close(server_fd);

    return 0;
}
/*
lpm.h

*/
#ifndef _LPM_H_

#define _LPM_H_

#define NONE            "\e[0m"
#define BLACK           "\e[0;30m"
#define L_BLACK         "\e[1;30m"
#define RED             "\e[0;31m"
#define L_RED           "\e[1;31m"
#define GREEN           "\e[0;32m"
#define L_GREEN         "\e[1;32m"
#define BROWN           "\e[0;33m"
#define YELLOW          "\e[1;33m"
#define BLUE            "\e[0;34m"
#define L_BLUE          "\e[1;34m"
#define PURPLE          "\e[0;35m"
#define L_PURPLE        "\e[1;35m"
#define CYAN            "\e[0;36m"
#define L_CYAN          "\e[1;36m"
#define GRAY            "\e[0;37m"
#define WHITE           "\e[1;37m" 
#define BOLD            "\e[1m"
#define UNDERLINE       "\e[4m"
#define BLINK           "\e[5m"
#define REVERSE         "\e[7m"
#define HIDE            "\e[8m"
#define CLEAR           "\e[2J"
#define CLRLINE         "\r\e[K" //or "\e[1K\r"

#define MAX_BUF         64

#define RECV_TIME_OUT   3

#define MAX_LISTEN      1

#define MAX_ACCEPT      2

#define OK              "ok"

#define EXIT            "exit"

#define ERROR           "error"

#define LOG()           printf("[%s %s] %s: %s: %d\n", __DATE__, __TIME__, __FILE__, __func__, __LINE__)

#endif
#Makefile
CC := gcc
INCLUDE = -I /home/demo/include/

all:
	$(CC) demo_select_server.c $(INCLUDE) -o demo_select_server -Wall -Werror
	$(CC) demo_select_client.c $(INCLUDE) -o demo_select_client -Wall -Werror

clean:
	rm demo_select_server demo_select_client
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

linux socket select(tcp) 的相关文章

  • 在嵌入式系统上将内核控制台发送到哪里?

    我正在开发一个嵌入式系统 该系统当前通过串行端口 1 上的控制台输出启动 Linux 使用启动加载程序中的控制台启动参数 然而 最终我们将使用这个串行端口 内核控制台输出的最佳解决方案是什么 dev null 能否以某种方式将其放在 pty
  • Apache LOG:子进程 pid xxxx 退出信号分段错误 (11)

    Apache PHP Mysql Linux 注意 子进程 pid 23145 退出信号分段错误 11 tmp 中可能存在 coredump 但 tmp下没有找到任何东西 我怎样才能找到错误 PHP 代码中函数的无限循环导致了此错误
  • 由于 abi::cxx11 符号导致的链接问题?

    我们最近收到一份报告 因为GCC 5 1 libstdc 和双 ABI http gcc gnu org onlinedocs libstdc manual using dual abi html 它似乎Clang 不知道 GCC 内联名称
  • 命名互斥体的 Mono 替代方案

    在 Windows NET 上 命名的互斥体可用于同步多个进程 不幸的是 Mono 在 Linux 上不太支持这一点 他们的发行说明 http www mono project com Release Notes Mono 2 8 Shar
  • 无法连接到 Azure Ubuntu VM - 公钥被拒绝

    我们在 Azure 上使用 Ubuntu VM 一段时间了 很少遇到任何问题 然而 其中一台虚拟机最近出现了问题 出乎意料的是 Ubuntu VM 开始拒绝公钥 ssh i azure key email protected cdn cgi
  • 如何使用 nohup 获取正在运行的程序列表

    我正在通过 SSH 连接访问运行 CentOS linux 发行版 的服务器 由于我无法始终保持登录状态 因此我使用 nohup command 来运行我的程序 我找不到如何获取我开始使用 nohup 的所有程序的列表 工作 只有在我注销之
  • 立即检测客户端与服务器套接字的断开连接

    如何检测客户端已与服务器断开连接 我的代码中有以下代码AcceptCallBack method static Socket handler null public static void AcceptCallback IAsyncResu
  • 通过名称获取进程ID

    我想在 Linux 下获得一个给定其名称的进程 ID 有没有一种简单的方法可以做到这一点 我还没有在 C 上找到任何可以轻松使用的东西 如果追求 易于使用 char buf 512 FILE cmd pipe popen pidof s p
  • GCC 详细模式输出解释

    我是 Linux 新手 谁能向我解释一下我的 hello world 程序的以下详细模式输出 另外 这些文件是做什么用的crt1 o crti o crtend o crtbegin o and crtn o and lc and lgcc
  • 用于 e NetworkManager VPN 连接的 dbus 信号处理程序

    我需要开发一些在建立 VPN 连接时执行的 python 代码 VPN 由 NetworkManager 控制 我试图弄清楚如何为此使用 NM DBUS 事件 使用 dbus monitor system 我能够识别连接信号 signal
  • 在 MacOS 上构建需要 net461 的 dotnet SDK 项目的最简单方法

    我有一个 dotnet SDK sln and a build proj with
  • TCP打孔问题

    我尝试使用 Python 3 中概述的原则为防火墙编写一个基本的 TCP 打孔器本文 http www bford info pub net p2pnat index html 不过 我无法连接任何东西 这是代码 usr bin pytho
  • 如何使用 PyAudio 选择特定的输入设备

    通过 PyAudio 录制音频时 如何指定要使用的确切输入设备 我的电脑有两个麦克风 一个内置 一个通过 USB 我想使用 USB 麦克风进行录音 这流类 https people csail mit edu hubert pyaudio
  • PHP mail() 函数不发送邮件

    我有一个小问题 我正在使用一个工作脚本 在我的测试帐户 共享服务器上工作 使用 mail 函数通过 PHP 发送邮件 我刚刚得到了一个专用服务器 但我还无法让该功能发挥作用 在过去的 10 个小时左右的时间里 我阅读了有关 BIND 用于
  • 删除 Python 中某些操作的 root 权限

    在我的 Python 脚本中 我执行了一些需要 root 权限的操作 我还创建并写入文件 我不想由 root 独占所有 而是由运行我的脚本的用户独占所有 通常 我使用以下命令运行脚本sudo 有办法做到上述吗 您可以使用以下方式在 uid
  • SQL Server 转换选择一列并将其转换为字符串

    是否可以编写一条从表中选择列并将结果转换为字符串的语句 理想情况下 我希望有逗号分隔的值 例如 假设 SELECT 语句看起来像这样 SELECT column FROM table WHERE column lt 10 结果是一列包含值的
  • PHP 日志文件颜色

    我正在编写一个 PHP 日志文件类 但我想为写入文件的行添加颜色 我遇到的问题是颜色也会改变终端的颜色 我想要实现的是仅更改写入日志文件的行的颜色 class logClass extends Singleton private funct
  • select() 可以在 Windows 下使用 Python 中的文件吗?

    我正在尝试在 Windows 下运行以下 python 服务器 An echo server that uses select to handle multiple clients at a time Entering any line o
  • 在 MacO 和 Linux 上安装 win32com [重复]

    这个问题在这里已经有答案了 我的问题很简单 我可以安装吗win32com蟒蛇API pywin32特别是 在非 Windows 操作系统上 我一直在Mac上尝试多个版本pip install pywin32 都失败了 下面是一个例子 如果你
  • 在Linux中使用C/C++获取机器序列号和CPU ID

    在Linux系统中如何获取机器序列号和CPU ID 示例代码受到高度赞赏 Here http lxr linux no linux v2 6 39 arch x86 include asm processor h L173Linux 内核似

随机推荐

  • 第四次游戏革命:全息游戏

    最近一个月 xff0c 把国内外十数款单机 网游大作横扫一遍 xff0c 感慨颇多 国内游戏 xff0c 抄袭遍地 xff0c 十足的坑爹 xff0c 浪费青春 反观国外 xff0c 韩国网游经典而耐玩 xff0c C9 43 洛奇英雄传
  • UART, IIC, SCI, SPI, 232, 485, 422, CAN, SDIO, GPIO, MODBUS, TCP/IP汇总简介

    UART IIC SCI SPI 232 485 422 CAN SDIO GPIO MODBUS TCP IP汇总简介 UART xff1a Universal Asynchronous Receiver Transmitter xff1
  • 二维数组的定义

    1 概念 二维数组就是一种数组的数组 xff0c 其本质上还是一个一维数组 xff0c 只是它的数据元素又是一个一维数组 如果你对这个概念想象不出来 xff0c 给大家举个栗子 xff0c 相信吸烟的同学一下子就会明白 一根烟 61 一个变
  • nginx不转发http header问题解决

    文章整理自网络 作者 64 loongshawn xff1a http blog csdn net loongshawn article details 78199977 xff0c 建议读者阅读原文 xff0c 确保获得完整的信息 1 问
  • SimpleFOC(二)—— 快速入门 (开环控制)

    目录 一 硬件介绍 1 驱动板版本说明 2 驱动板跳线 3 硬件准备 4 硬件连接 二 软件操作 1 安装Arduino IDE 2 安装SimpleFOC library 3 打开示例程序 三 电机控制 1 程序下载 2 开环速度控制 3
  • curl参数详解

    原文 xff1a http blog csdn net yanhui wei article details 21530811 cURL可以使用URL的语法模拟浏览器来传输数据 xff0c 因为它是模拟浏览器 xff0c 因此它同样支持多种
  • 嵌入式单片机基础篇(十八)之ILI9341 液晶控制器

    ILI9341 液晶控制器详解 1 ILI9341 液晶控制器简介 xff1a ILI9341 液晶控制器自带显存 xff0c 其显存总大小为 172800 xff08 24032018 8 xff09 xff0c 即 18 位模式 xff
  • LOTO 示波器软件功能演示——RS232串口解码

    LOTO 示波器软件功能演示 RS232串口解码 我们今天演示一下怎么用LOTO示波器对串口进行解码 xff0c 使用了一个USB转串口的设备 xff0c 来产生串口数据 xff0c 用OSCA02 LOTO示波器演示 示波器我们只需要建立
  • ROS 下实现相机图像采集与图像传输到服务器,socket图传

    前言 本文介绍一种Qt下进行ROS开发的完美方案 xff0c 同时给出一个使用TCPROS进行图像传输的一个例子 xff0c 使用的是ros industrial的Levi Armstrong在2015年12月开发的一个Qt插件ros qt
  • Digest来验证

    Apache默认使用basic模块验证 xff0c 但它只是明文验证 Digest验证 xff0c 是用md5摘要值进行对比 httpRequest getAuthType 方法 xff0c 可以得到网页的验证方式request BASIC
  • 使用vscode创建C++工程

    1 推荐文件目录 即一个C 43 43 工程文件中包含 vscode文件夹 build文件夹 include文件夹 src文件夹以及一个CMakeLists txt文件 2 vscode文件夹 vscode文件夹一般应该包含三个配置文件 x
  • 游戏中所存在的“真随机”与“伪随机”

    写这篇随笔的动机 xff0c 在于最近看了不少对于游戏中概率事件的提问 在这些相关讨论里 xff0c 总是能频繁看到 真随机 和 伪随机 这两个词汇 其中最常见的句子则莫过于宝典一般的 程序里没有真随机 这句话本身当然是没有问题的 但是大多
  • C语言socket编程----实现TCP通信

    TCP IP协议 xff08 Transmission Control Protocol Internet Protocol xff09 叫做传输控制 网际协议 xff0c 又叫网络通信协议 实际上 xff0c 它包含上百个功能的协议 xf
  • rtthread移植实现uorb

    uORB Micro Object Request Broker 微对象请求代理器 是PX4 Pixhawk系统中非常重要且关键的一个模块 xff0c 它肩负了整个系统的数据传输任务 xff0c 所有的传感器数据 GPS PPM信号等都要从
  • 环形缓冲区(ringbuffer)

    环形缓冲区 xff08 ringbuffer xff09 环形缓冲区是嵌入式系统中十分重要的一种数据结构 xff0c 比如在串口处理中 xff0c 串口中断接收数据直接往环形缓冲区丢数据 xff0c 而应用可以从环形缓冲区取数据进行处理 x
  • 大屏可视化数据面板分格渐变进度条、数字翻牌器及其刷新动效实现

    数据可视化大屏是当前可视化领域的一项热门应用 xff0c 通常可以分为信息展示类 数据分析类及监控预警类 这类应用的视觉设计通常效果炫酷 xff0c 动效丰富 xff0c 有时候一些页面布局和动画实现会对前端人员有一定的挑战性 xff0c
  • Fiddler系列教程1:初识Http协议抓包工具

    1 Fiddler简介 Fiddler是用一款使用C 编写的http协议调试代理工具 它支持众多的http调试任务 xff0c 能够记录并检查所有你的电脑和互联网之间的http通讯 xff0c 可以设置断点 xff0c 查看所有的 进出 F
  • http 认证 basic 和 digest

    HTTP协议 RFC2616 的两种认证机制 Basic和Digest SIP类似Http协议 其认证模式也一样 Http协议 xff08 RFC 2616 xff09 规定可以采用Basic模式和摘要模式 xff08 Digest sch
  • 优秀的C/C++框架和库整理,值得收藏

    1 ACE 庞大 复杂 xff0c 适合大型项目 开源 免费 xff0c 不依赖第三方库 xff0c 支持跨平台 http www cs wustl edu schmidt ACE html 2 Asio Asio基于Boost开发的异步I
  • linux socket select(tcp)

    1 概述 xff1a 该demo主要实现了linux下通过select tcp 方式的socket并发通讯 xff0c 相关接口介绍可以参考 lt lt UNIX环境高级编程 gt gt 2 场景 服务端 一 lt 多 客户端 xff1a