最新资讯

  • Linux:Http协议

Linux:Http协议

2025-05-03 05:00:37 1 阅读

一、认识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/7464.html

搜索文章

Tags

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