HTTP详解及代码实现

HTTP详解及代码实现

  • HTTP
    • 超文本
    • 传输
    • 协议
  • URL简述
  • 状态码
    • 常见的状态码
  • 请求方法
  • 请求报文
  • 响应报文
  • HTTP常见的Header
  • HTTP服务器代码

HTTP

  HTTP的也称为超文本传输协议。解释HTTP我们可以将其分为三个部分来解释:超文本,传输,协议。

超文本

  加粗样式超文本从字面意思就是超越了文本。在计算机发展初期,计算机只能传输使用文本数据,随着技术发展,出现了图片,视频,音频,超链接等,这些数据就可以统称为超文本

传输

  传输这个词从字面上很好理解,没有什么需要介绍的,事实也是如此。http中的传输意味着数据需要由发送方发送到接收方,这是废话,但是真正需要我们注意的是,HTTP是在两点之间传输的
  那可以实现多点之间传输吗?HTTP是基于TCP的,TCP是只能两点之间传输数据的。所以HTTP也只能是两点之间传输的。
  但是HTTP3.0是基于UDP的啊,可以多点传输吗?对于HTTP/3.0(基于QUIC),虽然它是基于UDP的,但它主要是为了解决传统TCP在性能上的一些限制而设计的,并不是为了直接支持多点传输。

协议

  协议这个词在我们的印象里面就是规则,在计算机世界里面,就是用来约束某些行为的规则。它是用计算机能够理解的语言,确立的一种计算机之间进行交流通信的一种规范,以及相关的控制和错误处理方式。


  所以HTTP是什么呢? 总结一下,就是在计算机世界里面,用来在两点之间传输图片,音频,视频,超链接等超文本数据的一种约定和规范。

URL简述

URL也叫做统一资源定位器,用来唯一定位全网中的资源。可以认为有了它就能够没有歧义的直接在网络中找到相应的资源。

https://editor.csdn.net/md?not_checkout=1&spm=1010.2135.3001.5352&articleId=137425440

上面这一串就是我当前写这篇博客的URL(当然你没有账号密码访问不了)。

  • https:// :这是协议名称,有http和https可选。
  • editor.csdn.net/ : 这是资源所在的服务器地址,也就是域名。
  • md : 这是资源的路径名。
  • ? : 这个是分割开资源路径名和查询参数的分隔符,**#**也有分隔的作用,不过上面URL中没有体现。
  • spm=1010.2135.3001.5352&articleId=137425440 : 这个是查询参数,用&分割开来,这些参数由客户端给服务端,用来指定资源。
  • # : 上面URL中没有体现这个。#后面是片段标识符id, 浏览器在加载该URL时会滚动到页面中对应的片段位置。

状态码

  HTTP的状态码是存在于响应报文中的。
它有以下五类:

  • 1** : 表示请求正在被处理,是一种中间状态。
  • 2** :表示请求成功被处理。
  • 3** :重定向状态码,这里的重定向指的是当前请求资源的位置发生变动,需要使用新的URL去请求新的资源。
  • 4** : 客户端的请求报文有误,服务端无法处理请求。
  • 5** : 服务端在处理请求时内部出现的问题导致无法正常处理请求。

常见的状态码

  • 200:请求成功,响应的头部会有body数据。
  • 204:请求成功,响应的头部没有body数据。
  • 301:永久重定向,请求的资源已经不存在了,需要使用新的URL请求资源。
  • 302:临时重定向,请求的资源还在,但是需要暂时的访问新的URL。
  • 304:这个是服务端告诉客户端,资源没有修改,浏览器本地缓存的资源依旧可以使用,重定向到本地资源。
  • 400:笼统的表示客户端发送的报文有误。
  • 403:服务器禁止访问,客户端并没有错误。
  • 404:在服务端中没有找到相应的资源或是资源不存在。
  • 500:笼统的表示服务端发生了错误。
  • 501:客户端请求的功能还不支持。
  • 502:通常是服务器作为网关或者代理的时候返回的状态码,表示自己没有问题,但是发送给后端服务器时发生了错误。
  • 503:服务器繁忙,暂时无法响应客户端。

