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);
}