如何从头开始以正确的面向对象方式创建Java Web Framework

您如何用Java设计Web应用程序? 您安装了Spring,阅读了手册,创建了控制器 ,创建了一些视图,添加了一些注释 ,它就可以工作了。 如果没有Spring (Ruby中没有Ruby on Rails,PHP中没有Symphony,也没有…等),您将怎么办? 让我们尝试从头开始创建一个Web应用程序,从一个纯Java SDK到一个功能齐全的Web应用程序(由单元测试覆盖)结束。 几周前,我录制了第42号网络研讨会 ,但本文应该对此进行更详细的说明。


蒂芙尼早餐(Blake Edwards,1961年)

首先,我们必须创建一个HTTP服务器,该服务器将打开服务器套接字,侦听传入的连接,读取他们必须说的所有内容(HTTP请求)并返回任何Web浏览器想要的信息(HTTP响应)。 您知道HTTP的工作原理吧? 如果您不这样做,这里有个简短的提醒:

Web浏览器向服务器发送请求,该请求看起来像这样(这是纯文本数据):

GET /index.html HTTP/1.1
Host: www.example.com

服务器必须阅读此文本,准备答案(必须是浏览器可读HTML页面),然后像这样返回:

HTTP/1.1 200 OK
Content-Type: text/html; charset=UTF-8
Content-Length: 26<html>Hello, world!</html>

而已。 这是一个非常简单的原始协议。 用Java实现Web服务器也不是那么复杂。 这是一个非常简单的形式:

import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.net.ServerSocket;
import java.net.Socket;
import java.net.SocketTimeoutException;
import java.util.Arrays;
public class Main {public static void main(String... argv) {try (ServerSocket server = new ServerSocket(8080)) {server.setSoTimeout(1000);while (true) {try (Socket socket = server.accept()) {try (InputStream input = socket.getInputStream();OutputStream output = socket.getOutputStream()) {byte[] buffer = new byte[10000];int total = input.read(buffer);String request = new String(Arrays.copyOfRange(buffer, 0, total));String response = "HTTP/1.1 200 OK\r\n\r\nHello, world!";output.write(response.getBytes());}} catch (SocketTimeoutException ex) {if (Thread.currentThread().isInterrupted()) {break;}}}}}
}

尝试运行它,它应该可以工作。 您应该能够在浏览器中打开http://localhost:8080页面,然后看到Hello, world!Hello, world! 文本。

它还不是Web应用程序,而是一个框架,它可以将HTTP请求简单地分配到HTTP响应中。 尽管其中没有严重的面向对象的问题。 这是相当程序化的方法,但确实可行。 现在,我们应该关注一个更重要的问题:如何为Web应用程序添加更多功能,并使其能够处理不同的页面,呈现更大的内容并处理错误? 上面代码段中的request变量应该以某种方式转换为response

最简单的方法是1)将请求转换为内部包含所有详细信息的DTO ,然后2)将其发送到知道如何处理DTO数据的“控制器”,然后3)接收响应DTO从控制器中取出数据并呈现响应。 这就是春天和 所有其他框架都可以做到。 但是,我们不会走这条路,我们将尝试做到无DTO且纯粹面向对象。

我不得不说,可能有多种设计,全部都是OOP风格。 现在,我仅向您显示这些选项之一。 您无疑会知道我们几年前诞生的Takes框架-它具有自己的设计,也面向对象。 但是我现在建议的那个似乎更好。 您可能还会提出其他建议,因此不要犹豫,在下面的评论中发表您的想法,甚至创建GitHub存储库并在那里分享您的想法。

我建议我们引入两个接口: ResourceOutputResource是服务器端实体,它根据传入的请求参数而发生变化。例如,当我们只知道请求是GET / ,它就是一种资源。 但是,如果我们也知道该请求具有例如Accept: text/plain ,则可以更改该请求并创建一个新请求,该请求将传递纯文本。 这是界面:

interface Resource {Resource refine(String name, String value);
}

这是我们创建和变异的方法:

Resource r = new DefaultResource().refine("X-Method", "GET").refine("X-Query", "/").refine("Accept", "text/plain");

