最新资讯

  • Linux:Http协议

Linux:Http协议

2025-04-29 06:00:22 0 阅读

一、认识URL

平时我们俗称的 "网址" 其实就是说的 URL-统一资源定位符

(1)我们在网络编程中,知道了可以通过ip地址标识唯一的一台主机,可事实是使用ip地址并不符合我们普通人的使用习惯,因此我们需要在内部建立域名到ip的映射,这样用户在使用的时候只需要通过域名,然后由浏览器帮我们去解析成ip地址,所以其实浏览器的内部是内置了域名解析的方法的!!(比如www.baidu.com)

(2) 我们在网络编程中,还知道要通过端口号来标识这台主机上的唯一一个服务,而该进程自带着网络协议http的解析方法,可是我们普通人使用的时候是没有端口号这个概念的! 那么浏览器怎么知道你这个端口号呢?? 因为浏览器会默认使用http协议,所以他必须知道绑定443号端口,所以默认会在请求里给我们添加端口号

(3) 可是,我找到了这台主机和http服务,但是我想访问的是什么呢???比如我们平时在搜图片的时候,这个图片其实是被存储在该主机上的,而我们知道Linux一切皆文件,所以每个资源在自己的单机上都有自己的所属路径。因此我们还需要有一个带层次的文件路径。来标识我们具体想访问该主机上哪个地方的资源(一般来说可能是从web根目录开始 也有可能是相对路径)!/是文件路径分隔符

(4)当我们找到了这个资源的路径,那么这个路径可能有很多很多格式各种的资源,那么我们继续需要锁定其中的一个,就需要有查询字符串,一般在查询标示符?的后面,表明该路径下的唯一资源,而要锁定唯一资源,就需要传入一些指定的参数,不同参数之间用&去分隔,让url支持多参数的提交!

(5) 而#后面的是片段标示符,也就是说可能我们想访问的是该资源的某一个位置,可以用这个去进行标识! 

 ——>总结

域名和路径                             标识唯一一台主机上的一个唯一文件资源 

端口号和协议                          可以唯一确认用什么样的服务来获取或者上传资源

后面的资源路径(字符串)    可以标识唯一能找到的一张网页资源

凡是能通过url访问的资源,我们都叫做万维网!!

二、urlencode和urldecode

像 / ? : 等这样的字符, 已经被url当做特殊意义理解了. 因此这些字符不能随意出现.

比如, 某个参数中需要带有这些特殊字符, 就必须先对特殊字符进行转义.

转义的规则如下:

      浏览器会将需要转码的字符转为16进制,然后从右到左,取4位(不足4位直接处理),每2位做一位,前面加上%,编码成%XY格式

"+" 被转义成了 "%2B"   

urldecode就是urlencode的逆过程; 

 三、Http协议格式

 最常见的两个网络行为:

(1)把别人的东西拿下来 GET(2)把自己的东西传上去 POST

 我们会发现一共包含四个部分:请求行、请求报头、空行、请求正文(这个不一定得有)

HTTP请求:

首行: [方法] + [url] + [版本]

Header: 请求的属性, 冒号分割的键值对;每组属性之间使用 分隔;遇到空行表示Header部分结束 

Body: 空行后面的内容都是Body. Body允许为空字符串. 如果Body存在, 则在Header中会有一个

Content-Length属性来标识Body的长度;

 HTTP响应:

首行: [版本号] + [状态码] + [状态码解释]

Header: 请求的属性, 冒号分割的键值对;每组属性之间使用 分隔;遇到空行表示Header部分结束

Body: 空行后面的内容都是Body. Body允许为空字符串. 如果Body存在, 则在Header中会有一个

Content-Length属性来标识Body的长度; 如果服务器返回了一个html页面, 那么html页面内容就是在

body中.

 问题1:你怎么确保正文部分能够一字不落地读完呢??

——>报头属性里面有 标注正文部分的长度的属性,所以他只要根据换行规则去读取整个报文,然后一直读到空行就说明报头读完了,然后再根据length字段 直接向后读相应长度的字节 就可以把正文部分也给读完了

 问题2:为什么信息需要有版本号呢??

——>版本号是便于我们双方标识对应版本号的功能,因为可能有的用户没有更新客户端,所以服务端是有可能会接受到各种不同版本的客户端请求的,而较新的版本可能会存在一些旧版本没有的功能!!(比如有的版本可以发消息,而有的版本还可以发朋友圈)

问题3:状态码的含义

——>一般来讲不同的状态码对应不同的状况 比如常见的200是正常 404就是出错

问题4:常见的两个网络测试工具

——>fiddler和postman

fildder原理:先一步抓包拿到请求和响应 (默认是加密的)

postman:可以通过他发送具体请求

ubunt 20.04 最新版本 

 四、Http版本1之Hello World 

 发送请求用到的方法:

响应用到的方法:

     平时浏览器看到的内容,底层是通过浏览器给我们返回的按协议解析的字符串信息,我们只会看到他的正文部分 

问题1:我们的网页信息,难道每次都要静态编码写到服务器里面吗?? 

-——> 关于网页,我们一般需要用到html !这是属于前端的知识了!!

header包含的是头部的报文信息

 body包含的是主体信息,然后整体再用html包起来!!

这样未来我们想要修改网页的内容,只需要在html文件里面做,从而不需要修改源代码 

 问题2:如何理解web根目录呢??

——> 按理来说,我们如果访问的时不带路径(只用IP加端口号),那么默认其实就是web根目录,可是web根目录难道就是Linux的根目录吗?? 显然不一定,因为我们在底层的Linux代码中去设置某个路径来充当Web根目录!

   并且我们要知道,网页文件肯定不止一个,所以必然需要有目录结构,所以未来我们需要将我们写的网页、图片、视频、首页等信息以树状的结构放在这个目录下让其他人去访问,所以这个就叫做Web根目录!未来设置的任何一个网页都可以放在web根目录下! 

     所以浏览器访问服务器的本质实际上是把服务器上指定路径下的文件资源获取道浏览器中,然后由 浏览器进行解释最后呈现的就是网页资源!! 

 问题3:我们的一些常量信息难道一定也是要静态编码进去吗??

——>既然网页信息都可以写到html文件里了!!那么我们的一些重要的常量信息(比如说我们设置的Web根目录)也可以写到配置文件里,这样未来我们就可以通过修改配置文件来修改这些常量,而源代码里只需要提供读取配置文件的方法即可,从而就不需要大幅度变动代码,并且可以在配置文件里一次性修改很多信息更加方便!!

 问题4:用户如果什么路径也不带,那么默认访问的就是web根目录,可是这不就相当于把整个网站资源都呈现给你了吗??

——>这显然是不合理的,就像我们的百度一样,百度首页其实也是一个Web根目录下的一个文件,因此我们需要在底层自己做一个网站的首页,然后当我们发现用户填的路径是/的时候,我们将他转移到我们的网站上并呈现给对方!!

问题5:C++其实提供了一些方法能够帮助我们快速分割字符串!!

 

所以协议的本质其实就是对字符串做一个规范的解析!

 五、Http方法

但是一般来说,最常用的就是GET和POST 

 问题:我们日常使用某些网站(http/https),究竟是如何把数据提交给服务器的?

----->数据都是通过表单提交的!!!

当我们需要提交参数给服务器

