Windows TCP 通信测试_1

一、单对单通信测试

应用函数 socket、bind、connect、listen、accept、recv、send(win下的函数)等

1、客户端demo client.cpp

#include<WINSOCK2.H>
#include<STDIO.H>
#include<iostream>
#include<cstring>
using namespace std;
#pragma comment(lib, "ws2_32.lib")
#define CONNECT_COUNT_MAX  20
#define CONNECT_INTERVAL_MS 1000
int main()
{
	WORD sockVersion = MAKEWORD(2, 2);
	WSADATA data;
	if(WSAStartup(sockVersion, &data)!=0)
	{
		return 0;
	}
while(true){
        SOCKET sclient = socket(AF_INET, SOCK_STREAM, 0);//IPPROTO_TCP
        if(sclient == INVALID_SOCKET)
        {
            printf("invalid socket!");
            WSACleanup();
            system("pause");
            return 0;
        }
        // 设定客户端协议族、端口、ip信息
        sockaddr_in clientAddr;
        clientAddr.sin_family = AF_INET;
        clientAddr.sin_port = htons(8888);
        clientAddr.sin_addr.S_un.S_addr = inet_addr("127.6.6.2");

        if (bind(sclient,(struct sockaddr *)&clientAddr,sizeof(clientAddr))==0){    //client信息绑定到clientfd上
            printf("client socket bind success\n");
        }; 
        
        // 设定服务端协议族、端口、ip信息
        sockaddr_in serAddr;
        serAddr.sin_family = AF_INET;
        serAddr.sin_port = htons(9999);
        serAddr.sin_addr.S_un.S_addr = inet_addr("127.6.6.254");
    // while(true){
        int connect_flag_ = 0; //TCP connect 结果标志位
        int connect_count_ = 0; //TCP connect 重试次数
        while(connect_count_ <  CONNECT_COUNT_MAX){
            connect_flag_ = connect(sclient, (sockaddr *)&serAddr, sizeof(serAddr));
            if(connect_flag_ == SOCKET_ERROR){  //连接失败 
                printf("connect error, retry time: [%d]!\n", connect_count_);
                connect_count_ ++;
                Sleep(CONNECT_INTERVAL_MS); //休眠一段时间再重试connect
            }else{
                printf("connect success, retry time: [%d]!\n", connect_count_);
                break;
            }
        }
        if(connect_count_ == CONNECT_COUNT_MAX){      //达到了最大connect次数,结束
            printf("connect error ,retry finshed!\n");
            closesocket(sclient);
            WSACleanup();
            system("pause");
            return 0;
        }
        string client_id = "[client_1] : ";
        bool input_flag_ = false;   //选择传入手动输入消息还是默认消息

        // string data;
        const char * sendData;
        int client_send_flag_ = 0;
        if(input_flag_){
            char data[128] = {};
            cin >> data;
            sendData = (client_id + data).c_str();   //string转const char*
            cout << "client msg : " << sendData << endl;
            client_send_flag_ = send(sclient, sendData, strlen(sendData), 0);
        }else{
            char *sendData = "我是北方的狼";
            cout << "client msg : " << sendData << endl;
            client_send_flag_ = send(sclient, sendData, strlen(sendData), 0);
            Sleep(3000);
        };

        //send()用来将数据由指定的socket传给对方主机
        /*
            int send(int s, const void * msg, int len, unsigned int flags)
            s为已建立好连接的socket,msg指向数据内容,len则为数据长度,参数flags一般设0
            成功则返回实际传送出去的字符数,失败返回-1,错误原因存于error 
        */
        if(client_send_flag_ >= 0){   //send 失败或成功提示
            printf(" send msg success!\n");
        }else{
            printf(" send msg fail!!\n");
        }

        char recData[255];
        int ret = recv(sclient, recData, 255, 0);
            /*  函数原型:int recv( SOCKET s, char *buf, int len, int flags) 
                    功能: 用recv函数从TCP连接的另一端接收数据。 
                    参数一:指定接收端套接字描述符; 
                    参数二:指明一个缓冲区,该缓冲区用来存放recv函数接收到的数据; 
                    参数三:指明buf的长度; 
                    参数四 :一般置为0。 
                返回值 : 读出来的字节大小
            */
        if(ret>0){
            recData[ret] = 0x00;
            printf(recData);
        } 
        closesocket(sclient);
    }
    
	// }
	
	WSACleanup();
	return 0;
	
}

