HTTP协议初识·中篇

加上目录,会出现导向不正确的情况,可能是bug,目录一长就容易出错?

本篇主要讲解了:

        网页分离(网页代码和.c文件分离)

        html链接跳转

        网页添加图片

        确认并返回资源类型

        填写正文长度属性

        添加表单

        临时重定向

补充知识:

        前后端区别

        浏览器是如何显示页面的

        http各字段属性是什么意思

        提交给指定的路径有什么意义?

        http方法

        http状态码

        HTTP常见Header

        写入cookie信息

源码

承接上文

 HTTP协议初识·上篇_清风玉骨的博客-CSDN博客

第二点在上文中已经完成 

1:网页分离

网页分离1.0:404文件一定会存在

readFile函数1.1:读取文件中的内容

读取文件中的内容

 

404html文件

在vscode有快捷生成网页,!+  Tab键 

测试1:网页分离与404文件

成功

 

404访问

2:html链接跳转

2.0添加点击键与返回键

其实游览器的返回键可以直接使用,效果是一样的

测试2:返回键与404测试

返回键

404(返回时候不存在)

故意写错

别人的404网页

高级一点…… 

补充1:后端与前端区别做那些部分?

3:添加图片

测试3:图片测试

在wwwroot下创建一个image文件夹,把图片放进去就行了,这里不显示的原因估计是云服务器配置太低,导致图片加载不出来,因为这个.jpg的格式一般都比较大,这边测试用的是大概200kb的图片,果然配置低了点

补充2:浏览器是如何显示网页的?

        其实很简单,例如访问首页,就是把的全部资源下载下来,再组合显示过来就行了

        即用户看到的网页结果,可能是多个资源组合而成的!所以要获取一张网页效果,浏览器一定会发起多次http请求,这也是为什么会有高并发访问原因

4:如何正确返回资源类型?

为了返回正确的文件信息,我们得填写Content-Type来,让它不要一直指向首页,这里直接搜索对照表即可找到相应的对照

suffixToDesc函数4.1后缀分离函数

测试4:图片再测试

        不过通常这类的图片音频之类的比较吃资源的文件一般并不会直接放置在服务器里面,而是放置在别的专门放置资源的服务器上,为了进一步测试,我们还可以使用别人的图床中的图片,走别人的服务器

这一次很幸运的显示出来的,要注意很多网站的图片其实并不能直接使用,视乎做了处理,这里就不深加讨论了

5:添加正文长度

认识系统调用stat

测试5:添加正文长度

补充2:http各个字段的意思

6:添加表单

目前我们写的都是单向的,无法做到交互,因为使用的是,为了做到可以交互我们得引用表单

补充6.0:表单各部分意思

测试6:表单测试

GET方法:数据拼接到url后面

 

POST方法:数据放到正文部分

GET与POST方法只有私密性这一点不同

要想安全,得到https这一点我们后续再详谈

这里百度一下就是直接使用GET方法的

补充3:提交给指定的路径有什么意义?

补充4:使用unordered_map进行修改可以实现功能路由

这里是伪代码,仅提供思路

补充5:HTTP的方法

        其实除了GET和POST方法我们常使用,其他的方法我们使用的频率并不高,所以我们这里也就不详谈了,不是重点

所以GET和POST相比

如果你只想单纯的从远端获取资源:GET

如果你想要提参:GET/POST

无私密性:GET(简单)

私密性:POST

补充6:HTTP的状态码

        不过我们需要知道,虽然这些状态码是协议标准,不过解释权还是在写代码的人在操作,并且现在一般也不需要严格遵守,比如5开头的状态码我们平时就很少见,这一点想必大家都会意识到这一点,毕竟有点掉价,所以大部分服务器的错误,基本上都是返回4开头的状态码,因为这些错误要么就是很苛刻,要么就是大家都懂得,而且很重要的一点,根据5状态码的信息,一些非法分子可能就会利用这个错误码来进行攻击服务器,自然而然的这也成为了不成文规定了,基于这一点关于状态码与前端后端程序员对于状态码的认识注定是不一样的。

        不过现在的浏览器很智能了,它一般也不会通过状态码来规定网页显示内容,一般都是通过正文部分来显示的,这一点做得很好了