1、如果我们使用的是GET方法,我们提交的参数是通过url提交的,但是(1)参数会受限 (2)不够私密

2、而如果使用POST方法,会采用请求的正文提交参数,相对来说更私密一点!

         其实提交参数的本质意义就是,fork子进程让他通过进程替换去执行这个程序,然后我们再通过进程间通信的方法将参数传递给他,然后让他完成相应的功能!!
 

六、Http状态码

  最常见的状态码, 比如 200(OK), 404(Not Found), 403(Forbidden), 302(Redirect, 重定向), 504(Bad Gateway)

404表示资源不存在,比如客户端请求的资源,服务器打开失败所以产生错误,是客户端的错误(所以一般来说我们除了要写一个网站首页的文件,还需要写一个如果请求失败的一个返回404页面的网页)

403是禁止访问,一般就是因为你没有授权然后去访问了服务端不让你访问的信息 

服务器错误,一般来说就是比如连接不上,或者是服务的线程创建失败

3XX一般是由于我要访问的服务器可能处于某种原因无法为我提供服务,所以他会传一个地址告诉客户端你想要的资源应该去这里查找(通过参数Location),这就是重定向!  

更多的信息可以去百度搜状态码列表

七、 HTTP常见Header

1、Content-Type: 数据类型(text/html等)

        读html的时候他是文本文件,那么就可以按照字符串读,但是如果我们是图片资源(png),那么就得按照二进制文件的方式来读!!因此在读取html之前需要确认一下文件格式,而不同的格式对应的content-type可以到网上搜索对照表。所以读网页之前必须确定一下后缀,然后不同的后缀就根据content-type对照表去找

2、Content-Length: Body的长度

3、Host: 客户端告知服务器, 所请求的资源是在哪个主机的哪个端口上;

4、User-Agent: 声明用户的操作系统和浏览器版本信息;

(1)他可以用来反爬,比方说他可以看看你这个客户端是不是一个合法的客户端,如果不合法的话就不给你做http响应。

(2)看看你是什么样的平台,然后可以根据你的平台给你推送信息比如APP

5、referer: 当前页面是从哪个页面跳转过来的;

6、location: 搭配3xx状态码使用, 告诉客户端接下来要去哪里访问;

重定向分为永久重定向和临时重定向

 讲个故事:

       比如说你的学校东门有一家火锅店非常有名,但是他需要长时间的装修,为了继续生意,他将生意开到了学校的西门,然后在东门这里贴了一张纸条“正在装修,请到西门用餐”,这个时候你和你的同学到东门时看到了这个纸条,于是就到西门去吃了了,过了一礼拜你们还打算去的时候,你知道东门那边只是暂时关闭,所以你还是会先去东门看看,如果还没开的话才会去西门,这个其实就是临时重定向,因为我知道这个店只是暂时开到西门,随时可能会回来!

      而后来东门开了的时候,为了能够让之前去西门吃饭的同学知道东门开启了,他会在之前西门的地方贴上“东门已经装修完成,以后请都前往东门用餐” 这个时候你们就知道这个店会一直在东门,这其实就是永久重定向!!

      所以有的网站时间比较老,做更新的时候需要把域名和网址换了,可是很多老用户并不知道,所以就会给老网站部署永久性定向服务,让用户直接跳转到新网站!

7、connection:是否支持长连接(是基于服务费和客户端的版本去协商的)

短连接其实就是一次请求响应一个资源,然后就关闭连接

长连接其实就是一次请求连接上之后可以一直服务直到服务结束再关闭连接

为什么要有长连接呢??

---——>首先我们要知道,一个巨大的网页上的元素是很多的,而每一个元素其实就是一个资源,所以我们在发出http请求申请到网页资源的时候,同时也需要把网页上附带的资源(比如图片、音频) 都申请了,所以如果用短连接的话显然效率是不够高的!!

         长连接和短连接并没有绝对的优劣,只不过应用场景不一样!!

         但是要注意的是,具体采用长连接还是短连接,是要基于双方的HTTP版本协商的,其中HTTP1.0其实就是短连接  而HTTP1.1其实就是长连接   要支持长连接的话必须要求双方的版本都是1.1   这样Connection就会呈现keep-alive表示支持长连接 

8、Cookie: 用于在客户端存储少量信息. 通常用于实现会话(session)的功能;

       比方说我们的b站,要看一个视频的话必须要先登录,你如果没登录他会提示你,然后你输入了你的用户名密码之后,你成功登录了,可是当你关闭浏览器然后再打开一次的时候,却不需要登录了,可是浏览器究竟是如何知道你此刻属于登录状态呢? 其实这就是http的会话保持功能!

     首先我们要知道,HTTP默认是无状态的,而他之所以能够知道你你处在登录状态,是因为你之前登录的时候,在浏览器里形成了一个cookie文件,这个文件里存储着你在这个网站的认证信息,而当你打开这个网站时,浏览器向对应服务端发送请求的时候会将你的认证信息(就是用户名和密码)放在cookie参数里面带过去直接认证(认证其实就是拿着你的用户名和密码去他后端的数据库做搜索,所以你想使用的前提是必须得注册,才能在他的数据库里留存数据),认证通过之后会直接将你从原先的登录页面重定向到目标页面,这样你就不需要再次登录了!!当然这个保存一般是有时间限制的!

     而一般cookie文件有两种类型,一种是内存级,一种是文件级,不同的浏览器可能不一样,如果是内存级的话,一般来说就是你的浏览器一旦关闭那么这个信息就不在了,但是如果是文件级的话,可能会有具体的过期时间,在你没过期之前你只要还是用该浏览器打开这个页面,那么cookie的存在就可以帮助你免去登录的过程!

        可是cookie的存在虽然方便了用户的使用体验,但他同样也伴生了安全问题,比如说别人如果把你的cookie文件窃取了,等于说看到了你的用户名和密码,那么问题就很严重了!!(qq盗号原理)

        所以我们会面临两个问题(1)cookie被盗取 (2)个人信息泄露

       我们要知道我们小白用户的防范能力基本为0,他们的电脑在黑客的眼里其实就相当于裸奔,比如你不小心点开了一个病毒,你或许可以通过杀毒软件去杀毒,但是也很有可能在你清理这个病毒之前你的信息就已经被窃取了!! 所以显然不能让客户端来维护这个安全问题,必须由更专业的服务端来维护!

      所以引入了session技术,我们输入用户名密码的时候,他会将这个用户名密码存在服务端,然后生成一个session id(数字指纹)返回给客户端存在cookie文件里,然后服务端会将session id管理起来(redis),这样认证的时候就会用cookie文件里面存储的session id去服务端做对比,只要通过了就可以直接登录了!!

    可是你可能会问,如果我的session id泄露了,那别人不是也可以拿我的session id以我的身份去访问我的信息吗???

----->确实session id被盗取,是客户端的行为,这个是很难避免的,可能你不小心点开了一个病毒就被窃取了,并且我们作为小白本身是无感的,但是跟之前最大的区别就是session id是由服务端统一管理的,也就意味着他具备回收、停用、甄别异常等能力(比如说他发现你的ip地址突然发生了很大的变动,察觉异常就会暂时给你停止,等你发现问题了再次申诉的时候,他就会把你之前的session id清理掉然后重新分配) 这其实就是达到了控制客户端的目的!! 

八、代码

HttpServer.hpp