特点

1、轮询重试

2、测试时可先于服务端启动

3、发送消息可以选择周期发或者手动输入;(bug:目前手动输入过长字符会有发送异常的情况出现)

2、服务端demo server.cpp

#include <stdio.h>  
#include <winsock2.h>  
#include<iostream>
#include<cstring>
#include<Windows.h>
#include<future>

// #include<promise>

#pragma comment(lib,"ws2_32.lib")  

#define DATA_SIZE 1024
using namespace std;
#if 0
future_status getRecvResult(SOCKET sClientfd, char *revData , int* ret){
    std::future<int> ret1 = std::async(std::launch::async, recv , sClientfd, revData, DATA_SIZE, 0); //异步调用
    std::future_status sts = ret1.wait_for(std::chrono::seconds(1)); //设置超时时间 s'
    *ret = ret1.get();
    return sts;
}
#endif
int main(int argc, char* argv[])  
{  
    //初始化WSA  
    WORD sockVersion = MAKEWORD(2,2);  
    WSADATA wsaData;  
    if(WSAStartup(sockVersion, &wsaData)!=0)  
    {  
        return 0;  
    }  
  
    //创建套接字  
    SOCKET slisten = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);  
    if(slisten == INVALID_SOCKET)  
    {  
        printf("socket error !");  
        return 0;  
    }  
  
    //绑定IP和端口  
    sockaddr_in serveraddr;  
    serveraddr.sin_family = AF_INET;  
    serveraddr.sin_port = htons(9999);  
    // serveraddr.sin_addr.S_un.S_addr = INADDR_ANY;   //可以是系统端口 
    serveraddr.sin_addr.S_un.S_addr = inet_addr("127.6.6.254");
    if(bind(slisten, (LPSOCKADDR)&serveraddr, sizeof(serveraddr)) == SOCKET_ERROR)  
    {  
        printf("bind error !");  
    }  
  
    //开始监听  
    if(listen(slisten, 5) == SOCKET_ERROR)
    {  
        printf("listen error !");  
        return 0;  
    }  
  
    //循环接收数据  
    SOCKET sClientfd;  
    sockaddr_in remoteAddr;  
    int nAddrlen = sizeof(remoteAddr);  
      
    while (true)  
    {  
        printf("等待连接...\n");  
        sClientfd = accept(slisten, (SOCKADDR *)&remoteAddr, &nAddrlen);
        /*
        listen监听客户端来的链接,accept将客户端的信息绑定到一个socket上,也就是给客户端创建一个socket,通过返回值返回给我们客户端的socket。
        accept函数的第一个参数为服务器的socket描述字,第二个参数为指向struct sockaddr *的指针,用于返回客户端的协议地址,第三个参数为协议地址的长度。
        如果accpet成功,那么其返回值是由内核自动生成的一个全新的描述字,代表与返回客户的TCP连接。
            成功:
                返回值就是给客户端包好的socket; 与客户端通信就靠这个
                
            失败:
                返回INVALIE_SOCKET; 通过WSAGetLastError()得到错误码
        */
    // while (true)  
    // {    
        if(sClientfd <= 0 )  
        {  
            printf("accept error !");
            cout << WSAGetLastError() << endl;
            Sleep(2000);  
            continue;  
        }  
        printf("接受到一个连接:%s \r\n", inet_ntoa(remoteAddr.sin_addr));  
          
        //接收数据 
        // char revData[2] = {}; 
        // int data_size = sizeof(revData);
        string tmp = "";
        
        int ret = 0;
        do    //怕数据太长,一次recv copy不完,为了把client端一次发的的消息拼接、打印
        {
            char revData[DATA_SIZE] = {}; 
            revData[DATA_SIZE] = 0x00;   // 抵消数组后面的字符
            cout << "start " << revData <<endl;

            /*  // 下面的异步超时操作是为了把client端一次发的的消息拼接、打印(主要是一条消息最后一次数据大小 = DATA_SIZE的情况);;不过有bug,没想好怎么解决
                std::future<int> ret1 = std::async(std::launch::async, recv , sClientfd, revData, DATA_SIZE, 0); //异步调用
                std::future_status sts = ret1.wait_for(std::chrono::seconds(1)); //设置超时时间 s'
            
                if(sts == std::future_status::ready){
                    // ret = ret1.get();
                    printf("超时时间内收到了回复\n");
                }else if(sts == std::future_status::timeout){
                    printf("设定时间内无消息\n");
                    ret = 0;
                    break;      
            }
            */
            ret = recv(sClientfd, revData, DATA_SIZE, 0);  // 在主线程操作,如果recv大小正好是个整数包,一直阻塞在这,不会给该client返回消息;
                //这是上面do while想做的事情可能带来的影响
            cout << "server ret = " << ret << endl;
            if(ret <= 0){
                printf("receive msg error\n");
                // system("pause");
                Sleep(3000);
                break;
            }
            
            if(ret  < DATA_SIZE){
                revData[ret] = 0x00;
                cout << "last phase " << revData <<endl;
            }
            tmp = tmp + revData;
            cout << "tmp :[" << tmp << "] , leave"<<endl;

        } while (ret  == DATA_SIZE);
        cout <<'['<< tmp << ']'<<endl;

  
        //发送数据  
        const char * sendData = "你好,TCP客户端!\n";  
        send(sClientfd, sendData, strlen(sendData), 0);  
        closesocket(sClientfd);
        Sleep(1000);
    }  
      
    closesocket(slisten);  
    WSACleanup();  
    return 0;  
} 

