Java实现基于NIO的多线程Web服务器实例

所属分类: 软件编程 / java 阅读数: 59
收藏 0 赞 0 分享

代码地址:https://github.com/iyuanyb/webserver

实现了

静态、动态资源获取;

Cookie、Session、HTTP 长连接,及 Session 和 HTTP 长连接的定时清除;

类似 Spring MVC 的注解式编程,如 @RequestMapping @RequestParam 等,方法中可以根据参数名从前台获取数据,可以传递对象,也支持级联属性,如:

// GET /page?pageSize=10&pageNum=1 HTTP/1.1
@RequestMapping("/page")
String page(@RequestParam(value="pageSize", defaultValue="10") Integer pageSize, Integer pageNum) {...}
 
/**
 * POST /login HTTP/1.1
 * ...
 * user.name=admin&user.passwd=admin&user.data.val=ok
 * *******
 * User 类:String name; String passwd; Data data;
 * Data 类: String val;
 */
@RequestMapping("/login", method = HttpMethod.POST)
String login(User user) {...}

方法可以返回一个字符串表示模板路径,模板使用正则实现,仅可以从 request 和 session 域中获取属性值,如 ${request.user.id};

日志记录(使用 java.util.logging 内置日志记录器,自定义了日志格式):服务器运行相关日志(server-n.log),HTTP 请求日志(access-n.log)。

API 汇总:

@Controller
@RequestMapping
@RequestParam
@RequestHeader
@CookieValue
HttpRequest
HttpResponse
HttpSession
Cookie
HttpMethod

使用方法

导入 jar 包(见 release)即可使用相关注解,还需要在 classpath 下创建一个 webapp 目录,表示静态 web 资源的根路径。然后在主类的 main 方法中调用 BootStrap.run(),由于扫描控制器是通过遍历目录实现的,所以项目不支持打包,必须以 class 文件的形式发布。参数配置需要在 classpath 中提供一个 server-config.properties 配置文件即可,包含如下配置项:

# 服务器端口
PORT=80
# 存储日志文件的路径
LOG_FILE_STORAGE_PATH=E:\\
# 连接过期时间,单位毫秒
CONNECTION_EXPIRY_TIME=30000
# 清理过期连接的周期,单位毫秒
CONNECTION_CLEANING_CYCLE=30000
# Session 过期时间,单位毫秒
SESSION_EXPIRY_TIME=30000
# 清理过期 Session 的周期,单位毫秒
SESSION_CLEANING_CYCLE=30000
# 监听客户端读事件的线程数目
POLLER_THREAD_COUNT=2
# 处理具体请求的线程池的大小
REQUEST_PROCESSOR_THREAD_COUNT=4

演示

项目结构

EchoController.java

package com.test;
 
// 导包省略...
 
@Controller // 只有被 @Controller 标记的才会被认为是控制器
      // 支持在类上使用 @RequestMapping 注解,
public class EchoController {
 
  // 线程安全
  private static final DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss");
 
  @RequestMapping("/echo") // 映射到 "/echo"
  public String echo(HttpRequest request, @RequestParam(value = "msg", defaultValue = "输入为空") String msg) {
 
    LocalDateTime localDateTime = LocalDateTime.ofEpochSecond(request.getSession().getLastAccessedTime() / 1000, 0, ZoneOffset.ofHours(8));
    request.setAttribute("lastAccessedTime", localDateTime.format(formatter));
    request.setAttribute("msg", msg);
    return "test.html"; // 要渲染的的模板路径(classpath:webapp/test.html)
  }
 
  public static void main(String[] args) {
    BootStrap.run();
  }
}

test.html

<!DOCTYPE>
<html lang="en">
  <head>
    <title>Test</title>
  </head>
  <body>
    <p>Echo: ${request.msg}</p>
    <p>Last Accessed Time: ${request.lastAccessedTime}</p>
    <p><img src="img/girl.jpg" alt="girl" width="320" height="480"/></p>
  </body>
</html>

关于实例图片,GitHub暂时失效,后期脚本之家小编会补充进去。

更多精彩内容其他人还在看

JavaWeb项目部署到服务器详细步骤详解

这篇文章主要介绍了JavaWeb项目如何部署到服务器,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
收藏 0 赞 0 分享

IDEA基于支付宝小程序搭建springboot项目的详细步骤

这篇文章主要介绍了IDEA基于支付宝小程序搭建springboot项目的详细步骤,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
收藏 0 赞 0 分享

详解SpringBoot应用服务启动与安全终止

这篇文章主要介绍了SpringBoot应用服务启动与安全终止,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
收藏 0 赞 0 分享

Spring Boot启动及退出加载项的方法

这篇文章主要介绍了Spring Boot启动及退出加载项的方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
收藏 0 赞 0 分享

Spring Data Jpa 自动生成表结构的方法示例

这篇文章主要介绍了Spring Data Jpa 自动生成表结构的方法示例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
收藏 0 赞 0 分享

IDEA中osgi的开发应用指南详解

这篇文章主要介绍了IDEA中osgi的开发应用指南详解,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
收藏 0 赞 0 分享

详解用maven将dubbo工程打成jar包运行

这篇文章主要介绍了详解用maven将dubbo工程打成jar包运行,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
收藏 0 赞 0 分享

详解Java合并数组的两种实现方式

这篇文章主要介绍了Java合并数组的两种实现方式,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
收藏 0 赞 0 分享

使用Jenkins Pipeline自动化构建发布Java项目的方法

这篇文章主要介绍了使用Jenkins Pipeline自动化构建发布Java项目的方法,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
收藏 0 赞 0 分享

使用Maven配置Spring的方法步骤

这篇文章主要介绍了使用Maven配置Spring的方法步骤,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
收藏 0 赞 0 分享
查看更多