请求方法

  出现在请求报文中,用来说明使用什么样的方法请求资源。
在这里插入图片描述

请求报文

  这是HTTP请求资源时发送的报文。
在这里插入图片描述
在这里插入图片描述
  首行包括:请求方法 + URL + HTTP版本号
在这里插入图片描述
  首行下面的就是请求头部,使用key:value的格式,简单且利于理解,每一对key:value占据一行,行末使用\r\n回车换行来分割每一对key:value。
  头部末尾空出一行来将头部和body分割开来。

在这里插入图片描述
  这是Body,可以发送一些数据来指定资源或者对指定资源做出处理,允许为空,若不为空,头部会使用Content-Length来表示Body长度。
  大家或许发现了Body中的数据和URL中?后面的参数很像,这并不是偶然。他们两个的作用都是为了获取指定资源或是对指定的资源做出处理,如果请求方法为POST,则参数在Body中,如果请求方法为GET,则参数在URL中,具体细节本文不做展开解释。当然Body中不仅仅局限于传递参数的作用。

响应报文

  响应报文是服务端用来对客户端请求的响应的。
在这里插入图片描述

  • 首行:版本号 + 状态码 + 状态码解释
  • 响应头部:跟请求头部一样使用key:value的格式。
  • Body:如果有Body,则Body和头部之间会有一个空行(\r\n),上图中的Body里面就是一个html代码,向客户端返回了一个html页面。

HTTP常见的Header

  • Host:请求报文中。用于指明请求的服务器的域名。
  • Connection:请求和响应报文中都有。用于说明是否使用长连接,如果使用,则设为 keep-alive。如果不使用长连接,也就是使用短连接,则设为:clone
  • Content-Type:请求报文和响应报文都有。其中的value是Body数据的长度。Header和Body的区分是通过一个空行实现的,有了Content-Type字段,也就可以确定了Body的界限了。
  • Content-Type:响应报文中。用来告诉客户端返回的数据是什么格式的。
  • Accept-Type:请求报文中,用来告诉服务端自己能够接受什么类型的。
  • Content-Encoding:响应报文。用来告诉服务器返回的数据使用了什么压缩格式。
  • Accept-Encoding:请求报文:用来告诉服务端自己能够接受什么压缩格式。
  • Cookie: 请求报文和响应报文都有。用于存储客户端的少量信息. 通常用于实现会话(session)的功能;
  • Location:响应报文中。当状态码为3**时,说明为重定向,Location字段就是指明了重定向的位置,可以为URL外部资源,也可以指向本地资源(比如304)。
  • Referer: 当前页面是从哪个页面跳转过来的;

HTTP服务器代码