特点

1、为了一次打印client发送的消息,做了一点愚蠢的特殊处理。。。

2、没有做超时,socket没有消息会一直阻塞;

3、验证

编译:

g++ server.cpp -o server -lwsock32

g++ client.cpp -o client -lwsock32  

二、多个客户端通信测试(有bug)

新增加select的使用

1、客户端新增demo client_2.cpp

#include<WINSOCK2.H>
#include<STDIO.H>
#include<iostream>
#include<cstring>
using namespace std;
#pragma comment(lib, "ws2_32.lib")
#define CONNECT_COUNT_MAX  20
#define CONNECT_INTERVAL_MS 1000
#define INPUT_FLAG   true
int main()
{
	WORD sockVersion = MAKEWORD(2, 2);
	WSADATA data;
	if(WSAStartup(sockVersion, &data)!=0)
	{
		return 0;
	}
while(true){
        SOCKET sclient = socket(AF_INET, SOCK_STREAM, 0);//IPPROTO_TCP
        if(sclient == INVALID_SOCKET)
        {
            printf("invalid socket!");
            WSACleanup();
            system("pause");
            return 0;
        }
        // 设定客户端协议族、端口、ip信息
        sockaddr_in clientAddr;
        clientAddr.sin_family = AF_INET;
        clientAddr.sin_port = htons(8888);
        clientAddr.sin_addr.S_un.S_addr = inet_addr("127.6.6.4");

        if (bind(sclient,(struct sockaddr *)&clientAddr,sizeof(clientAddr))==0){    //client信息绑定到clientfd上
            printf("client socket bind success\n");
        }; 
        
        // 设定服务端协议族、端口、ip信息
        sockaddr_in serAddr;
        serAddr.sin_family = AF_INET;
        serAddr.sin_port = htons(9999);
        serAddr.sin_addr.S_un.S_addr = inet_addr("127.6.6.254");
    // while(true){
        int connect_flag_ = 0; //TCP connect 结果标志位
        int connect_count_ = 0; //TCP connect 重试次数
        while(connect_count_ <  CONNECT_COUNT_MAX){
            connect_flag_ = connect(sclient, (sockaddr *)&serAddr, sizeof(serAddr));
            if(connect_flag_ == SOCKET_ERROR){  //连接失败 
                printf("connect error, retry time: [%d]!\n", connect_count_);
                connect_count_ ++;
                Sleep(CONNECT_INTERVAL_MS); //休眠一段时间再重试connect
            }else{
                printf("connect success, retry time: [%d]!\n", connect_count_);
                break;
            }
        }
        if(connect_count_ == CONNECT_COUNT_MAX){      //达到了最大connect次数,结束
            printf("connect error ,retry finshed!\n");
            closesocket(sclient);
            WSACleanup();
            system("pause");
            return 0;
        }
        string client_id = "[client_1] : ";
        bool input_flag_ = INPUT_FLAG;   //选择传入手动输入消息还是默认消息

        // string data;
        const char * sendData;
        int client_send_flag_ = 0;
        if(input_flag_){
            char data[128] = {};
            cin >> data;
            sendData = (client_id + data).c_str();   //string转const char*
            cout << "client msg : " << sendData << endl;
            client_send_flag_ = send(sclient, sendData, strlen(sendData), 0);
        }else{
            char *sendData = "我是北方的羊";
            cout << "client msg : " << sendData << endl;
            client_send_flag_ = send(sclient, sendData, strlen(sendData), 0);
            Sleep(3000);
        };

        //send()用来将数据由指定的socket传给对方主机
        /*
            int send(int s, const void * msg, int len, unsigned int flags)
            s为已建立好连接的socket,msg指向数据内容,len则为数据长度,参数flags一般设0
            成功则返回实际传送出去的字符数,失败返回-1,错误原因存于error 
        */
        if(client_send_flag_ >= 0){   //send 失败或成功提示
            printf(" send msg success!\n");
        }else{
            printf(" send msg fail!!\n");
        }

        char recData[255];
        int ret = recv(sclient, recData, 255, 0);
            /*  函数原型:int recv( SOCKET s, char *buf, int len, int flags) 
                    功能: 用recv函数从TCP连接的另一端接收数据。 
                    参数一:指定接收端套接字描述符; 
                    参数二:指明一个缓冲区,该缓冲区用来存放recv函数接收到的数据; 
                    参数三:指明buf的长度; 
                    参数四 :一般置为0。 
                返回值 : 读出来的字节大小
            */
        if(ret>0){
            recData[ret] = 0x00;
            printf(recData);
        } 
        closesocket(sclient);
    }
    
	// }
	
	
	WSACleanup();
	return 0;
	
}

