利用Socket动手实现简单HTTP协议

☆* o(≧▽≦)o *☆嗨~我是小奥🍹
📄📄📄个人博客:小奥的博客
📄📄📄CSDN:个人CSDN
📙📙📙Github:传送门
📅📅📅面经分享(牛客主页):传送门
🍹文章作者技术和水平有限,如果文中出现错误,希望大家多多指正!
📜 如果觉得内容还不错,欢迎点赞收藏关注哟! ❤️

文章目录

  • 利用Socket动手实现简单HTTP协议
    • 一、实现
    • 二、完整代码

利用Socket动手实现简单HTTP协议

一、实现

我们知道HTTP协议是在应用层解析内容的,所以我们只需要按照它的报文格式封装和解析数据即可。

关于具体的传输协议,使用的还是Socket,关于Socket,可以移步这篇文章:Java中的Socket你了解吗

这里我将带领大家使用NioServer实现一个简单的HTTP协议,注意,这里主要是出于了解HTTP协议是如何实现的,而不是真正实现一个HTTP协议。

因为HTTP协议是在接收到数据之后才会用到的,所以,我们只需要在Handler处理方法中实现就可以了。

我们首先来看一下效果。

当我们启动程序,然后在浏览器中输入http://localhost:8080/发起请求,这时控制台会打印如下信息:

GET / HTTP/1.1
Host: localhost:8080
Connection: keep-alive
Cache-Control: max-age=0
sec-ch-ua: "Not_A Brand";v="8", "Chromium";v="120", "Google Chrome";v="120"
sec-ch-ua-mobile: ?0
sec-ch-ua-platform: "Windows"
Upgrade-Insecure-Requests: 1
User-Agent: Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/120.0.0.0 Safari/537.36
Accept: text/html,application/xhtml+xml,application/xml;q=0.9,image/avif,image/webp,image/apng,*/*;q=0.8,application/signed-exchange;v=b3;q=0.7
Sec-Fetch-Site: none
Sec-Fetch-Mode: navigate
Sec-Fetch-User: ?1
Sec-Fetch-Dest: document
Accept-Encoding: gzip, deflate, br
Accept-Language: zh-CN,zh;q=0.9Method:	GET
url:	/
HTTP Version:	HTTP/1.1GET /favicon.ico HTTP/1.1
Host: localhost:8080
Connection: keep-alive
sec-ch-ua: "Not_A Brand";v="8", "Chromium";v="120", "Google Chrome";v="120"
sec-ch-ua-mobile: ?0
User-Agent: Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/120.0.0.0 Safari/537.36
sec-ch-ua-platform: "Windows"
Accept: image/avif,image/webp,image/apng,image/svg+xml,image/*,*/*;q=0.8
Sec-Fetch-Site: same-origin
Sec-Fetch-Mode: no-cors
Sec-Fetch-Dest: image
Referer: http://localhost:8080/
Accept-Encoding: gzip, deflate, br
Accept-Language: zh-CN,zh;q=0.9Method:	GET
url:	/favicon.ico
HTTP Version:	HTTP/1.1

浏览器显示结果如图所示:

在这里插入图片描述

这里只是一个简单的示例,目的是让大家了解HTTP协议中的实现方法,功能不够完善,而且也不能实现请求的处理。

对于了解HTTP协议实现的方法,个人觉得还是有必要看看的。

二、完整代码

思路:

整个过程非常简单,我们只需要按照报文的格式来读取和发送就可以了。

  • 接收到数据后按\r\n分割成每一行,空行前面都是报文头;
  • 空行下面如果有内容就是报文主体,注意,Get请求没有报文主体;

完整代码如下:

import java.io.IOException;
import java.net.InetSocketAddress;
import java.nio.ByteBuffer;
import java.nio.channels.SelectionKey;
import java.nio.channels.Selector;
import java.nio.channels.ServerSocketChannel;
import java.nio.channels.SocketChannel;
import java.nio.charset.Charset;
import java.util.Iterator;/*** @Author zal* @Date 2024/01/15  21:10* @Description: 基于NioServer实现简单HTTP协议* @Version: 1.0*/
public class NioServer {public static void main(String[] args) throws Exception {// 创建ServerSocketChannel,并监听8080端口ServerSocketChannel ssc = ServerSocketChannel.open();ssc.socket().bind(new InetSocketAddress(8080));// 设置为非阻塞模式ssc.configureBlocking(false);// 为ssc注册选择器Selector selector = Selector.open();ssc.register(selector, SelectionKey.OP_ACCEPT);// 创建处理器while (true) {// 等待请求,每次等待阻塞3s,超过3s后线程继续向下执行,如果传入0或者不传参数则一直阻塞if (selector.select(3000) == 0) {continue;}// 获取等待处理的SelectionKeyIterator<SelectionKey> keyIterator = selector.selectedKeys().iterator();while (keyIterator.hasNext()) {SelectionKey key = keyIterator.next();// 启动新县城处理SelectionKeynew Thread(new HttpHandler(key)).run();// 处理完成后,从待处理的SelectionKey中移除当前使用的keykeyIterator.remove();}}}/*** 静态内部类,用于处理连接和读取数据*/private static class HttpHandler implements Runnable {private int bufferSize = 1024;private String localCharset = "UTF-8";private SelectionKey key;public HttpHandler(SelectionKey key) {this.key = key;}/*** 处理接受连接事件** @throws IOException*/public void handleAccept() throws IOException {// 通过服务器套接字通道接受客户端连接SocketChannel sc = ((ServerSocketChannel) key.channel()).accept();// 配置为非阻塞模式sc.configureBlocking(false);// 将客户端套接字通道注册到选择器,关注事件为可读,同时附带一个缓冲区sc.register(key.selector(), SelectionKey.OP_READ, ByteBuffer.allocate(bufferSize));}/*** 处理读取数据事件** @throws IOException*/public void handleRead() throws IOException {// 获取ChannelSocketChannel sc = (SocketChannel) key.channel();// 获取附加到事件的缓冲区ByteBuffer buffer = (ByteBuffer) key.attachment();buffer.clear(); // 清空缓冲区,准备读取数据// 从客户端通道读取数据到缓冲区,如果返回-1表示客户端关闭连接if (sc.read(buffer) == -1) {// 关闭channelsc.close();} else {// 切换buffer为读模式buffer.flip();// 将buffer中的数据解码为字符串后保存到receivedStringString receivedString = Charset.forName(localCharset).newDecoder().decode(buffer).toString();// 控制台打印请求报文头String[] requsetMessage = receivedString.split("\r\n");for (String s : requsetMessage) {System.out.println(s);// 遇到空行说明报文头已经打印完成if (s.isEmpty()) {break;}}// 控制台打印首行信息String[] firstLine = requsetMessage[0].split(" ");System.out.println();System.out.println("Method:\t" + firstLine[0]);System.out.println("url:\t" + firstLine[1]);System.out.println("HTTP Version:\t" + firstLine[2]);System.out.println();// 返回客户端StringBuffer sendString = new StringBuffer();// 响应报文首行sendString.append("HTTP/1.1 200 OK\r\n");sendString.append("Content-Type:text/html;charset=" + localCharset + "\r\n");sendString.append("\r\n");sendString.append("<html><head><title>显示报文</title></head></body>");sendString.append("接收到的请求报文是:<br/>");for (String s : requsetMessage) {sendString.append(s + "<br/>");}sendString.append("</body></html>");buffer = ByteBuffer.wrap(sendString.toString().getBytes(localCharset));sc.write(buffer);sc.close();}}@Overridepublic void run() {try {// 接收到请求连接时if (key.isAcceptable()) {handleAccept();}// 读数据if (key.isReadable()) {handleRead();}} catch (IOException e) {e.printStackTrace();}}}
}

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

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

相关文章

HDFS HA 集群搭建 - 基于Quorum Journal Manager(hadoop2.7.1)

0、前置概念 0.1 checkpoint 检查点 在Hadoop分布式文件系统(HDFS)中,检查点(Checkpointing)是一个关键的过程,它涉及到将文件系统的命名空间状态持久化到磁盘。这个状态由两部分组成:EditLogs和FsImage。 EditLogs:记录了自FsImage生成后对文件系统所做的所有修改。…

python 元组的详细用法

当前版本&#xff1a; Python 3.8.4 文章目录如下 1. 介绍元组 2. 定义元组 3. 访问元组 4. 查询元组 1. 介绍元组 元组&#xff08;Tuple&#xff09;是一个有序的、不可变的数据序列。它可以包含各种类型的数据&#xff0c;例如数字、字符串、列表等。元组使用圆括号()来…

Python3 如何做数据类型转换

介绍 在Python中&#xff0c;数据类型用于分类特定类型的数据&#xff0c;确定可以分配给该类型的值以及可以对其执行的操作。在编程时&#xff0c;有时我们需要在不同类型之间转换值&#xff0c;以便以不同的方式操作值。例如&#xff0c;我们可能需要将数字值与字符串连接&a…

Elasticsearch:聊天机器人教程(二)

这是继上一篇文章 “Elasticsearch&#xff1a;聊天机器人教程&#xff08;一&#xff09;”的续篇。本教程的这一部分讨论聊天机器人实现中最有趣的方面&#xff0c;以帮助你理解它并对其进行自定义。 数据摄入 在此应用程序中&#xff0c;所有示例文档的摄取都是通过 flask …

2024年信息安全不完全预测

不需要专家预言就能知道&#xff0c;计算机安全将在2024年出现在新闻中&#xff0c;而且可能不是什么好事。但2024年网络犯罪分子将如何试图突破防御并窃取有价值的数据&#xff0c;值得我们看一看安全专家们的看法和预测。 不需要专家预言就能知道&#xff0c;计算机安全将在…

【深度学习I-基础知识】

深度学习I-基础知识 1 基础知识1.1 模型的基本概念1.2 机器学习1.2.1 概率建模1.2.2 核方法1.2.3 决策树、随机森林和梯度提升机 1.3 深度学习1.3.1 张量1.3.2 数据批量1.3.3 张量运算1.3.4 训练过程 1 基础知识 1.1 模型的基本概念 模型是现实世界中一类具有泛化共性的真实系…

canvas截取视频图像(图文示例)

查看专栏目录 canvas示例教程100专栏&#xff0c;提供canvas的基础知识&#xff0c;高级动画&#xff0c;相关应用扩展等信息。canvas作为html的一部分&#xff0c;是图像图标地图可视化的一个重要的基础&#xff0c;学好了canvas&#xff0c;在其他的一些应用上将会起到非常重…

2023一带一路暨金砖国家技能发展与技术创新大赛“网络安全”赛项省选拔赛样题卷②

2023金砖国家职业技能竞赛"网络安全" 赛项省赛选拔赛样题 2023金砖国家职业技能竞赛 省赛选拔赛样题第一阶段&#xff1a;职业素养与理论技能项目1. 职业素养项目1. 职业素养项目2. 网络安全项目3. 安全运营 第二阶段&#xff1a;安全运营项目1. 操作系统安全配置与加…

Docker部署的gitlab升级指南(15.11.X容器里升级PostgreSQL到13.8)

一、确定当前版本 #进入当前版本容器产看gitlab版本 docker exec -it gitlab cat /opt/gitlab/embedded/service/gitlab-rails/VERSION#显示版本如下 14.4.0二、备份数据&#xff0c;防止升级发生意外 #执行备份命令 docker exec -ti gitlab gitlab-rake gitlab:backup:creat…

go的安装及配置

go的官方下载地址&#xff1a;All releases - The Go Programming Language​​​​​​ 1、找到对应的版本包下载&#xff0c;例如 wget https://golang.google.cn/dl/go1.21.6.linux-amd64.tar.gz 2、下载完成后配置解压Go源码包 tar -zxf go1.21.6.linux-amd64.tar.gz 3…

【IC设计】ICer‘s 乾坤大挪移——FSM状态机

目录 理论解读状态机定义状态转移图Mealy和Moore型状态机推荐写“新两段式状态机” 设计实战可乐机两种state的FSM&#xff08;异步复位&#xff09;4种状态的one-hot状态机4种状态的同步复位状态机蓄水池问题 参考链接 理论解读 状态机定义 状态机简写为 FSM&#xff08;Fin…

Python3 索引下标及切片完全指南

介绍 Python 字符串数据类型是由一个或多个字符组成的序列&#xff0c;可以包含字母、数字、空格字符或符号。由于字符串是一个序列&#xff0c;我们可以通过索引和切片的方式访问它&#xff0c;就像访问其他基于序列的数据类型一样。 本教程将指导您通过索引访问字符串&…

docker 学习命令整理

文章目录 docker 学习命令整理(积累中...)0. 启动/停止1. 运行2. 查看运行中docker3. 删除指定container4. 查看本地镜像5. 拉取指定镜像6. 新起终端进入同一container7. 取消sudo8. 查看docker状态9. 查看docker存储10.删除镜像11.删除容器12. qemu12.1 安装12.2 卸载qemu 附&…

网络攻击与检测防御:维护数字安全的关键挑战

随着数字化时代的深入&#xff0c;网络攻击已成为企业和个人面临的严峻挑战之一。本文将深入探讨不同类型的网络攻击&#xff0c;以及有效的检测和防御策略&#xff0c;以确保网络系统的安全性和稳定性。 1. 常见网络攻击类型&#xff1a; DDoS 攻击&#xff1a;分布式拒绝服…

IOC之Spring统一资源加载策略

前言 在学 Java的时候&#xff0c;我们学习了一个标准类 java.net.URL&#xff0c;该类在 Java SE 中的定位为统一资源定位器&#xff08;Uniform Resource Locator&#xff09;&#xff0c;但是我们知道它的实现基本只限于网络形式发布的资源的查找和定位。然而&#xff0c;实…

生成式对抗网络GAN

Generative Adversarial Nets由伊恩古德费洛&#xff08;Ian J.Goodfellow&#xff09;等人于2014年发表在Conference on Neural Information Processing Systems (NeurIPS)上。NeurIPS是机器学习和计算神经科学领域的顶级国际学术会议之一。 1. GAN在哪些领域大放异彩 图像生…

黑马苍穹外卖学习Day6

HttpClient 介绍 HttpClient 是 Apache 提供的一个开源的 Java HTTP 客户端库&#xff0c;用于发送 HTTP 请求和处理 HTTP 响应。它提供了一种更简便的方式来执行 HTTP 请求&#xff0c;并支持多种协议&#xff0c;如 HTTP、HTTPS、FTP 等。 使用 HttpClient 可以方便地与远程…

MySQL体系结构

MySQL体系结构 MySQL采用的是客户/服务器体系结构&#xff0c;实际是有两个程序&#xff0c;一个是MySQL服务器程序&#xff0c;指的是mysqld程序&#xff0c;运行在存放数据库的机器上&#xff0c;负责在网络上监听并处理来自客户的服务请求&#xff0c;根据这些请求去访问数据…

一种基于智能手机的地下停车场寻车系统

原文来自于《Help You Locate the Car: a Smartphone-based Car-finding System in Underground Parking Lot》 这篇论文提出了一种基于智能手机的地下停车场寻车系统。该系统旨在帮助驾驶员在没有额外设备和地图支持的情况下找到他们的车辆。主要内容和贡献如下: 系统概述:目…

CNN/DailyMail训练文本摘要模型

要使用 TensorFlow Datasets (TFDS) 来训练一个文本摘要模型&#xff0c;可以选择一个包含文章和摘要的数据集&#xff0c;例如 CNN/DailyMail 数据集。 这个数据集通常用于训练和评估文本摘要模型。 以下是使用 TFDS 加载数据集并训练一个简单的序列到序列 (seq2seq) 模型的…