#pragma once
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include "Socket.hpp"
#include "Log.hpp"

const std::string wwwroot="./wwwroot"; // web 根目录
const std::string sep = "
";
const std::string homepage = "index.html";

static const int defaultport = 8082;

class HttpServer;

class ThreadData
{
public:
    ThreadData(int fd, HttpServer *s) : sockfd(fd), svr(s)
    {
    }

public:
    int sockfd;
    HttpServer *svr;
};

class HttpRequest
{
public:
    void Deserialize(std::string req)
    {
        while(true)
        {
            std::size_t pos = req.find(sep);
            if(pos == std::string::npos) break;
            std::string temp = req.substr(0, pos);
            if(temp.empty()) break;
            req_header.push_back(temp);
            req.erase(0, pos+sep.size());
        }
        text = req;
    }
    // .png:image/png
    void Parse()
    {
        std::stringstream ss(req_header[0]);
        ss >> method >> url >> http_version;
        file_path = wwwroot; // ./wwwroot
        if(url == "/" || url == "/index.html") {
            file_path += "/";
            file_path += homepage; // ./wwwroot/index.html
        }
        else file_path += url; // /a/b/c/d.html->./wwwroot/a/b/c/d.html

        auto pos = file_path.rfind(".");
        if(pos == std::string::npos) suffix = ".html";
        else suffix = file_path.substr(pos);
    }
    void DebugPrint()
    {
        for(auto &line : req_header)
        {
            std::cout << "--------------------------------" << std::endl;
            std::cout << line << "

";
        }

        std::cout << "method: " << method << std::endl;
        std::cout << "url: " << url << std::endl;
        std::cout << "http_version: " << http_version << std::endl;
        std::cout << "file_path: " << file_path << std::endl;
        std::cout << text << std::endl;
    }
public:
    std::vector req_header;
    std::string text;

    // 解析之后的结果
    std::string method;
    std::string url;
    std::string http_version;
    std::string file_path; // ./wwwroot/a/b/c.html 2.png

    std::string suffix;
};

class HttpServer
{
public:
    HttpServer(uint16_t port = defaultport) : _port(port)
    {
       content_type.insert({".html", "text/html"});
        content_type.insert({".png", "image/png"});
        content_type.insert({".jpg", "image/jpeg"});
    }
    bool Start()
    {
        _listensock.Socket();
        _listensock.Bind(_port);
        _listensock.Listen();
        for (;;)
        {
            std::string clientip;
            uint16_t clientport;
            int sockfd = _listensock.Accept(&clientip, &clientport);
            if (sockfd < 0)
                continue;
            lg(Info, "get a new connect, sockfd: %d", sockfd);
            pthread_t tid;
            ThreadData *td = new ThreadData(sockfd, this);
            pthread_create(&tid, nullptr, ThreadRun, td);
        }
    }
    static std::string ReadHtmlContent(const std::string &htmlpath)
    {
        // 坑
        std::ifstream in(htmlpath, std::ios::binary);
        if(!in.is_open()) return "";

        in.seekg(0, std::ios_base::end);
        auto len = in.tellg();
        in.seekg(0, std::ios_base::beg);

        std::string content;
        content.resize(len);

        in.read((char*)content.c_str(), content.size());
        //std::string content;
        //std::string line;
        //while(std::getline(in, line))
        //{
        //    content += line;
        //}

        in.close();

        return content;
    }
 
      std::string SuffixToDesc(const std::string &suffix)
    {
        auto iter = content_type.find(suffix);
        if(iter == content_type.end()) return content_type[".html"];
        else return content_type[suffix];
    }

    void HandlerHttp(int sockfd)
    {
        char buffer[10240];
        ssize_t n = recv(sockfd, buffer, sizeof(buffer) - 1, 0); // bug
        if (n > 0)
        {
            buffer[n] = 0;
            std::cout << buffer << std::endl; // 假设我们读取到的就是一个完整的,独立的http 请求
            HttpRequest req;
            req.Deserialize(buffer);
            req.Parse();
            req.DebugPrint();

            //std::string path = wwwroot;
            //path += url; // wwwroot/a/a/b/index.html

             // 返回响应的过程
            std::string text;
            bool ok = true;
            text = ReadHtmlContent(req.file_path); // 失败?
            if(text.empty())
            {
                ok = false;
                std::string err_html = wwwroot;
                err_html += "/";
                err_html += "err.html";
                text = ReadHtmlContent(err_html);
            }

            std::string response_line;
            if(ok)
                response_line = "HTTP/1.0 200 OK
";
            else
                response_line = "HTTP/1.0 404 Not Found
";
            
            //response_line = "HTTP/1.0 302 Found
";
            std::string response_header = "Content-Length: ";
            response_header += std::to_string(text.size()); // Content-Length: 11
            response_header += "
";
            response_header += "Content-Type: ";
            response_header += SuffixToDesc(req.suffix);
            response_header += "
";
            response_header += "Set-Cookie: name=haha&&passwd=12345";
            response_header += "
";

            //response_header += "Location: https://www.qq.com
";
            std::string blank_line = "
"; // 


            std::string response = response_line;
            response += response_header;
            response += blank_line;
            response += text;

            send(sockfd, response.c_str(), response.size(), 0);
        }
        close(sockfd);
    }
    static void *ThreadRun(void *args)
    {
        pthread_detach(pthread_self());
        ThreadData *td = static_cast(args);
        td->svr->HandlerHttp(td->sockfd);
        delete td;
        return nullptr;
    }
    ~HttpServer()
    {
    }

private:
    Sock _listensock;
    uint16_t _port;
    std::unordered_map content_type;
};

HttpServer.cpp

#include "HttpServer.hpp"
#include 
#include 
#include 
#include "Log.hpp"


using namespace std;

int main(int argc, char *argv[])
{
    if(argc != 2)
    {
        exit(1);
    }
    
    uint16_t port = std::stoi(argv[1]);
    // HttpServer *svr = new HttpServer();
    // std::unique svr(new HttpServer());
    std::unique_ptr svr(new HttpServer(port));
    svr->Start();
    return 0;
}

Socket.hpp

#pragma once

#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include "Log.hpp"

enum
{
    SocketErr = 2,
    BindErr,
    ListenErr,
};

// TODO
const int backlog = 10;