2、服务端demo server_select.cpp

#include <stdio.h>  
#include <winsock2.h>  
#include<iostream>
#include<cstring>
#include<Windows.h>
#include<future>

// #include<promise>

#pragma comment(lib,"ws2_32.lib")  

#define DATA_SIZE 1024
using namespace std;
#if 0
future_status getRecvResult(SOCKET sClientfd, char *revData , int* ret){
    std::future<int> ret1 = std::async(std::launch::async, recv , sClientfd, revData, DATA_SIZE, 0); //异步调用
    std::future_status sts = ret1.wait_for(std::chrono::seconds(1)); //设置超时时间 s'
    *ret = ret1.get();
    return sts;
}
#endif
int main(int argc, char* argv[])  
{  
    //初始化WSA  
    WORD sockVersion = MAKEWORD(2,2);  
    WSADATA wsaData;  
    if(WSAStartup(sockVersion, &wsaData)!=0)  
    {  
        return 0;  
    }  
  
    //创建套接字  
    SOCKET slisten = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);  
    if(slisten == INVALID_SOCKET)  
    {  
        printf("socket error !");  
        return 0;  
    }  
  
    //绑定IP和端口  
    sockaddr_in serveraddr;  
    serveraddr.sin_family = AF_INET;  
    serveraddr.sin_port = htons(9999);  
    // serveraddr.sin_addr.S_un.S_addr = INADDR_ANY;   //可以是系统端口 
    serveraddr.sin_addr.S_un.S_addr = inet_addr("127.6.6.254");
    if(bind(slisten, (LPSOCKADDR)&serveraddr, sizeof(serveraddr)) == SOCKET_ERROR)  
    {  
        printf("bind error !");  
    }  
  
    //开始监听  
    if(listen(slisten, 5) == SOCKET_ERROR)  
    {  
        printf("listen error !");  
        return 0;  
    }  
  
    //循环接收数据  
    SOCKET sClientfd;  
    sockaddr_in remoteAddr;  
    int nAddrlen = sizeof(remoteAddr);  
    // select 相关设置
    fd_set rfds, readset, wfds, writeset;

    SOCKET Clientfd[1024]; 
    int maxfd = slisten; 
	FD_ZERO(&rfds);
    FD_ZERO(&wfds);
	FD_SET(slisten, &rfds);
    while (true)  
    {
        printf("等待连接...\n");
        int selectRet = select(maxfd+1,&rfds,NULL,NULL,NULL);//select()调用返回处于就绪状态并且已经包含在fd_set结构中的描述字总数;如果超时则返回0;
                                                                //否则的话,返回SOCKET_ERROR错误,应用程序可通过WSAGetLastError获取相应错误代码。
        if(selectRet <= 0){
            printf("select error [%d]\n", selectRet);       // >>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>
            Sleep(3000);
            // continue;
        }
        cout << "selectRet = " << selectRet << endl;
        cout << "maxfd = " << maxfd << endl;
        cout << "slisten = " << slisten << endl;
        for (int fd = 0; fd <= maxfd; fd++){
            if(FD_ISSET(fd,&rfds)){
                if(fd == slisten){
                    sClientfd = accept(slisten, (SOCKADDR *)&remoteAddr, &nAddrlen);  //如果监听的socket有消息,就接受client的socket
                    FD_SET(sClientfd, &rfds);
                    maxfd = sClientfd > maxfd ? sClientfd : maxfd;      // 更新maxfd
                    break;
                }else{
                    break;
                }
            }
            else{
                // cout << "fd = " << fd << endl;
            }
        }
        // if(FD_ISSET(slisten,&rfds)){
        //     sClientfd = accept(slisten, (SOCKADDR *)&remoteAddr, &nAddrlen);
        // }
        
        /*
        listen监听客户端来的链接,accept将客户端的信息绑定到一个socket上,也就是给客户端创建一个socket,通过返回值返回给我们客户端的socket。
        accept函数的第一个参数为服务器的socket描述字,第二个参数为指向struct sockaddr *的指针,用于返回客户端的协议地址,第三个参数为协议地址的长度。
        如果accpet成功,那么其返回值是由内核自动生成的一个全新的描述字,代表与返回客户的TCP连接。
            成功:
                返回值就是给客户端包好的socket; 与客户端通信就靠这个
                
            失败:
                返回INVALIE_SOCKET; 通过WSAGetLastError()得到错误码
        */
    // while (true)  
    // {    
        if(sClientfd <= 0 )  
        {  
            printf("accept error !");
            cout << WSAGetLastError() << endl;
            Sleep(2000);  
            continue;  
        }  
        printf("接受到一个连接:%s \r\n", inet_ntoa(remoteAddr.sin_addr));  
          
        //接收数据 
        // char revData[2] = {}; 
        // int data_size = sizeof(revData);
        string tmp = "";
        
        int ret = 0;
        do    //怕数据太长,一次recv copy不完,为了把client端一次发的的消息拼接、打印
        {
            char revData[DATA_SIZE] = {}; 
            revData[DATA_SIZE] = 0x00;   // 抵消数组后面的字符
            cout << "start " << revData <<endl;

            /*  // 下面的异步超时操作是为了把client端一次发的的消息拼接、打印(主要是一条消息最后一次数据大小 = DATA_SIZE的情况);;不过有bug,没想好怎么解决
                std::future<int> ret1 = std::async(std::launch::async, recv , sClientfd, revData, DATA_SIZE, 0); //异步调用
                std::future_status sts = ret1.wait_for(std::chrono::seconds(1)); //设置超时时间 s'
            
                if(sts == std::future_status::ready){
                    // ret = ret1.get();
                    printf("超时时间内收到了回复\n");
                }else if(sts == std::future_status::timeout){
                    printf("设定时间内无消息\n");
                    ret = 0;
                    break;      
            }
            */
            ret = recv(sClientfd, revData, DATA_SIZE, 0);  // 在主线程操作,如果recv大小正好是个整数包,一直阻塞在这,不会给该client返回消息;
                //这是上面do while想做的事情可能带来的影响
            cout << "server ret = " << ret << endl;
            if(ret <= 0){
                printf("receive msg error\n");
                // system("pause");
                Sleep(3000);
                break;
            }
            
            if(ret  < DATA_SIZE){
                revData[ret] = 0x00;
                cout << "last phase " << revData <<endl;
            }
            tmp = tmp + revData;
            cout << "tmp :[" << tmp << "] , leave"<<endl;

        } while (ret  == DATA_SIZE);
        cout <<'['<< tmp << ']'<<endl;

  
        //发送数据  
        const char * sendData = "你好,TCP客户端!\n";  
        send(sClientfd, sendData, strlen(sendData), 0);  
        closesocket(sClientfd);
        Sleep(1000);
    }  
      
    closesocket(slisten);  
    WSACleanup();  
    return 0;  
} 

