HTTP协议初识·中篇

这篇具有很好参考价值的文章主要介绍了HTTP协议初识·中篇。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

 

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

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

本篇主要讲解了:

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

        html链接跳转

        网页添加图片

        确认并返回资源类型

        填写正文长度属性

        添加表单

        临时重定向

补充知识:

        前后端区别

        浏览器是如何显示页面的

        http各字段属性是什么意思

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

        http方法

        http状态码

        HTTP常见Header

        写入cookie信息

源码

承接上文

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

HTTP协议初识·中篇,http,网络协议,网络

第二点在上文中已经完成 

1:网页分离

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

HTTP协议初识·中篇,http,网络协议,网络

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

读取文件中的内容

 HTTP协议初识·中篇,http,网络协议,网络

HTTP协议初识·中篇,http,网络协议,网络

404html文件

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

HTTP协议初识·中篇,http,网络协议,网络

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

成功

 HTTP协议初识·中篇,http,网络协议,网络

404访问

HTTP协议初识·中篇,http,网络协议,网络

2:html链接跳转

2.0添加点击键与返回键

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

HTTP协议初识·中篇,http,网络协议,网络

HTTP协议初识·中篇,http,网络协议,网络

测试2:返回键与404测试

返回键

HTTP协议初识·中篇,http,网络协议,网络

HTTP协议初识·中篇,http,网络协议,网络

404(返回时候不存在)

故意写错

HTTP协议初识·中篇,http,网络协议,网络

HTTP协议初识·中篇,http,网络协议,网络

别人的404网页

高级一点…… 

HTTP协议初识·中篇,http,网络协议,网络

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

HTTP协议初识·中篇,http,网络协议,网络

3:添加图片

测试3:图片测试

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

HTTP协议初识·中篇,http,网络协议,网络

HTTP协议初识·中篇,http,网络协议,网络

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

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

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

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

HTTP协议初识·中篇,http,网络协议,网络

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

HTTP协议初识·中篇,http,网络协议,网络

suffixToDesc函数4.1后缀分离函数

HTTP协议初识·中篇,http,网络协议,网络

测试4:图片再测试

HTTP协议初识·中篇,http,网络协议,网络

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

HTTP协议初识·中篇,http,网络协议,网络

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

HTTP协议初识·中篇,http,网络协议,网络

5:添加正文长度

认识系统调用stat

HTTP协议初识·中篇,http,网络协议,网络

HTTP协议初识·中篇,http,网络协议,网络

测试5:添加正文长度

HTTP协议初识·中篇,http,网络协议,网络

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

HTTP协议初识·中篇,http,网络协议,网络

6:添加表单

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

HTTP协议初识·中篇,http,网络协议,网络

HTTP协议初识·中篇,http,网络协议,网络

补充6.0:表单各部分意思

HTTP协议初识·中篇,http,网络协议,网络

测试6:表单测试

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

 HTTP协议初识·中篇,http,网络协议,网络

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

HTTP协议初识·中篇,http,网络协议,网络

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

HTTP协议初识·中篇,http,网络协议,网络

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

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

HTTP协议初识·中篇,http,网络协议,网络

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

HTTP协议初识·中篇,http,网络协议,网络

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

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

HTTP协议初识·中篇,http,网络协议,网络

补充5:HTTP的方法

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

所以GET和POST相比

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

如果你想要提参:GET/POST

无私密性:GET(简单)

私密性:POST

HTTP协议初识·中篇,http,网络协议,网络

补充6:HTTP的状态码

HTTP协议初识·中篇,http,网络协议,网络

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

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

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

HTTP协议初识·中篇,http,网络协议,网络

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

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

HTTP协议初识·中篇,http,网络协议,网络

7:临时重定向

HTTP常见Header

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

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

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

HTTP协议初识·中篇,http,网络协议,网络

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

修正bug代码 

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

HTTP协议初识·中篇,http,网络协议,网络

HTTP协议初识·中篇,http,网络协议,网络

        不过需要注意的是,这样会导致当资源不存在的时候,跳转的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脚本
    {
         首先建立进程间通信, pipe
         fork创建子进程, execl("/bin/puthon", test.py) 这样子进程就执行这个脚本
         父进程, 将req.parm 通过管道写过某些后端语言, C++并不适合写这个, 一般是指, python, java, php等
         这也是为什么服务器是C++写的,但是它的后端却是其他语言的原因
    }
    if(req.path == "/search")
    {
         req.parm
         使用我们自己写到C++方法,提供服务
    }
    */

    // for test
    cout << "----------------------- 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); //  这个操作一定能成功
    }

    /*
    测试 -- 这个代码会导致本地图片显示不出来,因为视频图片都是二进制的东西不能按行读取!要用read
    if (!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); // 第三个参数默认 0
            if (_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. send
            char 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 进行通信的fd
                struct 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 2
                waitpid(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\n
        std::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; // 未来可以进行修改  变成 ./wwwroot
        path += 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/宇宙.jpg
        auto pos = path.rfind(".");
        if (pos == std::string::npos)
            suffix = ".html";
        else
            suffix = path.substr(pos); // 截取到末尾

        // 5. 得到资源的大小
        struct stat st;
        int n = stat(path.c_str(), &st);// 这个系统调用会自动填充字段
        if(n == 0) size = st.st_size;   // 成功返回 0
        else 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 found
        
        in.read(buffer, size);  
        in.close();
        return true;
    }
    

    /*
    测试 -- 这个代码会导致本地图片显示不出来,因为视频图片都是二进制的东西不能按行读取!要用read
    static bool readFile(const std::string resource, std::string *out) // 视频图片都是二进制的东西不能按行读取!
    {               
        std::ifstream in(resource, std::ios::binary);  // 按照二进制的方式读取文件内容

        if (!in.is_open()) return false; // resource not found
            

        std::string line;
        while (std::getline(in, line))
        {
            *out += line;
        }

        in.close();
        return true;
    */
};

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

