基于jdk1.8 NIO手写http server,并且部署到阿里云服务器,浏览器可访问


如果没有tomcat,你的java程序是不是单机版?
首先明确一个知识点(大前提),网络中的进程是通过socket来通信的
最近在学习socket(俗称 套接字),然后顺便了解一下http协议,手写一个http服务端出来玩一下,如果想实现其实也不是很复杂,需要了解一下http协议,了解一下使用socket编程,那就一切变得很简单了 ~~

# 1.0 简要了解 “发送” 一个http请求包含的信息

比如说我从浏览器访问我的一篇博客 https://shaines.cn/blog/detail/1183794984409358336
在这里结合使用Fiddler抓取浏览器数据包获取到的数据,如下

  1. GET https://shaines.cn/blog/detail/1183794984409358336 HTTP/1.1
  2. Host: shaines.cn
  3. Connection: keep-alive
  4. Upgrade-Insecure-Requests: 1
  5. User-Agent: Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/77.0.3865.120 Safari/537.36
  6. Accept: text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,image/apng,*/*;q=0.8,application/signed-exchange;v=b3
  7. Sec-Fetch-Site: cross-site
  8. Accept-Encoding: gzip, deflate, br
  9. Accept-Language: zh-CN,zh;q=0.9

说明

在这里插入图片描述

# 2.0 简要了解 “响应” 一个http请求包含的信息

  1. HTTP/1.1 200
  2. Server: nginx/1.14.2
  3. Date: Sat, 19 Oct 2019 03:48:26 GMT
  4. Content-Type: text/html;charset=UTF-8
  5. Transfer-Encoding: chunked
  6. Connection: keep-alive
  7. Content-Language: zh-CN
  8. 1f77
  9. <!doctype html>
  10. <html xmlns="http://www.w3.org/1999/xhtml">
  11. <html xmlns="http://www.w3.org/1999/xhtml">
  12. <head>
  13. <meta charset="UTF-8">
  14. <meta http-equiv="X-UA-Compatible" content="IE=edge">
  15. <meta name="description" content="如果你知道python的requests有多好用,那么这个封装的就有多舒服!!!类似于一个浏览器端,自动管理cookie,对于爬虫来说,就是一个利器">
  16. <meta name="keywords" content="模仿jdk11中的HttpClient封装一个基于HttpURLConnection的实现">
  17. <meta name="viewport" content="width=device-width, initial-scale=1, maximum-scale=1, user-scalable=no">
  18. <title>houyu blog | 文章详情</title>
  19. <meta name="renderer" content="webkit">
  20. <meta http-equiv="Cache-Control" content="no-siteapp"/>
  21. <link rel="icon" type="image/png" href="/favicon.ico">
  22. <meta name="mobile-web-app-capable" content="yes">
  23. <meta name="apple-mobile-web-app-capable" content="yes">
  24. <meta name="apple-mobile-web-app-status-bar-style" content="black">
  25. <meta name="apple-mobile-web-app-title" content="Amaze UI"/>
  26. <meta name="msapplication-TileColor" content="#0e90d2">
  27. <link rel="stylesheet" href="/assets/css/amazeui.min.css">
  28. <link rel="stylesheet" href="/assets/css/app.css">
  29. <link rel="stylesheet" href="/assets/css/toastr.min.css">
  30. </head>
  31. <body id="blog-article-sidebar">
  32. ...
  33. </body>
  34. </html>
  35. 0

说明

在这里插入图片描述

# 3.0 使用jdk1.8 NIO实现http server

  1. import java.io.IOException;
  2. import java.net.InetSocketAddress;
  3. import java.nio.ByteBuffer;
  4. import java.nio.channels.SelectionKey;
  5. import java.nio.channels.Selector;
  6. import java.nio.channels.ServerSocketChannel;
  7. import java.nio.channels.SocketChannel;
  8. import java.nio.charset.Charset;
  9. import java.util.Iterator;
  10. import org.slf4j.Logger;
  11. import org.slf4j.LoggerFactory;
  12. /**
  13. * @description http 服务端
  14. * @date 2019-10-01 18:16:12
  15. * @author houyu for.houyu@foxmail.com
  16. */
  17. public class HttpServer {
  18. private static final Logger logger = LoggerFactory.getLogger(HttpServer.class);
  19. private Selector selector = null;
  20. public static void main(String[] args) throws IOException {
  21. new HttpServer().start();
  22. }
  23. private void start() throws IOException {
  24. int port = 10006;
  25. ServerSocketChannel channel = ServerSocketChannel.open();
  26. channel.configureBlocking(false);
  27. selector = Selector.open();
  28. channel.register(selector, SelectionKey.OP_ACCEPT);
  29. channel.bind(new InetSocketAddress(port));
  30. logger.info("server start... bing port {}", port);
  31. beginAccept();
  32. }
  33. private void beginAccept() {
  34. while(true) {
  35. accept();
  36. }
  37. }
  38. private void accept() {
  39. try {
  40. int select = selector.select();
  41. if(select == 0) {
  42. return;
  43. }
  44. Iterator<SelectionKey> iter = selector.selectedKeys().iterator();
  45. while(iter.hasNext()) {
  46. SelectionKey key = iter.next();
  47. HttpRequestHandler.build(key).run();
  48. iter.remove();
  49. }
  50. } catch(IOException e) {
  51. logger.warn("accept()", e);
  52. }
  53. }
  54. public static class HttpRequestHandler {
  55. private static final Charset CHARSET = Charset.forName("UTF-8");
  56. private static final int BUFFER_SIZE = 1024;
  57. private SelectionKey key;
  58. HttpRequestHandler(SelectionKey key) {
  59. this.key = key;
  60. }
  61. private static HttpRequestHandler build(SelectionKey key) {
  62. return new HttpRequestHandler(key);
  63. }
  64. private void run() {
  65. try {
  66. if(key.isAcceptable()) {
  67. acceptHttpRequest();
  68. } else if(key.isReadable()) {
  69. readHttpRequest();
  70. returnHttpResponse();
  71. }
  72. } catch(IOException e) {
  73. logger.warn("run()", e);
  74. }
  75. }
  76. private void acceptHttpRequest() throws IOException {
  77. SocketChannel socketChannel = ((ServerSocketChannel) key.channel()).accept();
  78. socketChannel.configureBlocking(false);
  79. socketChannel.finishConnect();
  80. //
  81. Selector selector = key.selector();
  82. // 注册读取事件
  83. socketChannel.register(selector, SelectionKey.OP_READ, ByteBuffer.allocate(BUFFER_SIZE));
  84. }
  85. private void readHttpRequest() {
  86. SocketChannel channel = (SocketChannel) key.channel();
  87. ByteBuffer byteBuffer = (ByteBuffer) key.attachment();
  88. try {
  89. if(channel.read(byteBuffer) == -1) {
  90. // 没读到内容关闭连接
  91. channel.close();
  92. } else {
  93. // 将channel改为读取状态(重置光标为0)
  94. byteBuffer.flip();
  95. String requestBody = CHARSET.newDecoder().decode(byteBuffer).toString();
  96. InetSocketAddress inetSocketAddress = (InetSocketAddress) channel.getRemoteAddress();
  97. logger.debug("get request body from client({}) :\r\n{}", inetSocketAddress, requestBody);
  98. byteBuffer.clear();
  99. }
  100. } catch(IOException e) {
  101. logger.warn("服务器读取错误", e);
  102. }
  103. }
  104. private void returnHttpResponse() throws IOException {
  105. SocketChannel channel = (SocketChannel) key.channel();
  106. //
  107. String ipAndPort = channel.getRemoteAddress().toString().replace("/", "");
  108. //
  109. StringBuilder builder = new StringBuilder(1024);
  110. // 增加响应消息行
  111. builder.append("HTTP/1.1 200 ok\r\n");
  112. // 增加响应消息头
  113. builder.append("Content-Type: application/json; charset=utf-8\r\n");
  114. // 空行
  115. builder.append("\r\n");
  116. // 响应消息体
  117. builder.append("{");
  118. builder.append("\"code\": 200, ");
  119. builder.append("\"message\": \"您的IP:").append(ipAndPort).append(", 请求成功\"");
  120. builder.append("}");
  121. ByteBuffer byteBuffer = (ByteBuffer) key.attachment();
  122. byteBuffer.clear();
  123. byteBuffer.put(builder.toString().getBytes(CHARSET));
  124. // 设置光标在开始处
  125. byteBuffer.flip();
  126. channel.write(byteBuffer);
  127. channel.close();
  128. }
  129. }
  130. }