补充7:3开头重定向状态码

301 永久重定向  307 临时重定向 

        重定向可能提出来会有些不了解,其实我们很容易见到这个的,比如当年点击这个网站的时候,就会跳转来一个广告页面,这就是重定向的一种,这种一般是临时重定向,那边是金主就给那边定向导流,这种强制重定向就是用3开头的状态码来实现的

7:临时重定向

HTTP常见Header

Content-Type: 数据类型(text/html等)
Content-Length: Body的长度
Host: 客户端告知服务器, 所请求的资源是在哪个主机的哪个端口上;
User-Agent: 声明用户的操作系统和浏览器版本信息;
Referer: 当前页面是从哪个页面跳转过

Location: 搭配3xx状态码使用, 告诉客户端接下来要去哪里访问;
Cookie: 用于在客户端存储少量信息. 通常用于实现会话(session)的功能;

我们需要填写好Location这个报头属性

        静态网页 -- 目前我们所写的,因为C++并不适合写网页,动态网页专注与交互,用C++来做比较麻烦,所以我们C++程序员一般也不适用写复杂网页

修正bug代码 

之前的图片不显示bug已经修复,是之前的函数用错的原因,现在进行更正

 

 

        不过需要注意的是,这样会导致当资源不存在的时候,跳转的404页面将会跳转不成功,这个问题出现在添加的读取内容大小那边,因为当资源不存在的时候读取大小为-1(本程序设计),导致404页面读取失败,这里就不进行修正了,注意这一点就好了

源码

HttpServer.cc