// http.cpp
#include <iostream>
#include <cstring>
#include <cerrno>
#include <unistd.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <jsoncpp/json/json.h>
#include <fstream>
#include <stdio.h>
#include <string>
#include <fstream>#include <sys/socket.h>
#include <sys/types.h>
#include <arpa/inet.h>
#include <netinet/in.h>#define ROOT "./wwwroot"
#define HOMEPAGE "a.html"#include "Util.hpp"int main(int argc, char *argv[])
{if (argc != 3){std::cout << "You should " << argv[0] << " ip port\n";return 1;}int sock = socket(AF_INET, SOCK_STREAM, 0);if (sock < 0){std::cout << "Fail to sock: " << strerror(errno) << std::endl;return 2;}int opt = 1;struct sockaddr_in local;local.sin_family = AF_INET;local.sin_port = htons(atoi(argv[2]));local.sin_addr.s_addr = inet_addr(argv[1]);int bind_flag = bind(sock, (struct sockaddr *)&local, sizeof(local));if (bind_flag < 0{std::cout << "Fail to bind: " << strerror(errno) << std::endl;return 3;}int listen_flag = listen(sock, 10);if (listen_flag < 0){std::cout << "Fail to listen: " << strerror(errno) << std::endl;return 4;}// Start Serverwhile (true){struct sockaddr_in client_addr;socklen_t client_len = sizeof(client_addr);int client_sock = accept(sock, (struct sockaddr *)&client_addr, &client_len);if (client_sock < 0){std::cout << "Fail to accept: " << strerror(errno) << std::endl;continue;}// char input_buffer[1024 * 10] = {0};// ssize_t input_size = read(client_sock, input_buffer, sizeof(input_buffer) - 1);// if (input_size < 0)// {//     std::cout << "Fail to read: " << strerror(errno) << std::endl;//     close(client_sock);//     continue;// }// char Server_buffer[1024 * 10] = {0};// std::cout << "[client request] " << input_buffer << std::endl;// Json::Value root;// int fd = open("../a.html", O_RDONLY);// const char *buf[1024 * 10];// read(fd, buf, sizeof(buf));// root["a"] = buf;// //Json::StyledWriter writer; // 这样会竖式的表示出来,便于调试// Json::FastWriter writer; // 这样会横式的表示出来// std::string s = writer.write(root);// const char *hello = "<br><br><br><br><br><h1>          你好你好!!我是汤环!!!</h1>";// sprintf(Server_buffer, "HTTP/1.0 200 OK\nConnection: keep-alive\nContent-Length: %lu\n\n%s", sizeof(hello), hello);char buffer[10240];ssize_t s = recv(client_sock, buffer, sizeof(buffer) - 1, 0);if (s > 0){buffer[s] = 0;std::cout << buffer << "--------------------\n"<< std::endl;}std::vector<std::string> vLine;Util::cutString(buffer, "\r\n", &vLine);std::vector<std::string> vBlock;Util::cutString(vLine[0], " ", &vBlock);std::string file = vBlock[1]; // 得到客户端请求中的文件路径std::string target = ROOT;target += file;std::string content;std::ifstream in(target);if (!in.is_open()){}else{std::string line;while (getline(in, line)){content += line;}in.close();Z}std::string HttpResponse;if (content.empty())HttpResponse = "HTTP/1.1 404 NOT FOUND\r\n";elseHttpResponse = "HTTP/1.1 200 OK\r\n";HttpResponse += "\r\n";HttpResponse += content;send(client_sock, HttpResponse.c_str(), HttpResponse.size(), 0);// write(client_sock, HttpResponse.c_str(), HttpResponse.size());close(client_sock);}return 0;
}
// Util.hpp#pragma once#include <iostream>
#include <vector>class Util
{
public:static void cutString(const std::string &s, const std::string &sep, std::vector<std::string> *out){std::size_t start = 0;while (start < s.size()){auto pos = s.find(sep, start);if (pos == std::string::npos)break;out->push_back(s.substr(start, pos - start));start += (pos - start);start += sep.size();}if (start < s.size())out->push_back(s.substr(start));}
};
// makefilehttp:http.cppg++ -o $@ $^ -std=c++11 -ljsoncpp.PHANY:clean
clean:rm -f http

     😄 创作不易,你的点赞和关注都是对我莫大的鼓励,再次感谢您的观看😄

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

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

相关文章

ObjectiveC-08-OOP面向对象程序设计-类的分离与组合

本节用一简短的文章来说下是ObjectiveC中的类。类其实是OOP中的一个概念&#xff0c;概念上简单来讲类是它是一组关系密切属性的集合&#xff0c;所谓的关系就是对现实事物的抽象。 上面提到的关系包括很多种&#xff0c;比如has a&#xff0c; is a&#xff0c;has some等&…

jenkins+docker实现可持续自动化部署springboot项目

目录 一、前言 二、微服务带来的挑战 2.1 微服务有哪些问题 2.2 微服务给运维带来的挑战 三、可持续集成与交付概述 3.1 可持续集成与交付概念 3.1.1 持续集成 3.1.2 持续交付 3.1.3 可持续集成与交付核心理念 3.2 可持续集成优点 3.3 微服务为什么需要可持续集成 四…

【JVM】如何定位、解决内存泄漏和溢出

目录 1.概述 2.堆溢出、内存泄定位及解决办法 2.1.示例代码 2.2.抓堆快照 2.3.分析堆快照 1.概述 常见的几种JVM内存溢出的场景如下&#xff1a; Java堆溢出&#xff1a; 错误信息: java.lang.OutOfMemoryError: Java heap space 原因&#xff1a;Java对象实例在运行时持…

AI结合机器人的入门级仿真环境有哪些?

由于使用真实的机器人开发和测试应用程序既昂贵又费时&#xff0c;因此仿真已成为机器人应用程序开发中越来越重要的部分。在部署到机器人之前在仿真中验证应用程序可以通过尽早发现潜在问题来缩短迭代时间。通过模拟&#xff0c;还可以更轻松地测试在现实世界中可能过于危险的…

【每日刷题】Day3

【每日刷题】Day3 &#x1f955;个人主页&#xff1a;开敲&#x1f349; &#x1f525;所属专栏&#xff1a;每日刷题&#x1f34d; 目录 1. 69. x 的平方根 - 力扣&#xff08;LeetCode&#xff09; 2. 70. 爬楼梯 - 力扣&#xff08;LeetCode&#xff09; 3. 118. 杨辉三…

ZYNQ学习Linux 基础外设的使用

基本都是摘抄正点原子的文章&#xff1a;《领航者 ZYNQ 之嵌入式Linux 开发指南 V3.2.pdf》&#xff0c;因初次学习&#xff0c;仅作学习摘录之用&#xff0c;有不懂之处后续会继续更新~ 工程的创建参考&#xff1a;《ZYNQ学习之Petalinux 设计流程实战》 一、GPIO 之 LED 的使…

docker安装jenkins 2024版

docker 指令安装安装 docker run -d --restartalways \ --name jenkins -uroot -p 10340:8080 \ -p 10341:50000 \ -v /home/docker/jenkins:/var/jenkins_home \ -v /var/run/docker.sock:/var/run/docker.sock \ -v /usr/bin/docker:/usr/bin/docker jenkins/jenkins:lts访问…

耐腐蚀耐高温实验室塑料烧杯进口高纯PFA材质反应器特氟龙烧杯

PFA烧杯在实验过程中可作为储酸容器或涉及强酸强碱类实验的反应容器&#xff0c;用于盛放样品、试剂&#xff0c;可搭配电热板加热、蒸煮、赶酸用。 外壁均有凸起刻度&#xff0c;直筒设计&#xff0c;带翻边&#xff0c;便于夹持和移动&#xff0c;边沿有嘴&#xff0c;便于倾…

深挖苹果Find My技术,伦茨科技ST17H6x芯片赋予产品功能

苹果发布AirTag发布以来&#xff0c;大家都更加注重物品的防丢&#xff0c;苹果的 Find My 就可以查找 iPhone、Mac、AirPods、Apple Watch&#xff0c;如今的Find My已经不单单可以查找苹果的设备&#xff0c;随着第三方设备的加入&#xff0c;将丰富Find My Network的版图。产…

GitOps - 为 OpenShift GitOps 配置邮件通知

《OpenShift 4.x HOL教程汇总》 说明&#xff1a;本文已经 在OpenShift 4.15 OpenShift GitOps 1.11.2 环境中验证 文章目录 ArgoCD 的 Notification 功能简介启动 OpenShift GitOps 的 Notification 功能配置邮件通知验证参考 说明&#xff1a;先根据《OpenShift 4 之 GitOp…

vue创建项目下载动态路由v-for mounted websocket :style :class store使用说明

在Vue中创建一个项目&#xff0c;并整合动态路由、v-for、mounted生命周期钩子、WebSocket、:style、:class以及Vuex的store&#xff0c;涉及到多个Vue核心特性的使用。下面我将简要说明如何逐步整合这些特性。 1. 创建Vue项目 使用Vue CLI创建项目&#xff1a; 2. 配置动态路…

sourcetree提交代码出现闪退报错(已解决)

当我在sourcetree提交代码时&#xff0c;点击提交按钮出现闪退关闭&#xff0c;并弹出下面的报错框&#xff0c;报错的图片如下&#xff1a; 那么经过了解&#xff0c;出现这样的报错原因是&#xff0c;git的提交时无法定位提交的人是谁&#xff0c;导致无法提交 那么解决的方…

git中对子模块的本地修改、提交和推送远程仓库

场景 当前的某个项目&#xff0c;其使用了另一个项目&#xff0c;我在本地需要对子项目进行修改&#xff0c;并将这些修改提交到github中的子项目和父项目。其实在github中&#xff0c;子项目都是特定的指向子项目的某次提交&#xff0c;因此对于父项目的修改&#xff0c;其实…

golang 选择排序

学习笔记&#xff5e; // Author sunwenbo // 2024/4/6 21:49 package mainimport "fmt"/* 选择排序基本介绍选择式排序也属于内部排序法&#xff0c;是从预排序的数据中按指定的规则选出某一元素&#xff0c;经过和其他元素重整&#xff0c;再依原则交换位置后达到…

设置你的第一个React应用

目录 一、React入门 1.1 你好React 1.2 创建React 1.3 应用结构 二、总结 2.1 定义组件 2.2 组件源码 三、组件详解 注意事项 3.1 组件三部曲 3.2 组件通信 —— props 3.3 对象数组迭代 —— map() 3.4 事件处理 3.5 钩子函数 —— useState() 初次学习最终效果…

ThreadLocal上传下载文件

文章目录 ThreadLocal1.基本介绍1.什么是ThreadLocal&#xff1f;2.示意图 2.快速入门1.创建普通java项目2.编写代码1.T1.java2.T1Service.java3.T2Dao.java4.Dog.java 3.结果 3.ThreadLocal源码解读1.set方法2.set方法总结3.get方法 上传下载文件1.基本介绍1.基本说明2.文件上…

Mybatis——一对多关联映射

一对多关联映射 一对多关联映射有两种方式&#xff0c;都用到了collection元素 以购物网站中用户和订单之间的一对多关系为例 collection集合的嵌套结果映射 创建两个实体类和映射接口 package org.example.demo;import lombok.Data;import java.util.List;Data public cla…

手动实现Tomcat底层机制+自己设计Servlet

文章目录 1.Tomcat整体架构分析自己理解 2.第一阶段1.实现功能2.代码1.TomcatV1.java 3.调试阶段1.阻塞在readLine导致无法返回结果 4.结果演示 3.第二阶段1.实现功能2.代码1.RequestHander.java2.TomcatV2.java 3.调试阶段1.发现每次按回车会接受到两次请求 4.结果演示 4.第三…

Redis常见数据类型(2)

String字符串 字符串是Redis最基础的数据类型, 关于字符串需要特别注意: (1)首先Redis中所有的键的类型都是字符串类型, 而且其它几种数据结构也是在字符串类型的基础上构建的, 例如列表和集合的元素都是字符串类型, 所以字符串类型能为其它4种数据结构的学习奠定基础. (2)其次…

基于Unet的BraTS 3d 脑肿瘤医学图像分割,从nii.gz文件中切分出2D图片数据

1、前言 3D图像分割一直是医疗领域的难题&#xff0c;在这方面nnunet已经成为了标杆&#xff0c;不过nnunet教程较少&#xff0c;本人之前跑了好久&#xff0c;一直目录报错、格式报错&#xff0c;反正哪里都是报错等等。并且&#xff0c;nnunet对于硬件的要求很高&#xff0c…