注意:每次调用.refine()返回一个新的接口Resource实例。 它们都是不可变的,就像对象必须是一样 。 由于这种设计,我们不会将数据与处理器分开。 资源是数据和处理器。 每个资源都知道如何处理数据,并且仅接收应该接收的数据。 从技术上讲,我们只是以面向对象的方式实现请求调度

然后,我们需要将资源转换为响应。 我们赋予资源使其能够响应的能力。 我们不希望数据以某种DTO的形式泄漏资源。 我们希望该资源打印响应。 如何为资源提供其他方法print()

interface Resource {Resource refine(String name, String value);void print(Output output);
}

然后,接口Output看起来像这样:

interface Output {void print(String name, String value);
}

这是Output的原始实现:

public class StringBuilderOutput implements Output {private final StringBuilder buffer;StringBuilderOutput(StringBuilder buf) {this.buffer = buf;}@Overridepublic void print(String name, String value) {if (this.buffer.length() == 0) {this.buffer.append("HTTP/1.1 200 OK\r\n");}if (name.equals("X-Body")) {this.buffer.append("\r\n").append(value);} else {this.buffer.append(name).append(": ").append(value).append("\r\n");}}
}

要构建HTTP响应,我们可以这样做:

StringBuilder builder = new StringBuilder();
Output output = new StringBuilderOutput(builder);
output.print("Content-Type", "text/plain");
output.print("Content-Length", "13");
output.print("X-Body", "Hello, world!");
System.out.println(builder.toString());

现在,让我们创建一个类,该类使用Resource的实例作为调度程序 ,以接收传入的请求String并生成响应String

public class Session {private final Resource resource;Session(Resource res) {this.resource = res;}String response(String request) throws IOException {Map<String, String> pairs = new HashMap<>();String[] lines = request.split("\r\n");for (int idx = 1; idx < lines.length; ++idx) {String[] parts = lines[idx].split(":");pairs.put(parts[0].trim(), parts[1].trim());if (lines[idx].empty()) {break;}}String[] parts = lines[0].split(" ");pairs.put("X-Method", parts[0]);pairs.put("X-Query", parts[1]);pairs.put("X-Protocol", parts[2]);App.Resource res = this.resource;for (Map.Entry<String, String> pair : pairs.entrySet()) {res = res.refine(pair.getKey(), pair.getValue());}StringBuilder buf = new StringBuilder();res.print(new StringBuilderOutput(buf));return buf.toString();}
}

首先,我们解析请求,将其标头分成几行,并忽略请求的主体。 您可以使用X-Body作为键,修改代码以解析主体并将其传递给refine()方法。 目前,上面的代码无法做到这一点。 但是你明白了。 片段的解析部分准备了可以在请求中找到的对,并将它们一对一地传递给封装的资源,对其进行变异直到最终形式。 始终返回文本的简单资源可能如下所示:

class TextResource implements Resource {private final String body;public TextResource(String text) {this.body = text;}@Overridepublic Resource refine(String name, String value) {return this;}@Overridepublic void print(Output output) {output.print("Content-Type", "text/plain");output.print("Content-Length", Integer.toString(this.body.length()));output.print("X-Body", this.body);}
}

根据查询的路径,关注查询字符串并将请求分派给其他资源的资源可能看起来像这样:

new Resource() {@Overridepublic Resource refine(String name, String value) {if (name.equals("X-Query")) {if (value.equals("/")) {return new TextResource("Hello, world!");} else if (value.equals("/balance")) {return new TextResource("256");} else if (value.equals("/id")) {return new TextResource("yegor");} else {return new TextResource("Not found!");}} else {return this;}}@Overridepublic void print(final Output output) {throws IllegalStateException("This shouldn't happen");}
}

我希望你有主意。 上面的代码很粗略,并且大多数用例都没有实现,但是如果您感兴趣,可以自己做。 该代码位于yegor256 / jpages存储库中。 请毫不犹豫地为请求请求做出贡献,并使这个小型框架成为现实。

翻译自: https://www.javacodegeeks.com/2019/03/how-to-create-a-java-web-framework-from-scratch-the-right-object-oriented-way.html

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

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

