【网络】http协议
🥁作者: 华丞臧.
📕专栏:【网络】
各位读者老爷如果觉得博主写的不错,请诸位多多支持(点赞+收藏+关注
)。如果有错误的地方,欢迎在评论区指出。
推荐一款刷题网站 👉 LeetCode刷题网站
文章目录
- 前言
- 一、http协议
- 1.1 认识URL
- http是做什么的
- urlencode和urldecode
- 1.2 http格式
- http请求
- http响应
- 1.3 http的方法
- 1.4 http的状态码
- 1.5 http常见Header
- 二、http服务器
- 2.1 最简单的http服务器
- 2.2 简单的http服务器
- 2.3 html的http服务器
- 2.4 表单
- get
- post
- get和post特点
- 2.5 永久和临时重定向
- 2.6 Cookie
前言
在上一篇文章中我们见识了序列化和反序列化,其是处于应用层的,是进行网络通信双方进行一种约定;在网络发展至今,已经有了一些非常成熟的场景,并且有大佬自定义的一些协议,写的很好因此成为了应用层特定协议的标准,
一、http协议
虽然我们说,应用层协议是我们程序猿自己定的。但实际上,已经有大佬们定义了一些现成的,又非常好用的应用层协议,供我们直接参考使用,http(超文本传输协议)就是其中之一,底层主流采用的是tcp协议,http是无连接的。
1.1 认识URL
平时我们所说的网址就是URL。
服务器地址是域名,域名就是字符串类型的字段,域名在我们访问网网站的时候必须被转换称为IP地址,访问网络服务必须具有网络号即端口号,网络通信的本质就是socket,IP+port。
一般在使用确定协议的时候,再url上面会缺省端口号,在进行网络通信时端口号会被(浏览器)自动添加上,所以浏览器访问指定url时,浏览器必须给我们自动添加port,浏览器就必须知道这些端口,所以特定的众所周知的服务端口号都是并且必须是确定的。因此,在前面学习端口号时,我们说用户自己使用的套接字不能在0~1023。
http --> 80
https --> 443
sshd --> 22
http是做什么的
http是以网页的形式呈现的,比如说查阅文档、查看视频;所以http是获取网页资源的,视频、音频等也都是文件,而网页也是一个.html文件。
http是向特定服务器申请特定资源的,获取到本地进行展示或者某种使用的。
网页上的资源在该网页的网络服务器(软件)所在的服务器(硬件)上,而服务器都是Linux,资源文件存放在Linux服务器上;用户在网页上申请资源时,服务器需要根据路径找到该文件,所以URL中需要添加资源文件的路径。/
是Linux下的路径分隔符,但是需要注意这个/
不一定为根目录,因为软件层面可以做一些处理,比如在这个目录前添加一个路径。
urlencode和urldecode
像 / ? :等这样的字符,已经被url当做特殊意义理解了,因此这些字符不能随意出现。
比如, 某个参数中需要带有这些特殊字符,就必须先对特殊字符进行转义。在进行网络传输进行编码时,尤其是http中有些字段也会进行编码,服务端进行解码。编码是浏览器做的,解码是服务器做的。
转义的规则如下:
将需要转码的字符转为16进制,然后从右到左,取4位(不足4位直接处理),每2位做一位,前面加上%,编码成%XY格式。
“+” 被转义成了 “%2B”
urldecode就是urlencode的逆过程:
urlencode工具
网页的内容在开发者工具中可以看到,如下图:
1.2 http格式
http是基于行的协议。
http请求
- 首行:[方法] + [url] + [版本]
- Header:请求的属性, 冒号分割的键值对;每组属性之间使用\n分隔;遇到空行表示Header部分结束 。
- Body:空行后面的内容都是Body,Body允许为空字符串。如果Body存在,则在Header中会有一个 Content-Length属性来标识Body的长度。
http响应
- 首行: [版本号] + [状态码] + [状态码解释]
- Header:请求的属性,冒号分割的键值对;每组属性之间使用\n分隔;遇到空行表示Header部分结束。
- Body:空行后面的内容都是Body,Body允许为空字符串。如果Body存在, 则在Header中会有一个 Content-Length属性来标识Body的长度;如果服务器返回了一个html页面,那么html页面内容就是在 body中。
1.3 http的方法
方法 | 说明 | 支持的HTTP协议版本 |
---|---|---|
GET | 获取资源 | 1.0、1.1 |
POST | 传输实体主体 | 1.0、1.1 |
PUT | 传输文件 | 1.0、1.1 |
HEAD | 获取报文首部 | 1.0、1.1 |
DELETE | 删除文件 | 1.0、1.1 |
OPTIONS | 询问支持的方法 | 1.1 |
TRACE | 追踪路径 | 1.1 |
CONNECT | 要求用隧道协议链接代理 | 1.1 |
LINK | 建立和资源之间的联系 | 1.0 |
UNLINE | 断开连接关系 | 1.0 |
其中最常用的就是GET方法和POST方法.
1.4 http的状态码
类别 | 原因短语 | |
---|---|---|
1XX | Informational(信息状态码) | 接收的请求正在处理 |
2XX | Success(成功状态码) | 请求正常处理完毕 |
3XX | Redirection(重定向状态码) | 需要进行附加操作以完成请求 |
4XX | Client Error(客户端错误状态码) | 服务器无法处理请求 |
5XX | Server Error(服务器错误状态码) | 服务器处理请求出错 |
最常见的状态码, 比如 200(OK), 404(Not Found), 403(Forbidden), 302(Redirect, 重定向), 504(Bad Gateway)
1.5 http常见Header
- Content-Type: 数据类型(text/html等)
- Content-Length: Body的长度
- Host: 客户端告知服务器, 所请求的资源是在哪个主机的哪个端口上;
- User-Agent: 声明用户的操作系统和浏览器版本信息;
- referer: 当前页面是从哪个页面跳转过来的;
- location: 搭配3xx状态码使用, 告诉客户端接下来要去哪里访问;
- Cookie: 用于在客户端存储少量信息. 通常用于实现会话(session)的功能;
二、http服务器
2.1 最简单的http服务器
使用tcp协议的服务器端,在浏览器中可以连接该服务器,浏览器上默认使用http协议进行连接。在前几篇文章中已经编写了tcp套接字,这里只需要编写提供服务的代码即可。
#include <iostream>
#include <stdio.h>
#include <strings.h>
#include <unistd.h>
#include <string.h>
#include <errno.h>
#include <pthread.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <sys/types.h>
#include <sys/socket.h>
#include<sys/types.h>
#include<sys/wait.h>
#include "Log.hpp"using namespace std;
volatile bool quitSer = false;void Usage(void *vgs)
{std::cout << "Usage:./tcpserver port ip" << std::endl;
}
//提供服务
void Serverhttp(int sock)
{char buffer[10240];ssize_t s = read(sock, buffer, sizeof buffer);cout << s << endl;if(s > 0){logMessage(DEBUG, "read success..");std::cout << buffer;}
}class server
{
public:server(int port, std::string ip = ""): sockfd_(-1), ip_(ip), port_(port){}~server(){}public:void init(){// 1. 创建套接字sockfd_ = socket(AF_INET, SOCK_STREAM, 0);if (sockfd_ < 0){logMessage(FATAL, "socket:%s[%d]", strerror(errno), sockfd_);exit(SOCK_ERR);}logMessage(DEBUG, "socket success..");// 2.填充域struct sockaddr_in local;bzero(&local, sizeof(local));local.sin_family = AF_INET;local.sin_port = htons(port_);ip_.empty() ? (local.sin_addr.s_addr = INADDR_ANY) : (inet_aton(ip_.c_str(), &local.sin_addr));// 3. 绑定网络信息if (bind(sockfd_, (const struct sockaddr *)&local, sizeof(local)) < 0){logMessage(FATAL, "bind:%s[%d]", strerror(errno), sockfd_);exit(BIND_ERR);}logMessage(DEBUG, "bind success...");// 4. 监听套接字if (listen(sockfd_, 5) < 0){logMessage(FATAL, "listen:%s[%d]", strerror(errno), sockfd_);exit(LISTEN_ERR);}logMessage(DEBUG, "listen success...");// 完成}void start(){char inbuffer_[1024]; // 用来接收客户端发来的消息// 提供服务while (true){quitSer = false;struct sockaddr_in peer;socklen_t len = sizeof(peer);// 5. 获取连接, accept 的返回值是一个新的socket fdlogMessage(DEBUG,"start1");int serviceSock = accept(sockfd_, (struct sockaddr *)&peer, &len);logMessage(DEBUG,"start2");if (serviceSock < 0){// 获取链接失败logMessage(WARINING, "accept: %s[%d]", strerror(errno), serviceSock);continue;}logMessage(DEBUG, "accept success");pid_t id = fork();if(id == 0){pid_t tid = fork();if(tid == 0){uint16_t peerPort = htons(peer.sin_port);std::string peerIp = inet_ntoa(peer.sin_addr);Serverhttp(serviceSock);exit(0);}exit(0);}close(serviceSock);int ret = waitpid(id, nullptr, 0);if(ret < 0){logMessage(WARINING, "wait child error:%d", errno);}logMessage(DEBUG, "wait success...");}}private:int sockfd_;uint16_t port_;std::string ip_;
};// ./tcpserver port ip
int main(int argc, char *argv[])
{if (argc < 2 || argc > 3){Usage(argv[0]);exit(USAGE_ERR);}uint16_t port = atoi(argv[1]);std::string ip;if (argc == 3)ip = argv[2];std::cout << port << ":" << ip << std::endl; server tcpSer(port, ip);tcpSer.init();tcpSer.start();return 0;
}
服务端并没有提供服务,所以访问服务器时网页显示如下图:
在Linux服务端可以看到请求的相关属性,如下图:
2.2 简单的http服务器
服务器收到请求,提供服务向对应的套接字文件中写入一个“hello world”,注意响应的格式为首行+header+body,其中header和body之间需要空一行。
void Serverhttp(int sock)
{char buffer[10240];ssize_t s = read(sock, buffer, sizeof buffer);cout << s << endl;if(s > 0){logMessage(DEBUG, "read success..");std::cout << buffer;}//开始响应string response;response = "HTTP/1.0 200 OK\r\n"; //响应首行response += "\r\n"; //空行response += "hello world!!"; //正文send(sock, response.c_str(), response.size(), 0);
}
服务器收到的请求报文如下图:
浏览器收到的响应如下图:
使用telnet
可以查看服务器给用户的响应。
//安装telnet
sudo yum install -y telnet//使用格式
telnet ip port
//connected后
//出现escape character时使用ctrl+}
//输入请求
2.3 html的http服务器
作为服务端,它并不关心html
的信息也不需要管html的格式是什么,服务器只负责网络请求并且把网页信息给用户推过去就行了,所以实际上在进行网站开发时,网页的资源都是通过html文件推送给用户的,请求的文件在请求行中,其第二个字段就是用户需要访问的文件,如下图:
其中/
并不是根目录,而是web根目录,可以设置成为根目录。
//获取资源路径
string getPath(string in)
{size_t pos = in.find(CRLF);string request = in.substr(0, pos);cout << "request:" << request << endl;// get path http/1.0size_t first = request.find(SPACE); //第一个空格cout << "first:" << first << endl;if(first == string:: npos) return nullptr;size_t second = request.rfind(SPACE); //第二个空格cout << "second:" << second << endl;if(second == string::npos) return nullptr;string path = request.substr(first + SPACE_LEN, second- (first + SPACE_LEN));if(path.size() == 1 && path[0] == '/') path += HOME_PAGE; //用户访问根目录,就返回主页cout << "getPath:" << path << endl;return path;
}
//获取资源
string readFile(string& recource)
{ifstream in(recource);if(!in.is_open()) return "404";string content;string line;while(getline(in, line)) content += line;return content;
}//提供服务
void Serverhttp(int sock)
{//std::cout << peerIp << ":" << peerPort << std::endl;char buffer[10240];ssize_t s = read(sock, buffer, sizeof buffer);cout << s << endl;if(s > 0){logMessage(DEBUG, "read success..");std::cout << buffer;}// 1. 获取路径string path = getPath(buffer);string recource = ROOT_PATH;recource += path;cout << recource << endl;// 获取对应文件内容string html = readFile(recource);// 开始响应string response;response = "HTTP/1.0 200 OK\r\n";response += "Content-Type:text/html\r\n";response += ("Content-Length:" + std::to_string(html.size()) + "\r\n");response += "\r\n";//response += "<html><h1>hello world!!</h1><html>\r\n";response += html;send(sock, response.c_str(), response.size(), 0);
}
关于html的编写方法可以参考👉菜鸟教程
//index.html
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8"> //
<title>华丞臧(runoob.com)</title> //网页名称
</head>
<body> //正文<h1>我的第一个标题</h1> //标题<p>我的第一个段落。</p>
</body>
</html>
在浏览器中访问服务器可以看到网页显示如下图所示:
Liunx上收到的请求如下图,左为服务器收到的请求,右为客户端收到的响应。
2.4 表单
用户的网络行为无非有两种:
- 用户将自己的远端资源拿到本地:get /index.html http/1.1;
- 用户想将自己的属性字段提交到远端。
在浏览器上,网页通常会让用户注册一个账号,用来标识该用户,用户的存放在远端的资源也可以使用该账号标识,在html中这种端口称为表单。
get
在http中get会以明文方式将用户对应的参数信息拼接到url中。
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<title>华丞臧(runoob.com)</title>
</head>
<body><h1>hello world</h1><p>我的第一个段落。</p><form action="html_form_action.php" method="get">Username: <input type="text" name="user"><br>Password: <input type="password" name="passwd"><br><input type="submit" value="Submit"></form>
</body>
</html>
表单形式如下图所示:
post
在http中post会以明文方式将用户对应的参数信息拼接到正文中。
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<title>华丞臧(runoob.com)</title>
</head>
<body><h1>hello world</h1> <p>我的第一个段落。</p><form action="html_form_action.php" method="post">Username: <input type="text" name="user"><br> //用户名,<br>标识换行Password: <input type="password" name="passwd"><br> //密码<input type="submit" value="Submit"></form>
</body>
</html>
表单形式如下图所示:
在表单中填入用户名和密码后,可以看到url中并没有拼接用户参数,如下图:
使用progress telerik fiddler
可以查看用户发送的请求报文(Raw),可以看到用户参数被拼接到正文当中,如下图:
在服务端也可以收到用户发来的请求,如下图:
get和post特点
不难发现,不管是get方法还是post方法都是不安全的,都是以明文的方式传输;如果有第三方来攻击用户,无疑能非常简单的获取用户的数据;与get方法相比,post方法只是更为私密但还是明文传输,不过post方法会将参数添加到请求的正文中,因此当用户需要传输大资源的时候无疑使用post方法更加合适。
2.5 永久和临时重定向
301:永久重定向。
void Serverhttp(int sock)
{char buffer[10240];ssize_t s = read(sock, buffer, sizeof buffer);cout << s << endl;if(s > 0){logMessage(DEBUG, "read success..");std::cout << buffer;}string response = "HTTP/1.1 301 Permanent redirect\r\n"; //永久重定向response +="Location: https://blog.csdn.net/qq_59456417?spm=1011.2415.3001.5343r\n";response += "\r\n";send(sock, response.c_str(), response.size(), 0);
}
访问服务器结果如下:
302:临时重定向。
void Serverhttp(int sock)
{char buffer[10240];ssize_t s = read(sock, buffer, sizeof buffer);cout << s << endl;if(s > 0){logMessage(DEBUG, "read success..");std::cout << buffer;}string response = "HTTP/1.0 302 Temporary redirect\r\n"; //临时重定向response +="Location: https://www.qq.com/r\n";response += "\r\n";send(sock, response.c_str(), response.size(), 0);
}
访问服务器结果如下:
- 临时重定向:当客户端发起请求时,服务端对请求不进行处理,但是服务端会给可会端返回一个302状态码,并且在属性字段还会返回一个Location字段,客户端会根据这个字段进行自动跳转;如果服务器只是做一个局部性的临时的升级或者短时间不能使用,这时我们就可以进行临时重定向,如果有用户访问,就将临时重定向到对应的服务器上。。
- 永久重定向:当客户端发起请求时,服务端对请求不进行处理,但是服务端会给可会端返回一个301状态码,并且在属性字段还会返回一个Location字段,客户端会根据这个字段进行自动跳转;如果服务器永久不再使用,但是该服务器积累了一定的用户,这时我们就可以搞一个同样的服务器进行永久重定向,如果有用户访问,就重定向到对应的服务器上。
- 临时和永久重定向的区别在于客户端浏览器的处理,对于临时重定向浏览器不会记录重定向的目标位置,对于永久重定向浏览器会记录重定向的目标位置,所以永久重定向访问一次后再次访问时浏览器就会直接访问记录的服务器,注意并不是所有浏览器都会去做将永久重定向的网址更新。
- 服务器的资源是否愿意被临时访问还是愿意永久访问,决定是临时重定向还是永久重定向。
2.6 Cookie
http状态特点之一:无状态,用户各种资源的请求http协议不会记录,也就是说http不知道用户访问服务器次数;虽然http不会记录,但http可以借助其他的手段来记录用户的访问。
为什么需要记录呢?
因为用户需要会话保持,网站中某些资源是需要进行用户认证的,如果访问不被记录,就会导致每次访问资源都需要进行用户认证,比如在腾讯视频上每次访问vip影视都需要用户进行一次登录这显然是非常麻烦的的,所以服务器需要保持用户会话。
我们可以登录一下bilibili如下所示,点击网址左边的锁,可以看到有一个Cookie,Cookie是一种浏览器会话保持的策略。
在Cookie类表中可以找到以bilibili开头的:
当你把Cookie中关于bilibili的全部删除,下一次再进入bilibili时就需要重新登陆了。
Cookie内容:
- 用户名
- 密码
- 到期时间
- 地址
- 等等
void Serverhttp(int sock)
{char buffer[10240];ssize_t s = read(sock, buffer, sizeof buffer);cout << s << endl;if(s > 0){logMessage(DEBUG, "read success..");std::cout << buffer;}// 1. 获取路径string path = getPath(buffer);string recource = ROOT_PATH;recource += path;cout << recource << endl;string html = readFile(recource);// 开始响应string response;response = "HTTP/1.0 200 OK\r\n";response += "Content-Type:text/html\r\n";response += ("Content-Length:" + std::to_string(html.size()) + "\r\n");response += "Set-Cookie: this is my Cookie content\r\n";response += "\r\n";response += html;send(sock, response.c_str(), response.size(), 0);
}
Cookie就是浏览器给用户维护的一个文件,Cookie可以在磁盘上也可以在内存中,内存中的Cookie将浏览器关闭后Cookie就会释放,再次启动浏览器访问服务器时就需要重新登录。Cookie对用户的安全极大的影响,第三方如果拿到用户的Cookie,第三方就可以通过Cookie使用用户的身份登录服务器,更严重的第三方还是以通过Cookie拿到用户的用户名和密码。
Cookie策略存在极大的安全隐患,所以现在主流的方案不是Cookie,而是Cookie + session的方式。服务器将用户名和密码形成一个session文件,这个文件的文件名session_id在服务器上具有唯一性,服务器将session_id返回客户端,客户端将session_id写入本地的Cookie,后续访问都会通过session_id去访问服务器,此时用户的用户名和密码就是存放在服务器上,软件厂商来维护session和cookie,大公司有充足的动力和能力来保护自己不受攻击,并且还会有各种措施来防护。
补充:
- http-pipeline:怎么请求的,就按顺序来响应。
相关文章:
【网络】http协议
🥁作者: 华丞臧. 📕专栏:【网络】 各位读者老爷如果觉得博主写的不错,请诸位多多支持(点赞收藏关注)。如果有错误的地方,欢迎在评论区指出。 推荐一款刷题网站 👉 LeetCode刷题网站 文章…...
Thread::interrupted() 什么意思? 如何中断线程?
1、答: Thread::interrupted() 是一个静态方法,用于判断当前线程是否被中断,并清除中断标志位。 具体来说,当一个线程被中断后,它的中断状态将被设置为 true。如果在接下来的某个时间点内调用了该线程的 interrupted…...
Oracle OCP 19c 考试(1Z0-083)中关于Oracle不完全恢复的考点(文末附录像)
欢迎试看博主的专著《MySQL 8.0运维与优化》 下面是Oracle 19c OCP考试(1Z0-083)中关于Oracle不完全恢复的题目: A database is configured in ARCHIVELOG mode A full RMAN backup exists but no control file backup to trace has been taken A media…...
一起来学习配置Combo接口吧!
Combo接口是一个光电复用的逻辑接口,一个Combo接口对应设备面板上一个GE电接口和一个GE光接口。电接口与其对应的光接口是光电复用关系,两者不能同时工作(当激活其中一个接口时,另一个接口就自动处于禁用状态)…...
C++模拟实现红黑树
目录 介绍----什么是红黑树 甲鱼的臀部----规定 分析思考 绘图解析代码实现 节点部分 插入部分分步解析 ●父亲在祖父的左,叔叔在祖父的右: ●父亲在祖父的右,叔叔在祖父的左: 测试部分 整体代码 介绍----什么是红黑树 红…...
HTTPS协议之SSL/TLS详解(下)
目录 前言: SSL/TLS详解 HTTP协议传输安全性分析 对称加密 非对称加密 证书 小结: 前言: 在网络世界中,存在着运营商劫持和一些黑客的攻击。如果明文传输数据是很危险的操作,因为我们不清楚中间传输过程中就被哪…...
OLE对象是什么?为什么要在CAD图形中插入OLE对象?
OLE对象是什么?OLE对象的意思是指对象连接与嵌入。那为什么要在CAD图形中插入OLE对象?一般情况下,在CAD图形中插入OLE对象,是为了将不同应用程序的数据合并到一个文档中。本节内容小编就来给大家分享一下在CAD图形中插入OLE对象的…...
【微信小程序】-- 自定义组件 -- 数据、方法和属性(三十三)
💌 所属专栏:【微信小程序开发教程】 😀 作 者:我是夜阑的狗🐶 🚀 个人简介:一个正在努力学技术的CV工程师,专注基础和实战分享 ,欢迎咨询! &…...
【Spring 深入学习】AOP的前世今生之代理模式
AOP的前世今生之代理模式1. 概述 什么是代理模式呢??? 在不修改原有代码 或是 无法修改原有代码的情况下,增强对象功能,替代原来的对象去完成功能,从而达成了拓展的目的。 先给大家看下 JavaScript中实现方…...
操作系统复试
2017软学 给出操作系统的定义,分别从资源管理,任务调度,用户接口等三个方面论述操作系统的职能 操作系统是位于硬件层之上、所有其他系统软件层之下的一个系统软件,使得管理系统中的各种软件和硬件资源得以充分利用,方…...
藏经阁(五)温湿度传感器 SHT3x-DIS 手册 解析
文章目录芯片特性芯片内部框图芯片引脚定义芯片温湿度范围芯片寄存器以及时序讲解信号转换公式芯片特性 湿度和温度传感器完全校准,线性化温度补偿数字输出供电电压范围宽,从2.4 V到5.5 VI2C接口通讯速度可达1MHz和两个用户可选地址典型精度 2% RH和 0.…...
PCB焊盘设计基本原则
SMT的组装质量与PCB焊盘设计有直接的关系,焊盘的大小比例十分重要。如果PCB焊盘设计正确,贴装时少量的歪斜可以再次回流焊纠正(称为自定位或自校正效应),相反,如果PCB焊盘设计不正确,即使贴装位置十分准确,…...
mysql锁分类大全
前言 为什么会出现锁 MySQL中的锁是为了保证并发操作的正确性和一致性而存在的。 当多个用户同时对同一份数据进行操作时,如果不加控制地进行读写操作,就可能导致数据不一致的问题。例如,当多个用户同时对同一行数据进行写操作时ÿ…...
推荐几款主流好用的远程终端连接管理软件
一、介绍 远程终端连接管理软件是管理服务器、虚拟机等远程计算机系统不可或缺的工具之一,它可以通过网络连接到另一台计算机,以执行命令、编辑文件或进行其他管理任务,下面我将为大家介绍几款主流好用的远程终端连接管理软件,并…...
描述性统计
参考文献 威廉 M 门登霍尔 《统计学》 文章目录定性数据的描述方法条形图饼图帕累托图定量数据点图茎叶图频数分布直方图MINITAB 工具在威廉《统计学》一书将统计学分为描述统计学和推断统计学,他们的定义分别如下:描述统计学:致力于数据集的…...
第十四届蓝桥杯三月真题刷题训练——第 7 天
目录 第 1 题:三角回文数 问题描述 答案提交 运行限制 代码: 第 2 题:数数 问题描述 答案提交 运行限制 代码: 第 3 题:倍数问题_同余定理_分情况讨论 题目描述 输入描述 输出描述 输入输出样例 运行限…...
剑指 Offer 57. 和为s的两个数字
一、题目 输入一个递增排序的数组和一个数字s,在数组中查找两个数,使得它们的和正好是s。如果有多对数字的和等于s,则输出任意一对即可。 示例 1: 输入:nums [2,7,11,15], target 9 输出:[2,7] 或者 [7…...
PDF转word在线转换方法!操作简单又高效
相信很多已经工作的人都知道,PDF文件格式的优点在于兼容性强、安全性高,而且查看和传输给他人都很方便。但是,这种格式的文件也有不太方便的地方,那就是不能对文件内容进行编辑和修改。对于许多人来说,如果想要编辑修改…...
Jquery项目中使用vue.js
大家在工作的情况中,可能会遇到之前的老项目采用jq书写,或者修改或者新增功能在jq中,原始jq的项目,代码可维护性很差,一个页面几千行jq,可维护性很差,工作量巨大,所以这个时候大家可以引入vue.js。 第一步:引入vue.js…...
蓝桥杯 删除字符
题目描述 给定一个单词,请问在单词中删除 t 个字母后,能得到的字典序最小的单词是什么? 输入描述 输入的第一行包含一个单词,由大写英文字母组成。 第二行包含一个正整数 t。 其中,单词长度不超过 100,…...
析构函数 对象数组 对象指针
🐶博主主页:ᰔᩚ. 一怀明月ꦿ ❤️🔥专栏系列:线性代数,C初学者入门训练,题解C,C的使用文章 🔥座右铭:“不要等到什么都没有了,才下定决心去做” …...
Vue对Axios网络请求进行封装
一、为什么要对网络请求进行封装? 因为网络请求的使用率实在是太高了,我们有的时候为了程序的一个可维护性,会把同样的东西放在一起,后期找起来会很方便,这就是封装的主要意义。 二、如何进行封装? 1、将…...
Android framework HAL(HIDL)
简述 当你在Android系统中使用不同的硬件设备(例如摄像头、传感器、音频设备等)时,你需要与硬件抽象层(HAL)进行通信。 HAL是一个中间层,它充当了硬件和应用程序之间的桥梁。但是,由于硬件设备…...
QML 模型(ListModel)
LIstModel(列表模型) ListModel 是ListElement定义的简单容器,每个定义都包含数据角色。内容可以在 QML 中动态定义或显式定义。 属性: count模型中数据条目的数量dynamic动态角色,默认情况下,角色的类型…...
你还在调戏AI,有的公司已经用ChatGPT开展业务了
近日,OpenAI 正式宣布开放 ChatGPT 和 Whisper 两个模型的 API,API 版本的ChatGPT 不仅功能更多、性能更强,而且还更便宜一一相当于目前 GPT-3 模型价格打一折!划重点OpenAl正式开放 ChatGPT 和 Whisper 模型的 API,目前 SnapChat…...
DatenLord前沿技术分享 No.20
达坦科技专注于打造新一代开源跨云存储平台DatenLord,致力于解决多云架构、多数据中心场景下异构存储、数据统一管理需求等问题,以满足不同行业客户对海量数据跨云、跨数据中心高性能访问的需求。喷泉码具有极高的纠错能力,且具有低延迟、地复…...
基于vivado(语言Verilog)的FPGA学习(1)——了解viviado面板和编译过程
基于vivado(语言Verilog)的FPGA学习(1)——了解程序面板和编译过程 每日废话:最近找实习略微一些焦虑,不想找软件开发,虽然有些C和python基础(之前上课学的),…...
PACS(CT、CR、DR、MR、DSA、RF医院影像管理系统源码)
PACS具体功能介绍: 病人、采集、观片、三维、报告、照相、退出、文件、图像采集、观片操作、三维、测量标注、诊断报告、照相打印、统计报表、系统管理、帮助、病人浏览器、选择数据源、打开图像、病人登记、工作列表、采集、打开画廊。 DICOM查询/获取:…...
Centos7 安装Mysql8.0
1、到指定目录下下载安装包[rootVM-0-14-centos ~]# cd /usr/local/src2、下载mysql8[rootVM-0-14-centos src]# wget https://dev.mysql.com/get/Downloads/MySQL-8.0/mysql-8.0.20-linux-glibc2.12-x86_64.tar.xz3、解压mysql8, 通过xz命令解压出tar包, 然后通过t…...
2023年全国最新道路运输从业人员精选真题及答案18
百分百题库提供道路运输安全员考试试题、道路运输从业人员考试预测题、道路安全员考试真题、道路运输从业人员证考试题库等,提供在线做题刷题,在线模拟考试,助你考试轻松过关。 181.某客运企业拥有55辆营运客车,下列关于该企业设置…...
顶级策划公司/上海怎么做seo推广
SQL Server 计划作业 2018-09-10 11:10:03 ClearLoveQ 阅读数 771更多 分类专栏: MS SQL Server MS 版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。 本文链接:https:/…...
全响应式网站用什么做的/百度快速收录工具
JDK,JRE,JVM 今天我们讨论下这三个Java工具 JDK 全称Java Development ToolKit(Java 开发工具包)。 JDK是整个JAVA的核心,其包括了Java运行环境(Java Runtime Envirnment),一堆Java工具(javac/java/jdb等)和Java基础的类库(即Java…...
公司建站花费/百度小程序入口
原文链接译者:smallclover个人翻译,水平有限,如有错误欢迎指出,谢谢!设计模式-概述 设计模式体现了经验丰富的面向对象软件开发人员的最佳实践。设计模式是软件开发人员在整个软件开发的过程中面临普遍问题的解决方案。…...
网站搭建上海/百度搜索关键词排名人工优化
AutoRec AutoRec可以说是深度学习时代最古老、最简单、最容易理解的模型。该模型模拟无监督学习中用到的数据表征和压缩工具AutoEncoder, 自然地,其模型也可以写成重建函数形式: 其中 f(*) 和 g(*) 分别为输出层和隐藏层的激活函数。 跟AutoEncoder类似…...
企业怎么做网站/百度广告电话号码是多少
本文介绍如何通过信号实现子进程按序(同步)向管道中写入数据。要求: 编写程序实现进程的管道通信。用系统调用pipe()建立一管道,二个子进程P1和P2分别向管道各写一句话:Child 1 is sending a message!Child 2 is sending a message!父进程从管道中读出二个来自子进程…...