手写spring(简易版)
本文版权归 远方的风lyh和博客园共有,欢迎转载,但须保留此段声明,并给出原文链接,谢谢合作,如有错误之处忘不吝批评指正!
理解Spring本质:
相信之前在使用spring的时候大家都配置web.xml文件、会配置spring,(如下)配置其实就是一个Servlet,DispatcherServlet源码中,它(父类)重写了 HttpServlet接口,所有的请求将交给 DispatcherServlet来处理了 <servlet>
<servlet-name>spring-mvc</servlet-name>
<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
<init-param>
<param-name>contextConfigLocation</param-name>
<param-value>WEB-INF/spring/spring-mvc.xml</param-value>
</init-param>
<load-on-startup>1</load-on-startup>
<async-supported>true</async-supported>
</servlet>
<servlet-mapping>
<servlet-name>spring-mvc</servlet-name>
<url-pattern>/</url-pattern>
</servlet-mapping>
配置
web.xm: 配置一个servlet 并接收所有请求
<!DOCTYPE web-app PUBLIC
"-//Sun Microsystems, Inc.//DTD Web Application 2.3//EN"
"http://java.sun.com/dtd/web-app_2_3.dtd" > <web-app>
<display-name>Archetype Created Web Application</display-name>
<servlet>
<servlet-name>MySpringMVC</servlet-name>
<servlet-class>cn.lyh.mySpring.MyDispatcherServlet</servlet-class>
<init-param>
<param-name>contextConfigLocation</param-name>
<param-value>context.properties</param-value>
</init-param>
<load-on-startup>1</load-on-startup>
</servlet> <servlet-mapping>
<servlet-name>MySpringMVC</servlet-name>
<url-pattern>/*</url-pattern>
</servlet-mapping>
</web-app>
context.properties:
#包扫描
scan.package=cn.lyh.mySpringTest
注解类
@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface MyAutowired {
String value() default "";
} @Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface MyController {
String value() default "";
} @Target({ElementType.TYPE, ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface MyRequestMapping {
String value() default "";
} @Target(ElementType.PARAMETER)
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface MyRequestParam {
/**
* 表示参数的别名,必填
* @return
*/
String value(); } @Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface MyResponseAdvice {
} @Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface MyResponseBody {
} @Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface MyService {
String value() default "";
}
MyDispacherServlet(核心实现):
MyDispacherServlet实现了HttpServlet 并复写doGet、doPost、init 方法
·
package cn.lyh.mySpring; import cn.lyh.mySpring.Handler.ResponseBodyHandler;
import cn.lyh.mySpring.annotation.*;
import com.alibaba.fastjson.JSONObject;
import com.alibaba.fastjson.serializer.SerializerFeature;
import org.apache.log4j.Logger; import javax.servlet.ServletConfig;
import javax.servlet.ServletException;
import javax.servlet.ServletRequest;
import javax.servlet.ServletResponse;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.File;
import java.io.IOException;
import java.io.InputStream;
import java.io.PrintWriter;
import java.lang.reflect.Field;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.lang.reflect.Parameter;
import java.net.URL;
import java.util.*; /***
*dispatcherServlet
* @author lyh
*/
public class MyDispatcherServlet extends HttpServlet {
/***配置***/
private Properties contextConfig = new Properties();
/***扫描的类名列表****/
private List<String> classNames = new ArrayList<>();
/***ioc容器 存放实例****/
private Map<String, Object> ioc = new HashMap<>();
/***url映射****/
private Map<String, Method> handlerMapping = new HashMap<>();
private static Logger logger = Logger.getLogger(MyDispatcherServlet.class);
/***返回处理器****/
private ResponseBodyHandler responseBodyHandler; @Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
doPost(req, resp);
} @Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
doDispatcherServlet(req, resp);
} /****
* 加载启动
* @param config
* @throws ServletException
*/
@Override
public void init(ServletConfig config) throws ServletException {
String contextConfigLocation = config.getInitParameter("contextConfigLocation");
try {
initMyDispatcherServlet(contextConfigLocation);
} catch (Exception e) {
e.printStackTrace();
throw new ServletException(e.getMessage());
}
} /***
* url请求映射到具体方法
* @param request
* @param response
*/
private void doDispatcherServlet(HttpServletRequest request, HttpServletResponse response) {
invoke(request, response);
} private void invoke(HttpServletRequest request, HttpServletResponse response) {
String queryUrl = request.getRequestURI();
queryUrl = queryUrl.replaceAll("/+", "/");
Method method = handlerMapping.get(queryUrl);
if (null == method) {
PrintWriter pw = null;
try {
response.setStatus(404);
logger.debug("request fail(404): " + request.getRequestURI());
pw = response.getWriter();
pw.print("404 not find -> " + request.getRequestURI());
pw.flush();
} catch (IOException e) {
e.printStackTrace();
} finally {
pw.close();
}
} else {
//todo method parameters need to deal
Object[] paramValues = getMethodParamAndValue(request, response, method);
try {
String controllerClassName = toFirstWordLower(method.getDeclaringClass().getSimpleName());
Object object = method.invoke(ioc.get(controllerClassName), paramValues);
if (object != null) {
if (method.isAnnotationPresent(MyResponseBody.class)) {
response.setHeader("content-type", "application/json;charset=UTF-8");
if (null == responseBodyHandler) {
object = JSONObject.toJSONString(object, SerializerFeature.WriteMapNullValue);
} else {
object = responseBodyHandler.equals(object);
}
}
response.getWriter().print(object);
logger.debug("request-> " + request.getRequestURI() + ", response success ->" + response.getStatus());
}
} catch (IllegalAccessException e) {
e.printStackTrace();
} catch (InvocationTargetException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
} }
} /****
* @MyRequestParam
* 参数解析 复制
* @注意: 参数解析暂不完整 int float long double boolean string
* 实体接收暂不支持
* @param request
* @param response
* @param method
* @return
*/
private Object[] getMethodParamAndValue(HttpServletRequest request, HttpServletResponse response, Method method) {
Parameter[] parameters = method.getParameters();
Object[] paramValues = new Object[parameters.length];
for (int i = 0; i < parameters.length; i++) { if (ServletRequest.class.isAssignableFrom(parameters[i].getType())) {
paramValues[i] = request;
} else if (ServletResponse.class.isAssignableFrom(parameters[i].getType())) {
paramValues[i] = response;
} else {
String bindingValue = parameters[i].getName();
if (parameters[i].isAnnotationPresent(MyRequestParam.class)) {
bindingValue = parameters[i].getAnnotation(MyRequestParam.class).value();
}
String paramValue = request.getParameter(bindingValue);
paramValues[i] = paramValue;
if (paramValue != null) {
if (Integer.class.isAssignableFrom(parameters[i].getType())) {
paramValues[i] = Integer.parseInt(paramValue);
} else if (Float.class.isAssignableFrom(parameters[i].getType())) {
paramValues[i] = Float.parseFloat(paramValue);
} else if (Double.class.isAssignableFrom(parameters[i].getType())) {
paramValues[i] = Double.parseDouble(paramValue);
} else if (Long.class.isAssignableFrom(parameters[i].getType())) {
paramValues[i] = Long.parseLong(paramValue);
} else if (Boolean.class.isAssignableFrom(parameters[i].getType())) {
paramValues[i] = Boolean.parseBoolean(paramValue);
}
}
}
}
return paramValues;
} /****
* 初始化
* @param contextConfigLocation
* @throws Exception
*/
private void initMyDispatcherServlet(String contextConfigLocation) throws Exception {
logger.info("-----------------------------mySpring init start-----------------------------------------");
logger.debug("doLoadConfig:" + contextConfigLocation);
//加载配置
doLoadConfig(contextConfigLocation);
//扫描 包扫描
logger.debug("scan:" + contextConfig.getProperty("scan.package"));
doScanner(contextConfig.getProperty("scan.package"));
//创建实体类、ioc
doInstance();
//注入 di
doAutowired();
//url 映射
initHandlerMapping(); } /***
* 注入
*/
private void doAutowired() {
if (ioc.isEmpty()) {
return;
}
for (Map.Entry<String, Object> entry : ioc.entrySet()) {
Object object = entry.getValue();
Field[] fields = object.getClass().getDeclaredFields();
for (Field filed : fields) {
if (filed.isAnnotationPresent(MyAutowired.class)) {
MyAutowired myAutowired = filed.getAnnotation(MyAutowired.class);
String key = filed.getType().getName();
String val = myAutowired.value();
if (val != null && "".equals(val.trim())) {
key = val.trim();
}
filed.setAccessible(true);
try {
filed.set(object, ioc.get(key));
} catch (IllegalAccessException e) {
e.printStackTrace();
}
} else {
continue;
}
}
}
} /***
* 初始化HandlerMapper
*/
private void initHandlerMapping() {
if (ioc.isEmpty()) {
return;
}
for (Map.Entry<String, Object> entry : ioc.entrySet()) {
Object object = entry.getValue();
Class<?> clazz = object.getClass();
if (clazz.isAnnotationPresent(MyController.class)) {
Method[] methods = clazz.getDeclaredMethods();
MyRequestMapping requestMapping = clazz.getAnnotation(MyRequestMapping.class);
String crlRequstMapping = requestMapping.value() == null ? "" : requestMapping.value();
for (Method method : methods) {
if (method.isAnnotationPresent(MyRequestMapping.class)) {
String url = ("/" + crlRequstMapping + "/" + method.getAnnotation(MyRequestMapping.class).value()).replaceAll("/+", "/");
// check request url must only
if (handlerMapping.containsKey(url)) {
logger.error("mapping request url:" + url + "is already exist! request url must only");
new Exception("mapping:" + url + "is already exist!");
}
handlerMapping.put(url, method);
logger.debug("mapping: " + url);
} else {
continue;
}
}
} }
} /***
* 加载配置文件
* @param contextConfigLocation
* @throws Exception
*/
private void doLoadConfig(String contextConfigLocation) throws Exception {
InputStream is = this.getClass().getClassLoader().getResourceAsStream(contextConfigLocation);
if (is == null) {
logger.error("config:" + contextConfigLocation + " not exist");
throw new Exception("config:" + contextConfigLocation + " not exist");
} else {
try {
contextConfig.load(is);
} catch (IOException e) {
e.printStackTrace();
} finally {
//关流
if (null != is) {
try {
is.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
}
} /****
* 包扫描
* @param packageName
* @throws Exception
*/
private void doScanner(String packageName) throws Exception {
if (packageName == null || packageName.length() == 0) {
throw new Exception("init scan is empty");
} URL url = this.getClass().getClassLoader().getResource("/" + packageName.replaceAll("\\.", "/"));
if (null != url) {
File dir = new File(url.getFile());
for (File file : dir.listFiles()) {
if (file.isDirectory()) {
//递归读取包
doScanner(packageName + "." + file.getName());
} else {
String className = packageName + "." + file.getName().replace(".class", "");
logger.debug("scan class find:" + className);
classNames.add(className);
}
}
} } /****
* ioc实例化
*/
private void doInstance() {
if (classNames.isEmpty()) {
return;
}
for (String className : classNames) {
try {
// @MyController instance
Class<?> clazz = Class.forName(className);
if (clazz.isAnnotationPresent(MyController.class)) {
logger.debug("MyController instance: " + clazz.getName());
ioc.put(toFirstWordLower(clazz.getSimpleName()), clazz.newInstance());
} else if (clazz.isAnnotationPresent(MyService.class)) {
//todo @MyService instance
// 1 以自己本类或者用户自定义别名为key
Object newInstance = clazz.newInstance();
String key = toFirstWordLower(clazz.getSimpleName());
logger.debug("MyService instance: " + clazz.getName());
MyService service = clazz.getAnnotation(MyService.class);
String value = service.value().trim();
if (!"".equals(value)) {
key = value;
}
if (!ioc.containsKey(key)) {
ioc.put(key, newInstance);
} else {
logger.error("MyService instance: " + service.value() + " is exist");
throw new Exception("MyService instance: " + service.value() + " is exist");
}
//2 以所继承的接口为 key
Class<?>[] interfaces = clazz.getInterfaces();
for (Class<?> interClazz : interfaces) {
ioc.put(interClazz.getName(), clazz.newInstance());
} } else if (clazz.isAnnotationPresent(MyResponseAdvice.class)) {
if (clazz.isAssignableFrom(ResponseBodyHandler.class)) {
if (null != responseBodyHandler) {
continue;
}
responseBodyHandler = (ResponseBodyHandler) clazz.newInstance();
} else {
logger.error("class+'" + clazz.getName() + "' must implement ResponseBodyHandler");
throw new Exception("class+'" + clazz.getName() + "' must implement ResponseBodyHandler");
}
} else {
continue;
} } catch (Exception e) {
e.printStackTrace();
continue;
}
}
} /**
* 把字符串的首字母小写
*
* @param name
* @return
*/
private String toFirstWordLower(String name) {
char[] charArray = name.toCharArray();
charArray[0] += 32;
return String.valueOf(charArray);
} }
TestController:
import cn.lyh.mySpring.annotation.*;
import cn.lyh.mySpringTest.domain.User;
import cn.lyh.mySpringTest.service.TestService; import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.util.*; @MyController
@MyRequestMapping("/test")
public class TestController {
@MyAutowired
private TestService testService; @MyRequestMapping("test1")
public String test1(@MyRequestParam("name") String name,
@MyRequestParam("sex") Integer sex,
HttpServletRequest request,
HttpServletResponse response) throws IOException { return "name=" + name + "sex=" + sex;
} @MyRequestMapping("test2")
public void test2() { } @MyRequestMapping("test3")
@MyResponseBody
public Map<String, Object> test3(@MyRequestParam("name") String name,
@MyRequestParam("sex") Integer sex,
HttpServletRequest request,
HttpServletResponse response) throws IOException {
Map<String, Object> result = new HashMap<>();
result.put("name", name);
result.put("sex", name); return result;
} @MyRequestMapping("test4")
@MyResponseBody
public User test4(@MyRequestParam("name") String name,
@MyRequestParam("sex") Integer sex,
HttpServletRequest request,
HttpServletResponse response) throws IOException {
User user = new User();
user.setName(name);
user.setId(sex); return user;
} @MyRequestMapping("test5")
@MyResponseBody
public List test5(@MyRequestParam("name") String name,
@MyRequestParam("sex") Integer sex,
HttpServletRequest request,
HttpServletResponse response) throws IOException {
List list = new ArrayList();
User user = new User();
user.setName(name);
user.setId(sex);
list.add(user); return list;
} @MyRequestMapping("test6")
@MyResponseBody
public List test5(HttpServletRequest request,
HttpServletResponse response) throws IOException {
List list = new ArrayList();
User user = new User();
user.setName(null);
user.setId(1);
list.add(user); return list;
} }
pom文件依赖:
<dependency>
<groupId>javax.servlet</groupId>
<artifactId>javax.servlet-api</artifactId>
<version>4.0.1</version>
<scope>provided</scope>
</dependency>
<dependency>
<groupId>org.slf4j</groupId>
<artifactId>slf4j-api</artifactId>
<version>1.6.6</version>
</dependency>
<dependency>
<groupId>org.slf4j</groupId>
<artifactId>slf4j-log4j12</artifactId>
<version>1.7.2</version>
</dependency>
<dependency>
<groupId>log4j</groupId>
<artifactId>log4j</artifactId>
<version>1.2.17</version>
</dependency>
<dependency>
<groupId>com.alibaba</groupId>
<artifactId>fastjson</artifactId>
<version>1.2.47</version>
<scope>compile</scope>
</dependency>
最后附上源码地址(MySpring Moudle)
手写spring(简易版)的更多相关文章
- 来,我们手写一个简易版的mock.js吧(模拟fetch && Ajax请求)
预期的mock的使用方式 首先我们从使用的角度出发,思考编码过程 M1. 通过配置文件配置url和response M2. 自动检测环境为开发环境时启动Mock.js M3. mock代码能直接覆盖g ...
- 手写Promise简易版
话不多说,直接上代码 通过ES5的模块化封装,向外暴露一个属性 (function(window){ const PENDING = 'pending'; const RESOLVED = 'fulf ...
- 手写一个简易版Tomcat
前言 Tomcat Write MyTomcat Tomcat是非常流行的Web Server,它还是一个满足Servlet规范的容器.那么想一想,Tomcat和我们的Web应用是什么关系? 从感性上 ...
- 手写Spring MVC
闲及无聊 又打开了CSDN开始看一看有什么先进的可以学习的相关帖子,这时看到了一位大神写的简历装X必备,手写Spring MVC. 我想这个东西还是有一点意思的 就拜读了一下大佬的博客 通读了一遍相关 ...
- 我手写的简易tomcat
前述 自己手写的简易的tomcat,实现了tomcat的基本响应功能,项目代码已经上传到我的Github,刚刚开始学习这里,当前还存在很多问题 项目简述及代码 当我们的Web运行的时候,从浏览器发出的 ...
- 一个老程序员是如何手写Spring MVC的
人见人爱的Spring已然不仅仅只是一个框架了.如今,Spring已然成为了一个生态.但深入了解Spring的却寥寥无几.这里,我带大家一起来看看,我是如何手写Spring的.我将结合对Spring十 ...
- 【Spring】手写Spring MVC
Spring MVC原理 Spring的MVC框架主要由DispatcherServlet.处理器映射.处理器(控制器).视图解析器.视图组成. 完整的Spring MVC处理 流程如下: Sprin ...
- 我是这样手写 Spring 的(麻雀虽小五脏俱全)
人见人爱的 Spring 已然不仅仅只是一个框架了.如今,Spring 已然成为了一个生态.但深入了解 Spring 的却寥寥无几.这里,我带大家一起来看看,我是如何手写 Spring 的.我将结合对 ...
- 《四 spring源码》利用TransactionManager手写spring的aop
事务控制分类 编程式事务控制 自己手动控制事务,就叫做编程式事务控制. Jdbc代码: Conn.setAutoCommite(false); // 设置手动控制事务 Hibern ...
随机推荐
- Desktop Central —— Windows 管理工具
Desktop Central —— Windows 管理工具 定期维护对于保持系统性能平稳必不可少.诸如磁盘检查.磁盘碎片整理程序之类的工具在系统维护中至关重要.因为管理员很难定期手动执行维护. D ...
- netcore程序部署及守护
一.程序发布 1.在本机编译无误的情况下,选择发布成文件系统.注意如果使用了swagger 需要将生成的xml文档说明复制到发版包里面.否则会报错.(可以在项目的csproj 中加入 <Prop ...
- Codeforces 873 简要题解
文章目录 A题 B题 C题 D题 E题 F题 传送门 A题 传送门 题意: 一个人要做nnn件事,时间花费分别为a1,a2,...,an,a1≤a2≤a3≤...≤ana_1,a_2,...,a_n, ...
- usb 枚举流程简介
1. 枚举是什么? 枚举就是从设备读取一些信息,知道设备是什么样的设备,如何进行通信,这样主机就可以根据这些信息来加载合适的驱动程序.调试USB设备,很重要的一点就是USB的枚举过程,只 ...
- JS入门经典第四章总结
charAt():该函数有一个参数,即选择哪一个位置上的参数.返回值就是该位置上的字符. charCodeAt():该函数有一个参数,即选择哪一个位置上的参数.返回值是该位置字符在Unicode字符集 ...
- Forward团队-爬虫豆瓣top250项目-最终程序
托管平台地址:https://github.com/xyhcq/top250 小组名称:Forward团队 小组成员合照: 程序运行方法: 在python中打开程序并运行:或者直接执行程序即可运行 程 ...
- css格式比较及选择器类型总结
在前端入门的前三天把网页制作过程中常用的一些标签和属性都认识和练习了一遍,能够做出简单模块的框架.就像老师说的网页制作就像建一栋大楼,html是砖和水泥,css是精装,js是完善各个功能.现在就开始进 ...
- jQuery-爱奇艺图片切换
<!DOCTYPE html><html> <head> <meta charset="UTF-8"> <title>& ...
- eclipse不支持sun.*包的问题处理
在项目中使用BASE64Decoder,eclipse的编辑器莫名报错, Multiple markers at this line - Access restriction: The type BA ...
- shell指令(一)
ubuntu桌面窗口下进入shell窗口:Ctrl + Alt + F2~F6: 退出shell窗口:Ctrl + Alt + F7:从UI中进入UI命令窗口,Ctrl + Alt +T shell ...