特点

目前也没超时,不能很好的处理多个客户端的互相影响 

3、验证

编译执行,用到上面所有的client

手动输入的client会把服务端的进程阻塞住,导致另一个client的消息也进不来,手动输入消息后,会暂时解除当前阻塞,问题不小,小白太难了,后面再搞

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

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

相关文章

【C++】多线程(一):std::thread的使用

这篇文章应我朋友的邀请&#xff0c;写一篇文章介绍下C多线程。 编译环境准备 首先确定你的编译器支持std的thread&#xff0c;如果不支持&#xff0c;就会出现诸如“thread找不到”的问题。 以下假设你使用 gnu gcc 编译器&#xff0c;因为 MSVC 的我也不太熟悉。 linux …

【挑战业余一周拿证】二、在云中计算 - 第 1 节 - 模块2 简介

第 1 节 - 模块2 简介 无论你的企业是属于像医疗、保健、制造、保险等等行业 , 再或者 , 您的服务是向全世界的数百万用户提供视频、、图片或者文字服务,你也需要服务器来为您的业务和应用程序提供支持,服务器的作用是帮助您托管应用程序并提供满足您业务需求的计算能力. 当你使…

机器学习笔记 - 3D对象检测技术路线调研(未完)

一、3D对象检测简述 3D对象检测是计算机视觉中的一项任务&#xff0c;其目标是根据对象的形状、位置和方向在 3D 环境中识别和定位对象。它涉及检测物体的存在并实时确定它们在 3D 空间中的位置。这项任务对于自动驾驶汽车、机器人和增强现实等应用至关重要。 1、基本流程 给定…