class Sock
{
public:
    Sock()
    {
    }
    ~Sock()
    {
    }

public:
    void Socket()
    {
        sockfd_ = socket(AF_INET, SOCK_STREAM, 0);
        if (sockfd_ < 0)
        {
            lg(Fatal, "socker error, %s: %d", strerror(errno), errno);
            exit(SocketErr);
        }
        int opt = 1;
        setsockopt(sockfd_, SOL_SOCKET, SO_REUSEADDR, &opt, sizeof(opt));
    }
    void Bind(uint16_t port)
    {
        struct sockaddr_in local;
        memset(&local, 0, sizeof(local));
        local.sin_family = AF_INET;
        local.sin_port = htons(port);
        local.sin_addr.s_addr = INADDR_ANY;

        if (bind(sockfd_, (struct sockaddr *)&local, sizeof(local)) < 0)
        {
            lg(Fatal, "bind error, %s: %d", strerror(errno), errno);
            exit(BindErr);
        }
    }
    void Listen()
    {
        if (listen(sockfd_, backlog) < 0)
        {
            lg(Fatal, "listen error, %s: %d", strerror(errno), errno);
            exit(ListenErr);
        }
    }
    int Accept(std::string *clientip, uint16_t *clientport)
    {
        struct sockaddr_in peer;
        socklen_t len = sizeof(peer);
        int newfd = accept(sockfd_, (struct sockaddr*)&peer, &len);
        if(newfd < 0)
        {
            lg(Warning, "accept error, %s: %d", strerror(errno), errno);
            return -1;
        }
        char ipstr[64];
        inet_ntop(AF_INET, &peer.sin_addr, ipstr, sizeof(ipstr));
        *clientip = ipstr;
        *clientport = ntohs(peer.sin_port);

        return newfd;
    }
    bool Connect(const std::string &ip, const uint16_t &port)
    {
        struct sockaddr_in peer;
        memset(&peer, 0, sizeof(peer));
        peer.sin_family = AF_INET;
        peer.sin_port = htons(port);
        inet_pton(AF_INET, ip.c_str(), &(peer.sin_addr));

        int n = connect(sockfd_, (struct sockaddr*)&peer, sizeof(peer));
        if(n == -1) 
        {
            std::cerr << "connect to " << ip << ":" << port << " error" << std::endl;
            return false;
        }
        return true;
    }
    void Close()
    {
        close(sockfd_);
    }
    int Fd()
    {
        return sockfd_;
    }

private:
    int sockfd_;
};

log.hpp

#pragma once

#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 

#define SIZE 1024

#define Info 0
#define Debug 1
#define Warning 2
#define Error 3
#define Fatal 4

#define Screen 1
#define Onefile 2
#define Classfile 3

#define LogFile "log.txt"

class Log
{
public:
    Log()
    {
        printMethod = Screen;
        path = "./log/";
    }
    void Enable(int method)
    {
        printMethod = method;
    }
    std::string levelToString(int level)
    {
        switch (level)
        {
        case Info:
            return "Info";
        case Debug:
            return "Debug";
        case Warning:
            return "Warning";
        case Error:
            return "Error";
        case Fatal:
            return "Fatal";
        default:
            return "None";
        }
    }

    // void logmessage(int level, const char *format, ...)
    // {
    //     time_t t = time(nullptr);
    //     struct tm *ctime = localtime(&t);
    //     char leftbuffer[SIZE];
    //     snprintf(leftbuffer, sizeof(leftbuffer), "[%s][%d-%d-%d %d:%d:%d]", levelToString(level).c_str(),
    //              ctime->tm_year + 1900, ctime->tm_mon + 1, ctime->tm_mday,
    //              ctime->tm_hour, ctime->tm_min, ctime->tm_sec);

    //     // va_list s;
    //     // va_start(s, format);
    //     char rightbuffer[SIZE];
    //     vsnprintf(rightbuffer, sizeof(rightbuffer), format, s);
    //     // va_end(s);

    //     // 格式:默认部分+自定义部分
    //     char logtxt[SIZE * 2];
    //     snprintf(logtxt, sizeof(logtxt), "%s %s
", leftbuffer, rightbuffer);

    //     // printf("%s", logtxt); // 暂时打印
    //     printLog(level, logtxt);
    // }
    void printLog(int level, const std::string &logtxt)
    {
        switch (printMethod)
        {
        case Screen:
            std::cout << logtxt << std::endl;
            break;
        case Onefile:
            printOneFile(LogFile, logtxt);
            break;
        case Classfile:
            printClassFile(level, logtxt);
            break;
        default:
            break;
        }
    }
    void printOneFile(const std::string &logname, const std::string &logtxt)
    {
        std::string _logname = path + logname;
        int fd = open(_logname.c_str(), O_WRONLY | O_CREAT | O_APPEND, 0666); // "log.txt"
        if (fd < 0)
            return;
        write(fd, logtxt.c_str(), logtxt.size());
        close(fd);
    }
    void printClassFile(int level, const std::string &logtxt)
    {
        std::string filename = LogFile;
        filename += ".";
        filename += levelToString(level); // "log.txt.Debug/Warning/Fatal"
        printOneFile(filename, logtxt);
    }

    ~Log()
    {
    }
    void operator()(int level, const char *format, ...)
    {
        time_t t = time(nullptr);
        struct tm *ctime = localtime(&t);
        char leftbuffer[SIZE];
        snprintf(leftbuffer, sizeof(leftbuffer), "[%s][%d-%d-%d %d:%d:%d]", levelToString(level).c_str(),
                 ctime->tm_year + 1900, ctime->tm_mon + 1, ctime->tm_mday,
                 ctime->tm_hour, ctime->tm_min, ctime->tm_sec);

        va_list s;
        va_start(s, format);
        char rightbuffer[SIZE];
        vsnprintf(rightbuffer, sizeof(rightbuffer), format, s);
        va_end(s);

        // 格式:默认部分+自定义部分
        char logtxt[SIZE * 2];
        snprintf(logtxt, sizeof(logtxt), "%s %s", leftbuffer, rightbuffer);

        // printf("%s", logtxt); // 暂时打印
        printLog(level, logtxt);
    }
private:
    int printMethod;
    std::string path;
};

Log lg;

// int sum(int n, ...)
// {
//     va_list s; // char*
//     va_start(s, n);

//     int sum = 0;
//     while(n)
//     {
//         sum += va_arg(s, int); // printf("hello %d, hello %s, hello %c, hello %d,", 1, "hello", 'c', 123);
//         n--;
//     }

//     va_end(s); //s = NULL
//     return sum;
// }

 Makefile

HttpServer:HttpServer.cc
	g++ -o $@ $^ -std=c++11 -lpthread
.PHONY:clean
clean:
	rm -f HttpServer

        Web根目录 

问题1:公司里面是怎么去搭建服务器的???

-——>一般来说公司搭建服务器用的是现成的方案,就算要搭建新的框架,也不会从0开始,大部分情况下并不需要太高深的算法,而平时公司考察笔试单纯是你为了筛选出代码能力很强的人,但是具体能不能筛选出他们想要的也难说,但是总之不会太差。

问题2:为什么浏览器的种类有这么多??

——>08-12年最大的公司是百度,因为当时还没有什么app,所以大家要上网基本上都需要用百度搜索,他的流量很大,但是你想使用百度搜索的前提是你必须要有一个浏览器,而浏览器就意味着流量的入口,所以很多公司都有动力去做这个,比如windows除了内置有自己的office办公软件,还会内置一款自带的IE浏览器。但是国内大部分浏览器使用的都是clone内核,而微软和谷歌有自己的浏览器!

本文地址:https://www.vps345.com/5318.html

搜索文章

Tags