#include "HttpServer.hpp"
#include <memory>using namespace std;
using namespace server;void Usage(std::string proc)
{cerr << "Usage:\n\t" << proc << " port\r\n\r\n";
}std::string suffixToDesc(const std::string suffix)
{std::string ct = "Content-Type: ";if (suffix == ".html")ct += "text/html";else if (suffix == ".jpg")ct += "application/x-jpg";ct += "\r\n";return ct;
}// 1. 服务器和网页分离,html
// 2. url -> / : web根目录
// 3. 我们要正确给客户端返回资源类型,我们首先要自己知道!所有的资源都有后缀!!!
bool Get(const HttpRequest &req, HttpResponse &resp)
{/*伪代码if(req.path == "test.py") 假如提交到一个python脚本{首先建立进程间通信, pipefork创建子进程, execl("/bin/puthon", test.py) 这样子进程就执行这个脚本父进程, 将req.parm 通过管道写过某些后端语言, C++并不适合写这个, 一般是指, python, java, php等这也是为什么服务器是C++写的,但是它的后端却是其他语言的原因}if(req.path == "/search"){req.parm使用我们自己写到C++方法,提供服务}*/// for testcout << "----------------------- http start ------------------------------------" << endl;cout << req.inbuffer << endl; // 暂时不做其他处理,直接打印出来看请求内容std::cout << "method: " << req.method << std::endl;std::cout << "url: " << req.url << std::endl;std::cout << "httpversion: " << req.httpversion << std::endl;std::cout << "path: " << req.path << std::endl;std::cout << "suffix: " << req.suffix << std::endl;std::cout << "size: " << req.size << "字节" << std::endl;cout << "------------------------ http end -------------------------------------" << endl;// std::string respheader = "Content-Type: text/html\r\n"; 不用这个了,固定死的std::string respline = "HTTP/1.1 200 OK\r\n";  // 其实这里的状态码也应该修改,不过这边就不做字符串拼接的过程了,就这样显示// std::string respline = "HTTP/1.1 307 Temporary Redirect\r\n";std::string respheader = suffixToDesc(req.suffix);// 往后每次http请求,都会自动携带曾经设置的所有cookie,帮服务器进行鉴权行为respheader += "Set-Cookie: name=1234567abcdefg; Max-Age=120\r\n"; // 有时间限制的120秒,时间到自动失效// respheader += "Location: https://www.bilibili.com/\r\n";std::string respblank = "\r\n"; // 空行// 网页 -- 自己写一个简单的, 不要在C++中写html,这里是测试,很不方便std::string body; // 从文件中来body.resize(req.size+1);if (!Util::readFile(req.path, (char*)body.c_str(), req.size)){Util::readFile(html_404, (char*)body.c_str(), req.size); //  这个操作一定能成功}/*测试 -- 这个代码会导致本地图片显示不出来,因为视频图片都是二进制的东西不能按行读取!要用readif (!Util::readFile(req.path, &body)){Util::readFile(html_404, &body); //  这个操作一定能成功}*//*这里没用了,修正还是有错误,错误转移了if (req.size > 0){respheader += "Content-Length: ";  添加正文的长度respheader += std::to_string(req.size);respheader += "\r\n";}*/// 直接拼接就可以了,本身很简单respheader += "Content-Length: ";respheader += std::to_string(body.size());respheader += "\r\n";resp.outbuffer += respline;resp.outbuffer += respheader;resp.outbuffer += respblank;cout << "----------------------- http response start ------------------------------------" << endl;std::cout << resp.outbuffer << std::endl;cout << "----------------------- http response end ------------------------------------" << endl;resp.outbuffer += body;return true;
}// ./httpServer 8080    -- 这里实际是80端口号,不过这里是为了测试就不用了,并且80也绑定不了,前一千多号基本内部资源无法绑定
int main(int argc, char *argv[])
{if (argc != 2){Usage(argv[0]);exit(0);}uint16_t port = atoi(argv[1]);unique_ptr<HttpServer> httpsvr(new HttpServer(Get, port));/*伪代码httpsvr->registerCb("/", Get); // 这就是 功能路由!httpsvr->registerCb("/search", Search);httpsvr->registerCb("/test.py", Other);*/httpsvr->initServer();httpsvr->start();return 0;
}

HttpServer.hpp