相关文章

工业级光纤收发器使用“避坑”指南

工业级光纤收发器在使用中有很多的注意事项&#xff0c;往往这些注意事项经常被忽略。下面飞畅科技就整理了一些工业级光纤收发器使用“避坑”指南&#xff0c;大家可以仔细看看&#xff0c;引起重视。 使用工业级光纤收发器连接不同的设备时&#xff0c;必须注意使用的端口不…

工业级光纤收发器一般能正常使用多长时间?

无论是厂商还是购买商在生产、选购工业级光纤收发器的时候&#xff0c;重要的参考指标就是它的使用寿命。那么&#xff0c;工业级光纤收发器的正常使用寿命大概有多长时间呢&#xff1f;接下来飞畅科技就来和大家简单聊一聊工业级光纤收发器的使用寿命&#xff0c;一起来看看吧…

红帽 jboss_红帽正式宣布发布JBoss BPM Suite 6和JBoss BRMS 6

红帽 jboss红帽公司刚刚宣布了这些期待已久的产品的全面上市 &#xff01; 要花费大量精力将社区代码转换为企业质量的软件&#xff0c;客户和最终用户可以在Red Hat支持的生产环境中使用这些代码。 现在是现在和潜在客户了解该产品&#xff0c;让合作伙伴开始使用它并学习基本…

工业交换机的几大“择机”标准,你学会了吗?

工业交换机现在是越来越普及了&#xff0c;很多相对较苛刻、复杂的工作环境基本都靠工业交换机来进行数据网络通信。但是&#xff0c;市场上形形色色的工业交换机非常多&#xff0c;如果企业安排你去采购一批交换机&#xff0c;你会选吗&#xff1f;就这个问题&#xff0c;下面…

程序员的前20个搜索和排序算法面试问题

大家好&#xff0c;如果您正在准备编程工作面试或正在寻找新工作&#xff0c;那么您知道这不是一个容易的过程。 在您职业的任何阶段&#xff0c;您都必须幸运地接到电话并进行第一轮面试&#xff0c;但是在初学者方面&#xff0c;当您寻找第一份工作时就更加困难。 这就是为什…

工业交换机性能中的“自适应”该如何理解?

工业交换机诸多性能指标中&#xff0c;我们常常看见有“自适应”这个指标。它到底是什么意思呢&#xff1f;接下来飞畅科技就来给大家详细讲解一下&#xff0c;一起来看看吧&#xff01; 自适应也叫自动匹配、自协商&#xff0c;以太网技术发展到100M速率以后&#xff0c;出现…

【无线lora模块星型组网】lora无线模块专利技术 跳频扩频 支持200节点并发

E70 (433NW30S)采用星型组网模块&#xff0c;工作在433MHz频段&#xff0c;模块集协调器、终端为一体&#xff0c;具有长距离、高速率两种传输模式&#xff0c;一个协调器支持多达200个节点与其通讯&#xff0c;所有操作配置采用行业标准AT指令&#xff0c;极大简化用户操作&am…

控制器局域网can总线

很多的司机朋友以及工控圈的朋友都有听到过“CAN总线”一词&#xff0c;今天我们就来一起看一下CAN总线为什么这么火&#xff1f; CAN总线&#xff0c;也叫做控制器局域网总线&#xff08;Controller Area Network&#xff09;&#xff0c;是一种用于实时应用的串行通讯协议总…

交换机多少钱一个?影响工业交换机价格的因素?

工业交换机&#xff0c;即应用在工业领域的交换机。它的性能和各个安全指标要比普通&#xff08;商业&#xff09;交换机要求更稳定一些。因此&#xff0c;工业交换机的价格要比一般的交换机要稍贵一些。那一般工业交换机多少钱一个呢&#xff1f;是不是每个厂家的交换机价格都…

[渝粤教育] Huazhong University of Science and Technology Analog Electronics Technique 参考 资料

教育 -Analog Electronics Technique-章节资料考试资料-Huazhong University of Science and Technology【】 Test questions for The Ideal Operational Amplifier 1、【单选题】The two input terminals of an operational amplifier are labeled as: A、high and low B、pos…