HTTP协议初识·中篇,http,网络协议,网络

未完持续更新中……文章来源地址https://www.toymoban.com/news/detail-693982.html

到了这里,关于HTTP协议初识·中篇的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 网络协议 - HTTP 协议详解

    HTTP 协议详解,web开发必备。 URI 包含 URL 和 URN,目前 WEB 只有 URL 比较流行,所以见到的基本都是 URL。 URI(Uniform Resource Identifier,统一资源标识符) URL(Uniform Resource Locator,统一资源定位符) URN(Uniform Resource Name,统一资源名称) 1. 请求报文 2. 响应报文 客户端发送的 请求报文 第

    2024年02月04日
    浏览(57)
  • 【网络协议】聊聊http协议

    当我们输入www.baidu.com的时候,其实是先将baidu.com的域名进行DNS解析,转换成对应的ip地址,然后开始进行基于TCP构建三次握手的连接,目前使用的是1.1 默认是开启了keep-Alive。可以在多次请求中进行连接复用。 连接建立之后,就要发送HTTP的请求了, 整体其实就是三部分,请

    2024年02月06日
    浏览(61)
  • 网络协议与攻击模拟-21-HTTP协议

    1、 HTTP 协议结构 2、在 Windows server 去搭建 web 服务器 3、分析 HTTP 协议流量 1、概念 HTTP (超文本传输协议)是用于在万维网服务器上传输超文本( HTML )到本地浏览器的传输协议 属于 TCP / IP 协议簇的一员( HTML 文件、图片、查询结构等) 基于传输层 TCP 的80端口 2、万维网服

    2024年02月16日
    浏览(46)
  • 网络通信:http协议

    虽然我们说, 应用层协议是我们程序猿自己定的. 但实际上, 已经有大佬们定义了一些现成的, 又非常好用的应用层协议, 供我们直接参考使用. HTTP(超文本传输协议) 就是其中之一. 统一资源定位符(Uniform Resource Locator,缩写:URL),是对资源的引用和访问该资源的方法。俗称网址

    2024年02月07日
    浏览(64)
  • 【网络】HTTP协议详解

    😀大家好,我是 白晨 ,一个不是很能熬夜😫,但是也想日更的人✈。 如果喜欢这篇文章, 点个赞 👍, 关注一下 👀白晨吧!你的支持就是我最大的动力!💪💪💪 哟,大家好,我是白晨。距离上一次更新已经过了一段时间了,属实是当鸽子当惯了🤣。 上一篇文章我们

    2024年02月02日
    浏览(48)
  • [网络原理] HTTP协议

    要珍惜时间呀 HTTP协议是应用层协议,TCP/IP协议为传输层协议,负责传输数据.而HTTP协议相当于对传输的数据据怎样处理和使用进行说明. 每次,我们访问网站的时候,都会给网站的服务器发送一个HTTP请求,服务器收到请求后,会返回一个HTTP响应.如下图所示. 下图,为一个HTTP请求的格式

    2024年02月03日
    浏览(41)
  • 【计算机网络】HTTP 协议

    前面我们介绍了网络 TCP/IP 五层模型中的各个层,在这五层中,应用层是和我们程序员息息相关的,需要我们程序员写出代码来实现,前面我们只是简单讲了应用层中的自定义协议,虽然自定义协议显得很灵活可以根据需求随时更改,但是在实际生活中自定义的协议使用的还

    2024年02月05日
    浏览(83)
  • 【网络】应用层——HTTP协议

    🐱作者:一只大喵咪1201 🐱专栏:《网络》 🔥格言: 你只管努力,剩下的交给时间! 上篇文章中,本喵带着大家对HTTP有了一个初步的认识,今天就来详细讲解一下这个应用层协议。 如上图所示的 url (网址),里面包含有 / 以及 ? 等字符。 像这样的字符,已经被url当做 特殊

    2024年02月15日
    浏览(43)
  • 网络原理(六):http 协议(上)

    目录 HTTP 协议是什么 抓包工具 Fiddler 的下载 使用Fiddler HTTP 请求 (Request)   HTTP 请求格式 首行 请求头(Header) Cookie 还是老样子,在讲解http 之前我们先来了解以下什么叫做 http 。 HTTP(Hyper Text Transfer Protocol):超文本传输协议。这是一个应用非常广的协议,我们随便打开一

    2024年02月07日
    浏览(40)
  • 计算机网络【HTTP协议】

    HTTP/HTTPS协议 是应用层的网路协议 目前大多数情况HTTP在传输层是基于TCP(HTTP1/2 是基于TCP,最新的HTTP协议是基于UDP协议,但是我们目前常用的HTTP应用层协议是HTTP1.0) 应用层协议很多时候都是程序员自己定制的,需要根据具体的场景来制定应用层协议,但是由于程序员水平参差不

    2024年02月02日
    浏览(44)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包