PV计算 带宽计算 流量带宽 服务器带宽 上行带宽 上行速率 什么是上行带宽? CC攻击 攻击怎么办 流量攻击 DDOS攻击 服务器被攻击怎么办 源IP 服务器 linux 运维 游戏 云计算 javascript 前端 chrome edge 进程 操作系统 进程控制 Ubuntu python MCP 阿里云 网络 网络安全 网络协议 llama 算法 opencv 自然语言处理 神经网络 语言模型 ssh ubuntu deepseek Ollama 模型联网 API CherryStudio 科技 ai java 人工智能 个人开发 fastapi mcp mcp-proxy mcp-inspector fastapi-mcp agent sse harmonyos 华为 开发语言 typescript 计算机网络 数据库 centos oracle 关系型 安全 分布式 Flask FastAPI Waitress Gunicorn uWSGI Uvicorn RTSP xop RTP RTSPServer 推流 视频 tcp/ip 面试 性能优化 jdk intellij-idea 架构 GaN HEMT 氮化镓 单粒子烧毁 辐射损伤 辐照效应 php android 宝塔面板访问不了 宝塔面板网站访问不了 宝塔面板怎么配置网站能访问 宝塔面板配置ip访问 宝塔面板配置域名访问教程 宝塔面板配置教程 Dell R750XS vue.js audio vue音乐播放器 vue播放音频文件 Audio音频播放器自定义样式 播放暂停进度条音量调节快进快退 自定义audio覆盖默认样式 华为云 物联网 嵌入式硬件 单片机 c++ 温湿度数据上传到服务器 Arduino HTTP udp unity YOLO efficientVIT YOLOv8替换主干网络 TOLOv8 rust http 学习 WSL2 uni-app macos adb 编辑器 鸿蒙 鸿蒙系统 前端框架 ESP32 flask spring boot AI编程 AIGC 深度学习 目标检测 计算机视觉 vim Windsurf pycharm ide pytorch filezilla 无法连接服务器 连接被服务器拒绝 vsftpd 331/530 ssl golang 后端 机器学习 numpy HCIE 数通 node.js json html5 firefox vue3 HTML audio 控件组件 vue3 audio音乐播放器 Audio标签自定义样式默认 vue3播放音频文件音效音乐 自定义audio播放器样式 播放暂停调整声音大小下载文件 kubernetes 容器 学习方法 经验分享 程序人生 ollama 大模型 mac asm c# cpu 内存 实时 使用 cuda cudnn anaconda flutter Hyper-V WinRM TrustedHosts camera Arduino 电子信息 rabbitmq iot 自动化 蓝耘科技 元生代平台工作流 ComfyUI websocket windows 嵌入式 linux驱动开发 arm开发 react.js 前端面试题 持续部署 jenkins gitee redis mybatis github express oceanbase rc.local 开机自启 systemd 麒麟 ping++ 运维开发 深度优先 图论 并集查找 换根法 树上倍增 ddos llm gpu算力 chatgpt llama3 Chatglm 开源大模型 docker transformer zotero WebDAV 同步失败 代理模式 ffmpeg 音视频 jupyter conda pip asp.net大文件上传 asp.net大文件上传源码 ASP.NET断点续传 asp.net上传文件夹 asp.net上传大文件 .net core断点续传 .net mvc断点续传 部署 服务器配置 list 数据结构 缓存 银河麒麟服务器操作系统 系统激活 电脑 博客 windwos防火墙 defender防火墙 win防火墙白名单 防火墙白名单效果 防火墙只允许指定应用上网 防火墙允许指定上网其它禁止 系统架构 工业4.0 高效远程协作 TrustViewer体验 跨设备操作便利 智能远程控制 并查集 leetcode sql KingBase 漏洞 负载均衡 AI Agent 安全威胁分析 云原生 fpga开发 开源 智能手机 NAS Termux Samba Linux 计算机外设 软件需求 tomcat postman mock mock server 模拟服务器 mock服务器 Postman内置变量 Postman随机数据 产品经理 agi microsoft LDAP 交换机 telnet 远程登录 信息与通信 实时音视频 vscode IIS服务器 IIS性能 日志监控 腾讯云 笔记 maven intellij idea 安装教程 GPU环境配置 Ubuntu22 CUDA PyTorch Anaconda安装 MQTT mosquitto 消息队列 r语言 数据挖掘 数据可视化 数据分析 .netcore word图片自动上传 word一键转存 复制word图片 复制word图文 复制word公式 粘贴word图文 粘贴word公式 git ansible nginx dubbo 微信 微信分享 Image wxopensdk kamailio sip VoIP tcpdump 课程设计 mysql sqlserver rust腐蚀 框架搭建 .net c语言 豆瓣 追剧助手 迅雷 nas 远程工作 unity3d apache 孤岛惊魂4 低代码 aws googlecloud HarmonyOS Next mcu gitlab okhttp CORS 跨域 命名管道 客户端与服务端通信 webrtc 向日葵 kafka jmeter 软件测试 大模型微调 智能路由器 外网访问 内网穿透 端口映射 多线程服务器 Linux网络编程 监控 自动化运维 gateway Clion Nova ResharperC++引擎 Centos7 远程开发 大数据 爬虫 网络用户购物行为分析可视化平台 大数据毕业设计 1024程序员节 大数据平台 debian 测试工具 web安全 pillow live555 rtsp rtp Trae IDE AI 原生集成开发环境 Trae AI https WSL win11 无法解析服务器的名称或地址 spring 小程序 微信小程序域名配置 微信小程序服务器域名 微信小程序合法域名 小程序配置业务域名 微信小程序需要域名吗 微信小程序添加域名 ESXi Kali Linux 黑客 渗透测试 信息收集 C 环境变量 进程地址空间 DeepSeek YOLOv12 django web3.py ecmascript KVM Qwen2.5-coder 离线部署 n8n 工作流 workflow 代码调试 ipdb adobe threejs 3D 微服务 nvidia k8s 创意 社区 Dify DeepSeek-R1 API接口 源码剖析 rtsp实现步骤 流媒体开发 VMware安装Ubuntu Ubuntu安装k8s wireshark NPS 雨云服务器 雨云 springsecurity6 oauth2 授权服务器 token sas Cookie 僵尸进程 mongodb 远程控制 远程看看 远程协助 rsyslog ue4 着色器 ue5 虚幻 C语言 ipython springcloud DigitalOcean GPU服务器购买 GPU服务器哪里有 GPU服务器 bootstrap html YOLOv8 NPU Atlas800 A300I pro asi_bench 进程信号 Ubuntu Server Ubuntu 22.04.5 统信UOS bonding 链路聚合 飞牛NAS 飞牛OS MacBook Pro mount挂载磁盘 wrong fs type LVM挂载磁盘 Centos7.9 svn ollama下载加速 qt stm32项目 stm32 firewalld RoboVLM 通用机器人策略 VLA设计哲学 vlm fot robot 视觉语言动作模型 具身智能 devops springboot prometheus PVE netty SSH 服务 SSH Server OpenSSH Server dify ragflow dell服务器 go c IIS .net core Hosting Bundle .NET Framework vs2022 XFS xfs文件系统损坏 I_O error es jvm iftop 网络流量监控 shell 磁盘监控 make命令 makefile文件 virtualenv 相差8小时 UTC 时间 Java minicom 串口调试工具 生物信息学 openEuler 驱动开发 gitea file server http server web server 安卓 集成学习 集成测试 状态管理的 UDP 服务器 Arduino RTOS yum DNS Invalid Host allowedHosts vue 远程连接 rdp 实验 文件系统 路径解析 王者荣耀 硬件架构 计算机 Wi-Fi web Socket minio elasticsearch ruoyi 指令 ecm bpm mq rocketmq ip命令 新增网卡 新增IP 启动网卡 云电竞 云电脑 todesk MacOS录屏软件 mamba Vmamba 职场和发展 硬件 设备 GPU PCI-Express docker命令大全 kylin 5G 3GPP 卫星通信 UOS 统信操作系统 SysBench 基准测试 selenium 命令 游戏服务器 Minecraft 服务器繁忙 Erlang OTP gen_server 热代码交换 事务语义 RAGFLOW RAG 检索增强生成 文档解析 大模型垂直应用 MNN Qwen ip C++ ui 音乐服务器 Navidrome 音流 大语言模型 系统安全 媒体 NFS 远程桌面 监控k8s 监控kubernetes gaussdb playbook MQTT协议 消息服务器 代码 excel 在线预览 xlsx xls文件 在浏览器直接打开解析xls表格 前端实现vue3打开excel 文件地址url或接口文档流二进 dba protobuf 序列化和反序列化 安装 Dell HPE 联想 浪潮 bash freebsd webstorm 医疗APP开发 app开发 Docker Hub docker pull 镜像源 daemon.json ocr zabbix 强制清理 强制删除 mac废纸篓 思科模拟器 思科 Cisco 数据集 LLM 大模型面经 Deepseek 大模型学习 java-ee 压力测试 测试用例 功能测试 游戏程序 ios kind AI写作 AI作画 react next.js 部署next.js QQ 聊天室 spring cloud WebUI DeepSeek V3 AI大模型 kali 共享文件夹 虚拟机 其他 Ark-TS语言 矩阵 log4j EMUI 回退 降级 升级 剧本 muduo X11 Xming VMware xml 弹性计算 虚拟化 计算虚拟化 弹性裸金属 cmos 微信开放平台 微信公众平台 微信公众号配置 gpt linux上传下载 IPMI unix eureka 小番茄C盘清理 便捷易用C盘清理工具 小番茄C盘清理的优势尽显何处? 教你深度体验小番茄C盘清理 C盘变红?!不知所措? C盘瘦身后电脑会发生什么变化? Google pay Apple pay VMware安装mocOS macOS系统安装 openwrt USB网络共享 单元测试 Playwright 自动化测试 微信小程序 vscode 1.86 网站搭建 serv00 jetty undertow Linux无人智慧超市 LInux多线程服务器 QT项目 LInux项目 单片机项目 灵办AI kvm grafana 图形化界面 SSH k8s集群资源管理 云原生开发 云服务器 裸金属服务器 弹性裸金属服务器 p2p hadoop docker run 数据卷挂载 交互模式 vmware 卡死 银河麒麟 kylin v10 麒麟 v10 Cline 自动化编程 gcc Netty 即时通信 NIO AI代码编辑器 SWAT 配置文件 服务管理 网络共享 华为od 英语 ai小智 语音助手 ai小智配网 ai小智教程 智能硬件 esp32语音助手 diy语音助手 wsl2 wsl 银河麒麟桌面操作系统 Kylin OS 国产化 DeepSeek行业应用 Heroku 网站部署 技能大赛 tensorflow 系统开发 binder 车载系统 framework 源码环境 trae GCC crosstool-ng ros2 moveit 机器人运动 游戏机 hugo Ubuntu 24 常用命令 Ubuntu 24 Ubuntu vi 异常处理 micropython esp32 mqtt rpc nuxt3 postgresql 边缘计算 Linux PID eNSP 网络规划 VLAN 企业网络 显示过滤器 ICMP Wireshark安装 半虚拟化 硬件虚拟化 Hypervisor pygame 多层架构 解耦 LInux linux环境变量 田俊楠 k8s资源监控 annotations自动化 自动化监控 监控service 监控jvm code-server 算力 程序员 rime 分析解读 sqlite hive DBeaver 数据仓库 kerberos pgpool openssl 密码学 业界资讯 ci/cd 模拟退火算法 jar gradle visual studio code outlook 数据库系统 C# MQTTS 双向认证 emqx hibernate Docker Compose docker compose docker-compose remote-ssh 直流充电桩 充电桩 nac 802.1 portal 国产操作系统 ukui 麒麟kylinos openeuler W5500 OLED u8g2 TCP服务器 chfs ubuntu 16.04 统信 虚拟机安装 MacMini Mac 迷你主机 mini Apple 监控k8s集群 集群内prometheus 宠物 毕业设计 免费学习 宠物领养 宠物平台 宝塔面板 同步 备份 建站 TRAE 蓝桥杯 小艺 Pura X 大模型入门 大模型教程 VPS Nuxt.js Xterminal EtherCAT转Modbus ECT转Modbus协议 EtherCAT转485网关 ECT转Modbus串口网关 EtherCAT转485协议 ECT转Modbus网关 bug iBMC UltraISO 网络穿透 火绒安全 实时互动 ceph uniapp 恒源云 selete 高级IO DevEco Studio vSphere vCenter 软件定义数据中心 sddc RTMP 应用层 致远OA OA服务器 服务器磁盘扩容 OD机试真题 华为OD机试真题 服务器能耗统计 VR手套 数据手套 动捕手套 动捕数据手套 政务 分布式系统 监控运维 Prometheus Grafana Redis Desktop open webui 深度求索 私域 知识库 gpt-3 文心一言 设计模式 软件工程 echarts 输入法 matlab 传统数据库升级 银行 LLMs av1 电视盒子 机顶盒ROM 魔百盒刷机 中间件 可信计算技术 安全架构 网络攻击模型 服务器数据恢复 数据恢复 存储数据恢复 北亚数据恢复 oracle数据恢复 显卡驱动 3d 数学建模 IPMITOOL BMC 硬件管理 arm 机器人 opcua opcda KEPServer安装 大模型应用 能力提升 面试宝典 技术 IT信息化 oneapi matplotlib etcd 数据安全 RBAC XCC Lenovo 企业微信 Linux24.04 deepin frp 华为认证 网络工程师 繁忙 解决办法 替代网站 汇总推荐 AI推理 金融 移动云 远程 执行 sshpass 操作 MS Materials android studio 腾讯云大模型知识引擎 鲲鹏 FTP 服务器 报错 visualstudio safari 系统 chrome devtools chromedriver AP配网 AK配网 小程序AP配网和AK配网教程 WIFI设备配网小程序UDP开 信息可视化 网页设计 SSL 域名 skynet ArcTS 登录 ArcUI GridItem arkUI mariadb AISphereButler embedding LORA NLP JAVA 昇腾 npu 硬件工程 嵌入式实习 Cursor 线程 docker搭建nacos详解 docker部署nacos docker安装nacos 腾讯云搭建nacos centos7搭建nacos springboot远程调试 java项目远程debug docker远程debug java项目远程调试 springboot远程 pyqt RAID RAID技术 磁盘 存储 Kylin-Server 服务器安装 EasyConnect RustDesk自建服务器 rustdesk服务器 docker rustdesk 黑客技术 流式接口 URL 本地部署 api 服务器主板 AI芯片 ssrf 失效的访问控制 交互 WebRTC 多进程 yolov8 ux 多线程 open Euler dde wps 黑苹果 小游戏 五子棋 sdkman opensearch helm xrdp SRS 流媒体 直播 string模拟实现 深拷贝 浅拷贝 经典的string类问题 三个swap TrinityCore 魔兽世界 开发环境 SSL证书 linux安装配置 elk Python 网络编程 聊天服务器 套接字 TCP 客户端 环境迁移 服务器管理 配置教程 网站管理 崖山数据库 YashanDB VMware创建虚拟机 Ubuntu 24.04.1 轻量级服务器 redhat dash 正则表达式 tidb GLIBC Ubuntu DeepSeek DeepSeek Ubuntu DeepSeek 本地部署 DeepSeek 知识库 DeepSeek 私有化知识库 本地部署 DeepSeek DeepSeek 私有化部署 mysql离线安装 ubuntu22.04 mysql8.0 源码 pdf 群晖 文件分享 iis VSCode 混合开发 环境安装 JDK pyautogui 目标跟踪 OpenVINO 推理应用 银河麒麟操作系统 远程过程调用 Windows环境 京东云 直播推流 命令行 基础入门 编程 服务器部署ai模型 开机自启动 sqlite3 Anolis nginx安装 linux插件下载 Hive环境搭建 hive3环境 Hive远程模式 毕设 Portainer搭建 Portainer使用 Portainer使用详解 Portainer详解 Portainer portainer raid5数据恢复 磁盘阵列数据恢复 centos-root /dev/mapper yum clean all df -h / du -sh 无桌面 串口服务器 risc-v flash-attention bot Docker Reactor 三级等保 服务器审计日志备份 联想开天P90Z装win10 glibc nextjs reactjs 软考 css 压测 ECS cnn 搜索引擎 邮件APP 免费软件 宕机切换 服务器宕机 CLion CPU 主板 电源 网卡 图像处理 网工 curl wget DOIT 四博智联 数据库架构 数据管理 数据治理 数据编织 数据虚拟化 idm Linux awk awk函数 awk结构 awk内置变量 awk参数 awk脚本 awk详解 thingsboard deepseek r1 设置代理 实用教程 iDRAC R720xd linux 命令 sed 命令 相机 权限 sentinel 服务器无法访问 ip地址无法访问 无法访问宝塔面板 宝塔面板打不开 iphone 自动化任务管理 prompt easyui langchain midjourney 远程服务 前后端分离 实习 fd 文件描述符 镜像 飞牛nas fnos x64 SIGSEGV SSE xmm0 稳定性 看门狗 Jellyfin yum源切换 更换国内yum源 Typore FunASR ASR 佛山戴尔服务器维修 佛山三水服务器维修 云桌面 微软 AD域控 证书服务器 TrueLicense vr 我的世界服务器搭建 ruby 无人机 ROS 自动驾驶 超融合 干货分享 黑客工具 密码爆破 金仓数据库 2025 征文 数据库平替用金仓 单例模式 Spring Security 技术共享 我的世界 我的世界联机 数码 交叉编译 C++软件实战问题排查经验分享 0xfeeefeee 0xcdcdcdcd 动态库加载失败 程序启动失败 程序运行权限 标准用户权限与管理员权限 ISO镜像作为本地源 信号处理 eclipse 免费域名 域名解析 执法记录仪 智能安全帽 smarteye 线性代数 电商平台 服务器时间 DocFlow Linux环境 CH340 串口驱动 CH341 uart 485 李心怡 备份SQL Server数据库 数据库备份 傲梅企业备份网络版 Linux的权限 怎么卸载MySQL MySQL怎么卸载干净 MySQL卸载重新安装教程 MySQL5.7卸载 Linux卸载MySQL8.0 如何卸载MySQL教程 MySQL卸载与安装 影刀 #影刀RPA# bcompare Beyond Compare 模拟器 教程 dns是什么 如何设置电脑dns dns应该如何设置 xss H3C pppoe radius AI agent IM即时通讯 剪切板对通 HTML FORMAT Claude 云服务 npm saltstack AnythingLLM AnythingLLM安装 uni-file-picker 拍摄从相册选择 uni.uploadFile H5上传图片 微信小程序上传图片 DenseNet 嵌入式Linux IPC CrewAI edge浏览器 阿里云ECS figma qemu libvirt WebVM 基础环境 个人博客 流水线 脚本式流水线 can 线程池 增强现实 沉浸式体验 应用场景 技术实现 案例分析 AR GoogLeNet VS Code 软件构建 社交电子 虚拟显示器 apt 中兴光猫 换光猫 网络桥接 自己换光猫 EMQX 通信协议 链表 重启 排查 系统重启 日志 原因 游戏开发 网络药理学 生信 gromacs 分子动力学模拟 MD 动力学模拟 上传视频文件到服务器 uniApp本地上传视频并预览 uniapp移动端h5网页 uniapp微信小程序上传视频 uniapp app端视频上传 uniapp uview组件库 显示管理器 lightdm gdm 阻塞队列 生产者消费者模型 服务器崩坏原因 AD 域管理 less ros Ubuntu共享文件夹 共享目录 Linux共享文件夹 浏览器开发 AI浏览器 ssh漏洞 ssh9.9p2 CVE-2025-23419 换源 国内源 Debian 磁盘镜像 服务器镜像 服务器实时复制 实时文件备份 RAGFlow 本地知识库部署 DeepSeek R1 模型 etl HTTP 服务器控制 ESP32 DeepSeek lsb_release /etc/issue /proc/version uname -r 查看ubuntu版本 备选 网站 调用 示例 xcode 毕昇JDK tcp minecraft deekseek vasp安装 dity make 查询数据库服务IP地址 SQL Server 分布式训练 语音识别 AutoDL 本地部署AI大模型 EtherNet/IP串口网关 EIP转RS485 EIP转Modbus EtherNet/IP网关协议 EIP转RS485网关 EIP串口服务器 国标28181 视频监控 监控接入 语音广播 流程 SIP SDP 反向代理 searxng PPI String Cytoscape CytoHubba wordpress 无法访问wordpess后台 打开网站页面错乱 linux宝塔面板 wordpress更换服务器 Docker引擎已经停止 Docker无法使用 WSL进度一直是0 镜像加速地址 perf Radius qt项目 qt项目实战 qt教程 composer 银河麒麟高级服务器 外接硬盘 Kylin 产测工具框架 IMX6ULL 管理框架 信号 根服务器 clickhouse DIFY firewall openstack Xen cfssl junit Logstash 日志采集 epoll 做raid 装系统 Java Applet URL操作 服务器建立 Socket编程 网络文件读取 laravel 匿名管道 内网服务器 内网代理 内网通信 VM搭建win2012 win2012应急响应靶机搭建 攻击者获取服务器权限 上传wakaung病毒 应急响应并溯源 挖矿病毒处置 应急响应综合性靶场 nlp HarmonyOS 需求分析 规格说明书 rustdesk 聚类 程序 性能分析 飞书 dns 虚拟局域网 音乐库 飞牛 autodl 程序员创富 AD域 强化学习 网络结构图 yaml Ultralytics 可视化 rclone AList webdav fnOS OpenSSH 7z powerpoint 单一职责原则 neo4j 数据库开发 database Linux的基础指令 spark HistoryServer Spark YARN jobhistory 实战案例 Headless Linux 序列化反序列化 IPv4 子网掩码 公网IP 私有IP SSH 密钥生成 SSH 公钥 私钥 生成 主从复制 asp.net大文件上传下载 人工智能生成内容 keepalived sonoma 自动更新 CDN xshell termius iterm2 ShenTong 历史版本 下载 zookeeper flink nfs 华为机试 服务网格 istio 游戏引擎 自定义客户端 SAS seleium 虚拟现实 P2P HDLC GIS 遥感 WebGIS QT 5.12.12 QT开发环境 Ubuntu18.04 双系统 GRUB引导 Linux技巧 大大通 第三代半导体 碳化硅 回显服务器 UDP的API使用 Windows ai工具 armbian u-boot CVE-2024-7347 语法 uv h.264 WSL2 上安装 Ubuntu 项目部署到linux服务器 项目部署过程 ftp banner c/c++ 串口 vpn IO模型 MI300x 软负载 AI-native Docker Desktop 捆绑 链接 谷歌浏览器 youtube google gmail web3 用户缓冲区 模拟实现 vscode1.86 1.86版本 ssh远程连接 支付 微信支付 开放平台 ArkTs ArkUI 图形渲染 迁移指南 LLM Web APP Streamlit ubuntu24.04.1 big data cpp-httplib Mac内存不够用怎么办 sysctl.conf vm.nr_hugepages alias unalias 别名 cocoapods 读写锁 办公自动化 自动化生成 pdf教程 rnn SenseVoice 视频编解码 arcgis Kali 渗透 seatunnel odoo 服务器动作 Server action regedit 开机启动 g++ g++13 视觉检测 版本 考研 僵尸世界大战 游戏服务器搭建 IMX317 MIPI H265 VCU rag ragflow 源码启动 高效日志打印 串口通信日志 服务器日志 系统状态监控日志 异常记录日志 运维监控 webgl k8s二次开发 集群管理 UOS1070e 代码托管服务 trea idea 代理 小智AI服务端 xiaozhi TTS 玩机技巧 软件分享 软件图标 私有化 推荐算法 swoole FTP服务器 Ubuntu22.04 开发人员主页 v10 软件 ldap 架构与原理 lio-sam SLAM 多个客户端访问 IO多路复用 TCP相关API DeepSeek r1 Open WebUI cd 目录切换 OpenManus 宝塔 策略模式 自学笔记 小米 澎湃OS Android 上传视频至服务器代码 vue3批量上传多个视频并预览 如何实现将本地视频上传到网页 element plu视频上传 ant design vue vue3本地上传视频及预览移除 miniapp 真机调试 调试 debug 断点 网络API请求调试方法 HiCar CarLife+ CarPlay QT RK3588 Node-Red 编程工具 流编程 IDEA 键盘 移动魔百盒 tailscale derp derper 中转 USB转串口 triton 模型分析 大文件分片上传断点续传及进度条 如何批量上传超大文件并显示进度 axios大文件切片上传详细教 node服务器合并切片 vue3大文件上传报错提示错误 大文件秒传跨域报错cors 互信 NLP模型 代理服务器 bat Unity Dedicated Server Host Client 无头主机 aarch64 编译安装 HPC Deepseek-R1 私有化部署 推理模型 cursor lua MCP server C/S windows日志 vue-i18n 国际化多语言 vue2中英文切换详细教程 如何动态加载i18n语言包 把语言json放到服务器调用 前端调用api获取语言配置文件 常用命令 文本命令 目录命令 python3.11 安防软件 端口测试 域名服务 DHCP 符号链接 配置 deep learning 性能测试 word css3 企业网络规划 华为eNSP rpa 粘包问题 浏览器自动化 大模型部署 查看显卡进程 fuser navicat ArtTS 加解密 Yakit yaklang 抓包工具 UDP 带外管理 nvm whistle 流量运营 Attention vu大文件秒传跨域报错cors CentOS perl docker部署Python llama.cpp Python基础 Python教程 Python技巧 Unity插件 iventoy VmWare OpenEuler 环境配置 zip unzip 软链接 硬链接 hosts gnu 网络爬虫 多路转接 Xinference 多端开发 智慧分发 应用生态 鸿蒙OS 云耀服务器 rtsp服务器 rtsp server android rtsp服务 安卓rtsp服务器 移动端rtsp服务 大牛直播SDK 问题解决 ubuntu24 vivado24 沙盒 SEO hexo kotlin ssh远程登录 虚幻引擎 virtualbox grub 版本升级 扩容 元服务 应用上架 HarmonyOS NEXT 原生鸿蒙 Sealos 大模型推理 论文阅读 计算生物学 生物信息 基因组 centos 7 烟花代码 烟花 元旦 性能调优 安全代理 嵌入式系统开发 fstab 知识图谱 WLAN db rancher top Linux top top命令详解 top命令重点 top常用参数 信创 信创终端 中科方德 网络建设与运维 网络搭建 神州数码 神州数码云平台 云平台 ubuntu20.04 ros1 Noetic 20.04 apt 安装 react native ip协议 SVN Server tortoise svn ABAP 抗锯齿 CentOS Stream 物联网开发 HAProxy TCP协议 存储维护 NetApp存储 EMC存储 KylinV10 麒麟操作系统 Vmware milvus 健康医疗 互联网医院 MVS 海康威视相机 proxy模式 AI员工 onlyoffice 在线office 磁盘清理 智能音箱 智能家居 fast 容器技术 jina 软件卸载 系统清理 wpf 拓扑图 Linux权限 权限命令 特殊权限 wsgiref Web 服务器网关接口 docker搭建pg docker搭建pgsql pg授权 postgresql使用 postgresql搭建 iperf3 带宽测试 Qwen2.5-VL vllm ardunio BLE 对比 工具 meld DiffMerge 开发 nosql 合成模型 扩散模型 图像生成 java-rocketmq 项目部署 OpenHarmony 鸿蒙开发 移动开发 win服务器架设 windows server PX4 网卡的名称修改 eth0 ens33 MacOS sequoiaDB IMM deployment daemonset statefulset cronjob 进程优先级 调度队列 进程切换 AI Agent 字节智能运维 prometheus数据采集 prometheus数据模型 prometheus特点 MDK 嵌入式开发工具 论文笔记 sublime text 状态模式 区块链 本地化部署 kernel 端口聚合 windows11 su sudo chrome 浏览器下载 chrome 下载安装 谷歌浏览器下载 Linux find grep 钉钉 visual studio 内网环境 MySql System V共享内存 进程通信 ranger MySQL8.0 像素流送api 像素流送UE4 像素流送卡顿 像素流送并发支持 docker desktop image harmonyOS面试题 xpath定位元素 端口 查看 ss 防火墙 NAT转发 NAT Server 欧标 OCPP openvpn server openvpn配置教程 centos安装openvpn 极限编程 树莓派 VNC 安装MySQL docker部署翻译组件 docker部署deepl docker搭建deepl java对接deepl 翻译组件使用 conda配置 conda镜像源 Reactor反应堆 解决方案 西门子PLC 通讯 macOS vnc yum换源 mm-wiki搭建 linux搭建mm-wiki mm-wiki搭建与使用 mm-wiki使用 mm-wiki详解 开机黑屏 React Next.js 开源框架 电视剧收视率分析与可视化平台 搜狗输入法 中文输入法 数字证书 签署证书 copilot 智能电视 风扇控制软件 yolov5 ubuntu 18.04 服务器正确解析请求体 dock 加速 热榜 IO 视频平台 录像 视频转发 视频流 js MAC SecureCRT UEFI Legacy MBR GPT U盘安装操作系统 MobaXterm 搭建个人相关服务器 国产数据库 瀚高数据库 数据迁移 下载安装 达梦 DM8 MAVROS 四旋翼无人机 接口优化 通信工程 毕业 mcp服务器 client close 离线部署dify