【NBIoT无线模块DTU数传电台】串口服务器RS232/RS485端口工业路由信号传输

E840-DTU (NB-02)是为实现串口设备与网络服务器&#xff0c;通过NB网络相互传输数据而开发的产品&#xff0c;通过简单的AT指令进行设置&#xff0c;即可轻松使用本产品实现串口到网络的双向数据透明传输。 基本功能&#xff1a; 通过无线方式传输数字信号的高性能无线收发装…

[渝粤教育] Nanjing University of Aeronautics and Astronautics Grey Data Analysis 参考 资料

教育 -Grey Data Analysis-章节资料考试资料-Nanjing University of Aeronautics and Astronautics【】 Chapter 1 Concept and basic principle of grey system 1、【单选题】The research object of gray system theory is&#xff1a; A、A. Clear extension, clear connota…

工业级光纤收发器的“附加属性“功能介绍

工业级光纤收发器作为光电信号转换设备&#xff0c;除了其设计本身的功能外&#xff0c;还具有其他很多的拓展功能。今天飞畅科技的小编就来为大家简单介绍一下工业级光纤收发器的”附加属性“功能&#xff0c;一起来看看吧&#xff01; 工业级光纤收发器在数据传输上打破了以…

串口服务器E810-DTU实现以太网口与RS232数据透明传输

E810-DTU-V1.0是一款232转以太网的单串口服务器&#xff0c;实现了RJ45网口与RS232之间的数据透明传输。 模块搭载M0系列32位处理器&#xff0c;运行速率快&#xff0c;效率高。具备自适应网络速率(最高支持100M全双工)、TCP Server、TCP Client、UDP Server、UDP Client四种通…

无线数传电台rs232和rs485串口接口:230M数传电台

E51-DTU-2W是一款频率230M无线数传电台( 同时具有RS232/RS485接口)&#xff0c;透明传输方式&#xff0c;工作在225~237 .6MHz频段(默认230MHz)&#xff0c;工作电压范围8V~28V。具有载波频率: 230MHz、发射功率: 33dBm、通信距离: 8.0km、接口类型: RS232/RS485、供电电压: 8~…

[渝粤教育] University of Science and Technology Beijing Discrete Mathematics 参考 资料

教育 -Discrete Mathematics-章节资料考试资料-University of Science and Technology Beijing【】 Chapter 1 unit tests 1、【单选题】Which is a proposition in the following statements? A、Please don’t copy! B、The sun is a planet. C、Can I smoke here? D、x –…

工业级光纤收发器和协议转换器有什么区别呢?

工业级光纤收发器是将光电信号进行转换的&#xff1b;而协议转化器是进行协议转换的。那么&#xff0c;工业级光纤收发器和协议转换器具体有哪些区别呢&#xff1f;接下来飞畅科技的小编就来为大家具体分析一下二者之间的区别&#xff0c;一起来看看吧&#xff01; 工业级光纤…

中继在无线通讯中的应用

无线中继&#xff0c;即中继节点在无线网络中起到中继的作用&#xff0c;能实现信号的中继和放大&#xff0c;从而延伸无线网络的覆盖范围。就中继的运行机制而言&#xff0c;中继可分为洪泛中继、定向中继。 洪泛中继&#xff0c;是指中继设备在任何时刻收到的任意数据包&…

SI4463模块配合WDS的快速上手指南

1.前言&#xff1a; Wireless Development Suite (WDS)是Silicon Labs公司提供用于ISM频段的EZRadioPRO系列配置和调试的计算机终端软件。可以使用此软件对模块的射频参数&#xff0c;寄存器配置和引脚中断进行设置。 2.SI4463概述 SI4463是Silicon Labs一款高性能的射频收发…

工业级光纤收发器入网说明

我们都知道&#xff0c;一个网络是由各种光学器件组成的&#xff0c;工业级光纤收发器就是其中的一个重要组成部分。但由于我们常使用的网线(双绞线)的最大传输距离有很大的局限性&#xff0c;一般双绞线的最大传输距离为100米。因此,当我们在布置较大的网络的时候&#xff0c;…