leetcode_828_统计子串中的唯一字符

题意&#xff1a;所有子串中单个字符出现的次数和 问题转化&#xff1a;对于串中的每个字符&#xff0c;只包含其一次的所有子串的个数和 关于求只包含某位置字符一次的子串个数 class Solution { public:int uniqueLetterString(string s) {/* ...A...A...A...*/int n s.size…

DDD(领域驱动设计)一些基础概念

DDD、微服务和中台之间的关系 DDD、微服务和中台之间的关系。 中台本质是业务模型&#xff0c;微服务是业务模型的系统落地&#xff0c;DDD 是一种设计思想&#xff0c;可以同时指导中台业务建模和微服务设计&#xff0c;它们之间就是这样的一个铁三角关系。DDD 强调领域模型…

【深度学习笔记】05 线性回归

线性回归 线性回归基于几个简单的假设&#xff1a; 首先&#xff0c;假设自变量 x \mathbf{x} x和因变量 y y y之间的关系是线性的&#xff0c; 即 y y y可以表示为 x \mathbf{x} x中元素的加权和&#xff0c;这里通常允许包含观测值的一些噪声&#xff1b; 其次&#xff0c;我…

Educational Codeforces Round 158 [Rated for Div. 2]

A. Line Trip 还算比较简单的&#xff0c;不过本蒟蒻一开始以为是二分答案&#xff0c;二分写到一半突然想到油量直接取两个加油站之间的最大距离就好了。 最大距离能过&#xff0c;剩下必然都能过&#xff0c;要特判a[n]~x距离是两倍&#xff0c;因为x没有加油站&#xff0c…

Spring Cache框架,实现了基于注解的缓存功能。

个人简介&#xff1a;Java领域新星创作者&#xff1b;阿里云技术博主、星级博主、专家博主&#xff1b;正在Java学习的路上摸爬滚打&#xff0c;记录学习的过程~ 个人主页&#xff1a;.29.的博客 学习社区&#xff1a;进去逛一逛~ Spring Cache框架 简介Spring Cache 环境准备S…

如何提高API性能

下图给出了提高API性能的5个常用技巧 分页 当结果很大时&#xff0c;这是一种常见的优化。结果会流回客户端以提高服务响应能力。 异步日志记录 同步日志记录每次调用都会处理磁盘&#xff0c;并且会降低系统速度。异步日志记录首先将日志发送到无锁缓冲区并立即返回。日志将…

