jfinal2.2 源码阅读日记(一)

jfinal 最大亮点就是零配置极速 WEB (通过对servlet的极薄封装实现的极速)+ ORM (独创Db + Record模式,灵活便利)

那极速 WEB 和 独创Db + Record模式 是怎么实现的呢?

所以看了下jfinal2.2的源码。

jfinal是通过过滤器拦截服务来实现路由的分发(web.xml 中配置的filter)。

jfnal 的核心拦截器为:com.jfinal.core.JFinalFilter.java

所以从这个类开始看。

// final 关键字修饰,表示不可被继承,实现了Filter接口

public final class JFinalFilter implements Filter {}

然后实现了 init 初始化方法 在服务启动过的过程中就由容器调用

现在来看看 init 方法中具体干了哪些事

public void init(FilterConfig filterConfig) throws ServletException {

// 利用反射初始化开发项目中实现了JfinalConfig类的子类,并且赋值给当前对象的jfinalConfig 变量

createJFinalConfig(filterConfig.getInitParameter("configClass"));


// 然后调用 Jfinal 中的初始化方法

1、配置项目路径

2、配置 Config 在config中初始化一些项目配置

2.1、配置常量Constant 然后初始化日志使框架日志与系统日期保持一致,默认使用log4j日志,如果项目没有使用则直接使用jdk日志

2.2、配置路由Route,

2.3、配置插件(Plugin),配置完成后启动插件,调用插件的start方法。

2.4、配置拦截器(Interceptor),

2.5、配置处理器(Handler)

3、初始化ActionMapping(配置请求路径对应的控制器对象)

4、初始化处理器,增加ActionHandler()处理器,并且和用户自定义处理器串联连接。

5、初始化 render 视图渲染组件

6、初始化 OreillyCos 文件上传处理插件

7、初始化 TokenManager(token管理器)

if (jfinal.init(jfinalConfig, filterConfig.getServletContext()) == false)

throw new RuntimeException("JFinal init error!");

// 获取jfinal处理器(在 jfinal.init函数中 调用 initHandler 函数 初始化的)

// 在初始化处理器的时候系统会增加一个 ActionHandler() 处理器,然后把这些处理器,和用户配置的处理器串联起来了,形成了一个链式的结构。

// 最后 jfinal.getHandler(); 返回的是用户配置的最后一个处理器,然后链接的下一个处理器是倒数第二个配置的,一直到 最后一个ActionHandler()

// 所以用户在每个配置的Handler中都得调用一下 next.handler(...) 函数,如果不调用则最后处理到达不了Controller。

handler = jfinal.getHandler();

// 取出用户配置的常量

constants = Config.getConstants();

// 编码设置为用户在常量中设置的编码,如果用户没有设置,则使用默认的UTF-8编码

encoding = constants.getEncoding();

// 调用 afterJFinalStart() 函数处理,用户在配置前自定义处理。

jfinalConfig.afterJFinalStart();

// 设置 contextPath 到此,拦截器的初始化工作完毕,增加jfinal的初始化工作也结束了。

// 啰嗦一句如果项目在tomcat的webapps中不是以ROOT命名的目录,则contextPath值为用户的项目目录名称

// 否则 contextPath 的值为 /

String contextPath = filterConfig.getServletContext().getContextPath();

contextPathLength = (contextPath == null || "/".equals(contextPath) ? 0 : contextPath.length());

}

接下来,就是拦截器懒觉请求,并且处理请求。

拦截器拦截到的每个请求,都会执行doFilter函数,这个函数是Filter接口中规定的必须实现的函数。

JFinalFilter 类中的 doFilter 函数实现代码如下,一句一句的来看。

public void doFilter(ServletRequest req, ServletResponse res, FilterChain chain) throws IOException, ServletException {

// 把 ServletRequest 和 ServletResponse 类型的请求与应答强制转换为HttpServletRequest,和HttpServletResponse

// 这段代码没什么就是为了方便使用HttpServletRequest里的一些函数。基本上每个filter 都会这么做。

HttpServletRequest request = (HttpServletRequest)req;

HttpServletResponse response = (HttpServletResponse)res;

// 设置请求编码,这个编码就是在初始化函数中,用户设置的或者系统默认的utf-8编码。

request.setCharacterEncoding(encoding);

String target = request.getRequestURI(); // 获取请求 uri

if (contextPathLength != 0)// contextPathLength 不为0则获取真正的 uri

target = target.substring(contextPathLength);

boolean[] isHandled = {false};  // 默认处理器没有处理过

try {

// 调用 处理器处理函数(下一篇文章,重点讲解)

handler.handle(target, request, response, isHandled);

}catch (Exception e) {

if (log.isErrorEnabled()) {

String qs = request.getQueryString();

log.error(qs == null ? target : target + "?" + qs, e);

}

}

// 如果 最后处理到达了。ActionHandler 并且成功处理过了,则isHandled[0]会被标记为true

// 则调用chain.doFilter(request, response); 放行请求,如果用户在 web.xml 中还配置了其他的过滤器

// 则用户还会继续调用其他过滤器来过滤请求,直到最后返回请求内容给用户,web容器关闭请求。

// 一个完整的请求就这样结束了,下一篇内容着重讲解 ActionHandler类中的 handle 函数(handler.handle(target, request, response, isHandled))

// 还有一个,isHandled使用数组类型,是因为数组类型使用的是引用传递,而boolean使用的是值传递。

if (isHandled[0] == false)

chain.doFilter(request, response);

}




jfinal2.2 源码阅读日记(一)_第1张图片

你可能感兴趣的:(jfinal2.2 源码阅读日记(一))