# 4.0 运行 HttpServer.main

  1. 2019-10-01 14:32:04.392 [main] INFO cn.shaiens.http.server.HttpServer - server start... bing port 10006

# 5.0 使用浏览器访问

在这里插入图片描述

# 6.0 查看http server 后台打印日志

在这里插入图片描述

# 7.0 打成jar启动

看我的csdn博客
IDEA 普通java工程打jar包
https://blog.csdn.net/JinglongSource/article/details/101063948

# 8.0 部署到阿里云服务器

上传到服务端, 进入目录之后, 执行

  1. nohup java -jar http_server.jar >/dev/null 2>&1 &

关闭http_server

  1. ps -ef | grep http_server | grep -v grep | awk '{print $2}' | xargs kill -9

# 9.0 使用logback日志框架可能出现的问题

我在本地测试的时候,发现一直没有出现日志文件,然后我使用win rar打开jar发现根目录下没有logback.xml文件,
因此需要把logback.xml复制进去才可以,具体是什么问题导致logback.xml打包的时候没进入到jar里边我也不知道, 如果你知道的话麻烦留言告诉一下, 感谢!!

在这里插入图片描述

# 10. 源码地址

https://gitee.com/jinglongcode/http_server.git

  标签   Http、  
mail csdn

作者  :  houyu [后宇]

程序员[ 后宇 ],是一个关注编程,热爱技术的开发者,热衷于 【Java后端】,【数据爬虫】,【大数据】领域。

在这里会一直记录着我成长的点点滴滴,毕竟好记性不如烂笔头,如果你在博客中有所收获,这也将是我毕生的荣幸。如有差池,还望指出更正!



评论


暂时没有评论哦~,你来评论个吧!!