#pragma once#include <iostream>
#include <string>
#include <cstring>
#include <cstdlib>
#include <functional>
#include <unistd.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <sys/wait.h>
#include <signal.h>
#include <pthread.h>
#include <unordered_map>#include "Protocol.hpp"namespace server
{enum{USAGE_ERR = 1,SOCKET_ERR,BIND_ERR,LISTEN_ERR};static const uint16_t gport = 8080;static const int gbacklog = 5; // 10、20、50都可以,但是不要太大比如5千,5万using func_t = std::function<bool(const HttpRequest &, HttpResponse &)>;    // 回调class HttpServer{public:HttpServer(func_t func, const uint16_t &port = gport) : _func(func), _listensock(-1), _port(port){}void initServer(){// 1. 创建socket文件套接字对象 -- 流式套接字_listensock = socket(AF_INET, SOCK_STREAM, 0); // 第三个参数默认 0if (_listensock < 0){exit(SOCKET_ERR);}// 2.bind绑定自己的网路信息 -- 注意包含头文件struct sockaddr_in local;memset(&local, 0, sizeof(local));local.sin_family = AF_INET;local.sin_port = htons(_port);      // 这里有个细节,我们会发现当我们接受数据的时候是不需要主机转网路序列的,因为关于IO类的接口,内部都帮我们实现了这一功能,这里不帮我们做是因为我们传入的是一个结构体,系统做不到local.sin_addr.s_addr = INADDR_ANY; // 接受任意ip地址if (bind(_listensock, (struct sockaddr *)&local, sizeof(local)) < 0){exit(BIND_ERR);}// 3. 设置socket 为监听状态 -- TCP与UDP不同,它先要建立链接之后,TCP是面向链接的,后面还会有“握手”过程if (listen(_listensock, gbacklog) < 0) // 第二个参数backlog后面再填这个坑{exit(LISTEN_ERR);}}/*伪代码void registerCb(std::string servicename, func_t cb) // 写一个注册方法{funcs.insert(std::make_pair(servicename, cb));}*/void HandlerHttp(int sock){// 1. 读到完整的http请求// 2. 反序列化// 3. 反序列化后得到httprequest, 回调填写httpresponse, 利用_func(req, resp)// 4. 序列化resp// 5. sendchar buffer[4096];HttpRequest req;HttpResponse resp;size_t n = recv(sock, buffer, sizeof(buffer) - 1, 0);   // 大概率我们直接能读取到完整的http请求if(n > 0){buffer[n] = 0;req.inbuffer = buffer;req.parse();// funcs[req.path](req, resp); 有了哈希表就不需要下面这一句了, 使用路径绑定服务, 这里并没有实现_func(req, resp);  // 可以根据bool返回值进行判断,这里就不判断了send(sock, resp.outbuffer.c_str(), resp.outbuffer.size(), 0);}}void start(){for (;;) // 一个死循环{// 4. server 获取新链接// sock 和 client 进行通信的fdstruct sockaddr_in peer;socklen_t len = sizeof(peer);int sock = accept(_listensock, (struct sockaddr *)&peer, &len);if (sock < 0){continue;}/* 这里直接使用多进程版的代码进行修改version 2 多进程版(2) -- 注意子进程会继承父进程的一些东西,父进程的文件操作符就会被子进程继承,即父进程的文件操作符那个数字会被继承下来,指向同一个文件,但是文件本身不会被拷贝一份也就是说子进程可以看到父进程创建的文件描述符sock和打开的listensock     */pid_t id = fork();if (id == 0) // 当id为 0 的时候就代表这里是子进程{/* 关闭不需要的文件描述符 listensock -- 子进程不需要监听,所以我们要关闭这个不需要的文件描述符即使这里不关,有没有很大的关系,但是为了防止误操作我们还是关掉为好   */close(_listensock);if (fork() > 0) exit(0); // 解决方法1: 利用孤儿进程特性HandlerHttp(sock);close(sock);exit(0);}/* 一定要关掉,否则就会造成文件描述符泄漏,但是这里的关掉要注意了,这里只是把文件描述符的计数-1子进程已经继承过去了,所以这里也可以看做,父进程立马把文件描述符计数-1,只有当子进程关闭的时候,这个文件描述符真正的被关闭了所以后面申请的链接使用的还是这个4号文件描述符,因为计算机太快了close(sock);   *//* father那么父进程干嘛呢? 直接等待吗? -- 显然不能,这样又会回归串行运行了,因为等待的时候会阻塞式等待且这里并不能用非阻塞式等待,因为万一有一百个链接来了,就有一百个进程运行,如果这里非阻塞式等待一但后面没有链接到来的话.那么accept这里就等不到了,这些进程就不会回收了 */// 不需要等待了 version 2waitpid(id, nullptr, 0);}}~HttpServer() {}private:int _listensock;uint16_t _port;func_t _func;// std::unordered_map<std::string, func_t> funcs;   构建一个哈希表用来实现功能路由};} // namespace server

makefile

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

Protocol.hpp

#pragma once#include <iostream>
#include <string>
#include <vector>
#include <sstream> // 可以直接以空格作为分隔符来进行分割
#include <sys/types.h>
#include <sys/stat.h>
#include <unistd.h>#include "Util.hpp"const std::string sep = "\r\n";                  // 分隔符
const std::string default_root = "./wwwroot";    // web起始目录,前面的 ./ 加不加都可以
const std::string home_page = "index.html";      // 默认首页,任何服务器都会有这个默认首页
const std::string html_404 = "wwwroot/404.html"; // 这个文件一定会存在class HttpRequest
{
public:HttpRequest() {}~HttpRequest() {}void parse() // 解析{// 1. 从inbuffer中拿到第一行,分隔符\r\nstd::string line = Util::getOneline(inbuffer, sep);if (line.empty())return;// 2. 从请求行中提取三个字段 -- 下面放开的三个std::cout << "line: " << line << std::endl; // 打印出来显示一下std::stringstream ss(line);                 // 可以直接以空格作为分隔符来进行分割ss >> method >> url >> httpversion;/*2.1 /search?name=zhangsan&pwd=123456我们首先通过 ? 将左右进行分离如果是POST方法,本来就是分离的!左边path, 右边parm*/// 3. 添加web默认路径path = default_root; // 未来可以进行修改  变成 ./wwwrootpath += url;         // 到这一步之后就会  变成 ./wwwroot/a/b/c.html// 未来访问路径都会从这个路径下开始访问// 这边会遇到一个问题,当url是一个 / 的时候就不行,拼接的时候会变成 ./wwwroot/ 没有具体目标if (path[path.size() - 1] == '/')path += home_page; // 加一个判断就行了// 4. 获取path对应的资源后缀// ./wwwroot/index.html// ./wwwroot/test/a.html// ./wwwroot/image/宇宙.jpgauto pos = path.rfind(".");if (pos == std::string::npos)suffix = ".html";elsesuffix = path.substr(pos); // 截取到末尾// 5. 得到资源的大小struct stat st;int n = stat(path.c_str(), &st);// 这个系统调用会自动填充字段if(n == 0) size = st.st_size;   // 成功返回 0else size = -1;}public:std::string inbuffer;/*  我们可以细分许多字段,当需要什么就可以添加什么,这里为了简洁就不做这些工作了std::string reqline; // 请求行std::vector<std::string> reqheader; // 请求报头std::string body;   // 请求正文*/std::string method;      // 请求方法std::string url;         // 网址std::string httpversion; // 请求版本std::string path;        // web默认路径std::string suffix;      // 文件后缀int size;                // 正文的长度,把请求的资源的大小返回// std::string parm;        伪代码,用于放置 ? 右边的数据
};class HttpResponse
{
public:std::string outbuffer;
};

Util.hpp

#pragma once#include <iostream>
#include <string>
#include <fstream>class Util
{
public:// XXXX XXX XXX\r\nYYYYY  -- 格式// 第二个参数是分隔符,暴露在外部,让外部传进来static std::string getOneline(std::string &buffer, const std::string &sep) // 类内静态方法可以直接使用 -- 为了方便写,就定义成静态的{auto pos = buffer.find(sep);if (pos == std::string::npos)return "";                           // 没有找到分隔符std::string sub = buffer.substr(0, pos); // [ ) 左闭右开 拿到这一行字段// buffer.erase(0, sub.size() + sep.size());   删除这一行return sub;}static bool readFile(const std::string resource, char *buffer, int size) // 视频图片都是二进制的东西不能按行读取!{               std::ifstream in(resource, std::ios::binary);  // 按照二进制的方式读取文件内容if (!in.is_open()) return false; // resource not foundin.read(buffer, size);  in.close();return true;}/*测试 -- 这个代码会导致本地图片显示不出来,因为视频图片都是二进制的东西不能按行读取!要用readstatic bool readFile(const std::string resource, std::string *out) // 视频图片都是二进制的东西不能按行读取!{               std::ifstream in(resource, std::ios::binary);  // 按照二进制的方式读取文件内容if (!in.is_open()) return false; // resource not foundstd::string line;while (std::getline(in, line)){*out += line;}in.close();return true;*/
};

其他文件内容自行添加即可,这里为了简便篇幅就不粘贴了

未完持续更新中……

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Jdk8 动态编译 Java 源码为 Class 文件(三)

Jdk8 动态编译 Java 源码为 Class 文件 一.JDK版本二.工程介绍1.依赖2.启动类3.配置类&#xff08;用于测试依赖注入&#xff09;4.工具类1.Java 源码文件读取类2.SpringBoot 容器实例管理类 5.测试类1.抽象类2.接口类3.默认抽象实现4.默认接口实现 6.接口类1.测试接口2.类重载…

SpringWeb(SpringMVC)

目录 SpringWeb介绍 搭建 SpringWeb SpringWeb介绍 Spring Web是一个基于 Servlet API 构建的原始 web 框架&#xff0c;用于构建基于MVC模式的Web应用程序。在 web 层框架历经 Strust1&#xff0c;WebWork&#xff0c;Strust2 等诸多产品的历代更选 之后&#xff0c;目前业界普…

QT DAY4

一、使用鼠标时间完成组件的移动 #ifndef WIDGET_H #define WIDGET_H#include <QWidget> #include<QDebug> #include<QMouseEvent>QT_BEGIN_NAMESPACE namespace Ui { class Widget; } QT_END_NAMESPACEclass Widget : public QWidget {Q_OBJECTpublic:Widg…

LLMs:OpenAI官方重磅更新——新增GPT-3.5Turbo调和API更新功能

LLMs&#xff1a;OpenAI官方重磅更新——新增GPT-3.5Turbo调和API更新功能 导读&#xff1a;2023年8月22日&#xff0c;OpenAI官方发布&#xff0c;开发者现在可以使用自己的数据来定制适用于其用例的GPT-3.5 Turbo模型。GPT-3.5 Turbo的微调现在已经可用&#xff0c;GPT-4的微…

【算法与数据结构】106、LeetCode从中序与后序遍历序列构造二叉树

文章目录 一、题目二、解法三、完整代码 所有的LeetCode题解索引&#xff0c;可以看这篇文章——【算法和数据结构】LeetCode题解。 一、题目 二、解法 思路分析&#xff1a;首先我们要知道后序遍历数组的最后一个元素必然是根节点&#xff0c;然后根据根节点在中序遍历数组中的…

【LeetCode-中等题】994. 腐烂的橘子

文章目录 题目方法一&#xff1a;bfs层序遍历 题目 该题值推荐用bfs&#xff0c;因为是一层一层的感染&#xff0c;而不是一条线走到底的那种&#xff0c;所以深度优先搜索不适合 方法一&#xff1a;bfs层序遍历 广度优先搜索&#xff0c;就是从起点出发&#xff0c;每次都尝…

Android GB28181客户端开发(1):GB28181协议简介

Android GB28181客户端开发(1):GB28181协议简介 公共安全视频监控联网系统信息传输、交换、控制技术要求(2016版) 源码请翻到文章结尾 介绍GB28181协议 GB28181协议是一种基于IP网络的远程视频监控系统,它定义了设备之间的通信协议和数据格式。GB28181协议的主要特点是支…

【Rust】001-基础语法:变量声明及数据类型

【Rust】001-基础语法&#xff1a;变量声明及数据类型 文章目录 【Rust】001-基础语法&#xff1a;变量声明及数据类型一、概述1、学习起源2、依托课程 二、入门程序1、Hello World2、交互程序代码演示执行结果 3、继续上难度&#xff1a;访问链接并打印响应依赖代码执行命令 三…

Collections和CollectionUtils集合操作

0.引入依赖 <dependency><groupId>org.apache.commons</groupId><artifactId>commons-collections4</artifactId><version>4.4</version> </dependency> 一.Collections用法&#xff1a; 01、排序操作 reverse(List list)…

【摆烂之小左】Maven配置IDEA教程

Maven是什么 Maven项目对象模型(POM)&#xff0c;可以通过一小段描述信息来管理项目的构建&#xff0c;报告和文档的项目管理工具软件。 Maven 除了以程序构建能力为特色之外&#xff0c;还提供高级项目管理工具。由于 Maven 的缺省构建规则有较高的可重用性&#xff0c;所以常…

数学建模:模糊综合评价分析

&#x1f506; 文章首发于我的个人博客&#xff1a;欢迎大佬们来逛逛 数学建模&#xff1a;模糊综合评价分析 文章目录 数学建模&#xff1a;模糊综合评价分析综合评价分析常用评价方法一级模糊综合评价综合代码 多级模糊综合评价总结 综合评价分析 构成综合评价类问题的五个…

go语言基础操作--二

a : 10str : "mike"//匿名函数&#xff0c;没有函数名字 形成一个闭包,函数定义&#xff0c;还没有调用f1 : func() { //:自动推到类型fmt.Println("a ", a)fmt.Println("str ", str)}f1()//给一个函数类型起别名 这个写法不推荐type FuncType …

Flutter状态管理 — 探索Flutter中的状态

前言 随着响应式编程的理念&Flutter被大众所了解以来&#xff0c;状态管理一直是一个引人深思的话题。如果想要学习好Flutter这样的响应式的编程框架就一定是离不开状态管理的。我遇到过很多没有了解过响应式编程框架的&#xff0c;或者从事后端开发&#xff0c;自己想用F…

docker笔记4:高级复杂安装-mysql主从复制

1.主从搭建步骤 1.1新建主服务器容器实例3307 docker run -p 3307:3306 --name mysql-master \ -v /mydata/mysql-master/log:/var/log/mysql \ -v /mydata/mysql-master/data:/var/lib/mysql \ -v /mydata/mysql-master/conf:/etc/mysql \ -e MYSQL_ROOT_PASSWORDroot \ -d…

Linux的目录结构特点

Linux的目录结构特点 1、使用树形目录结构来组织和管理文件。 2、整个系统只有一个根目录&#xff08;树根&#xff09;&#xff0c;Linux的根目录用“/”表示。 3、其他所有分区以及外部设备&#xff08;如硬盘&#xff0c;光驱等&#xff09;都是以根目录为起点&#xff0…

【已解决】激活虚拟环境报错:此时不应有Anaconda3\envs\[envs]\Library\ssl\cacert.pem。

新建虚拟环境后&#xff0c;进入虚拟环境的时候出现这样的报错&#xff1a; 此时不应有Anaconda3 envs yolov5 Library ssl cacert.pem。 但是之前装的虚拟环境也还能再次激活&#xff0c;base环境也无任何问题&#xff0c;仅新装的虚拟环境无法激活。 查遍了百度谷歌&#xff…

ThreadLocal源码剖析(简单理解)

Thread部分源码 public class Thread implements Runnable {ThreadLocal.ThreadLocalMap threadLocals null; }ThreadLocal源码,其中ThreadLocal有一个静态内部类ThreadLocalMap,这个Map不是类似二叉树类型的,只是一个普通数组,其中具体使用什么算法其实我也不太理解. 然后对…

通过ref 操作dom , 点击按钮后跳转到页面指定图片位置

滚动图片到视图 定义了一个名为 scrollToIndex 的函数&#xff0c;它接受一个参数 index。当按钮被点击时&#xff0c;这个函数会被调用&#xff0c;并根据传入的 index 值来滚动到对应的图片。 以 alt 来标记图片位置 alt“Tom” import { useRef } from "react";c…

1.(python数模)单函数读取常用文件

Python单函数读取常用文件 代码如下&#xff1a; import pandas as pd# 读取数据文件 def readDataFile(readPath): # readPath: 数据文件的地址和文件名try:if (readPath[-4:] ".csv"):dfFile pd.read_csv(readPath, header0, sep",") # 间隔符为逗…

I2C与I3C的对比

I2C与I3C的对比 电气特性 I2C 1.半双工 2.串行数据线(SDA)和串行时钟线(SCL) 3.数据线漏极开路&#xff0c;即I2C接口接上拉电阻 4.I2C总线运行速度&#xff1a;**标准模式100kbit/s&#xff0c;快速模式400kbit/s&#xff0c;快速模式plus 1Mbit/s&#xff0c;**高速模式…