【企业微信连接问题】

1、个人可以创建企业微信的企业账号么&#xff1f; 答&#xff1a;可以的&#xff0c;只是没法认证。不过基础的功能还是有的。 注册步骤&#xff1a;企业微信注册步骤 2、集简云链接企业微信&#xff0c;在授权之后&#xff0c;找不到集简云怎么办&#xff1f; 答&#xff1a…

git分支命名规范

https://www.cnblogs.com/wq-9/p/16968098.html

一. BEV感知算法介绍

目录 前言1. BEV感知算法的概念2. BEV感知算法数据形式3. BEV开源数据集介绍3.1 KITTI数据集3.2 nuScenes数据集 4. BEV感知方法分类4.1 纯点云方案4.2 纯视觉方案4.3 多模态方案 5. BEV感知算法的优劣6. BEV感知算法的应用介绍7. 课程框架介绍与配置总结下载链接参考 前言 自动…

电力感知边缘计算网关产品设计方案-网关软件设计方案

网关采用网络协议和软件技术在通信网络中针对工业协议、互联网通用协议进行分析和记录,提升工业控制系统环境的安全防护能力。A类和B类网关采用容器技术的软件架构,采用C/S架构软件客户端提供应用软件平台,为管理员提供功能丰富的图形管理控制界面。 因A类和B类网关在产品定…

ubuntu20.04打不开github网址的有效解决方案

问题描述&#xff1a;重装的ubuntu系统&#xff0c;chrome浏览器刚开始还能打开github网址&#xff0c;然后突然就打不开了&#xff0c;换网络也不行。 解决方案步骤 1&#xff0c;查询你的电脑IP对应的github网址信息 2&#xff0c;修改host文件&#xff0c;添加第1步查询到…

Spring Cloud Gateway 的简单介绍和基本使用

前言 本文主要对Spring Cloud Gateway进行简单的概念介绍&#xff0c;并通过多模块编程的方式进行一个简单的实操。 文章目录 前言1 什么是网关&#xff08;概念&#xff09;2 微服务中的网关2.1 问题12.2 问题2 3 网关作用4 Spring Cloud Gateway组成5 Spring Cloud Gateway基…

局域网的网络ip不稳定问题

在局域网的多个设备&#xff0c;互相通信时好时坏&#xff0c;不稳定。 遭遇过的情况如下&#xff1a; 用两个开发板&#xff1a;972开发板1和2&#xff0c;网口同时互相ping&#xff0c;出现1ping 2通--此时2ping 1不通&#xff0c;过段时间&#xff0c;1ping2不通--但2ping又…

美化wordpress复制文章内容弹出版权提示框的源码代码

通过SweetAlert美化的提示框 将下面代码添加到当前主题模板函数functions.php文件最后即可&#xff1a; function zm_copyright_tips() { echo <link rel"stylesheet" type"text/css" rel"external nofollow" target"_blank" href…

【新手解答2】深入探索 C 语言:一些常见概念的解析

C语言的相关问题解答 写在最前面问题1变量名是否有可能与变量重名&#xff1f;变量名和变量的关系变量名与变量是否会"重名"举例说明结论 变量则是一个地址不变&#xff0c;值时刻在变的“具体数字”变量的地址和值变量名与数据类型具体化示例结论 问题2关于你给我的…

电商项目高级篇-03 商品上架

商品上架 1、商品上架1.1、设计&#xff1a;宽表设计 1、商品上架 上架的商品才可以在网站展示。 上架的商品需要可以被检索。 1.1、设计&#xff1a;宽表设计 优点&#xff1a;方便检索 缺点&#xff1a;数据冗余 商品数据模型设计&#xff1a; PUT product {"mappi…

tidyverse数据特征学习

目录 特征缩放 1&#xff0c;标准化-scale 2&#xff0c;归一化-rescale 3&#xff0c;行规范化 4&#xff0c;数据平滑 特征变换 1. 非线性特征 2. 正态性变换 3. 连续变量离散 特征降维 特征缩放 不同数值型特征的数据量纲可能相差多个数量级&#xff0c;这对很多…