摘要:只是暴露接口,配置信息最終保存在的靜態(tài)類中。整個項(xiàng)目只有一個,作為靜態(tài)類可以保證它是唯一的,而它的靜態(tài)成員也是整個項(xiàng)目中唯一的。至此的路由解析模塊就分析完了。
jFinal的路由解析是在JFinalFilter中做的,這個Filter也需要在web.xml中配置。JFinalFilter實(shí)現(xiàn)了javax.servlet.Filter接口,從這里也可以看出jFinal是基于Servlet的。JFinalFilter在初始化時負(fù)責(zé)初始化jFinal項(xiàng)目的配置(com.jfinal.core.Config)、路由表(Route)、映射表(ActionMapping)等;路由解析是在JFinalFilter的dofilter方法完成的。
關(guān)鍵詞: Route Handler Action ActionMapping
1. 項(xiàng)目配置分析jFinal的路由解析邏輯必須從jFinal的一般項(xiàng)目配置入手,配置的作用是為路由解析提供支持的。和一般Java Web MVC框架不同的是jFinal沒有采用xml配置的形式,但不是不需要配置,還是需要提供一個JFinalConfig的繼承實(shí)現(xiàn)類,實(shí)現(xiàn)configXXX方法來支持配置初始化,初始化的入口是JFinalFilter的init方法。
1.1 web.xmljFinal工程同樣需要web.xml配置文件,但是較其他MVC框架的web.xml文件內(nèi)容或許要簡單許多,除了配置welcome-file-list,只需要配置一個filter。
jfinal com.jfinal.core.JFinalFilter configClass com.app.common.Config jfinal /*
JFinalFilter是唯一需要配置的filter,只需要提供一個configClass參數(shù),它會在JFinalFilter的init方法中利用Class.forName(configClass).newInstance();被實(shí)例化。
1.2 JFinalConfig上面的configClass參數(shù)的值com.app.common.Config是項(xiàng)目定義的JFinalConfig的實(shí)現(xiàn)類,雖然整個項(xiàng)目沒有xml配置,但是這里就是,只不過是Java代碼的形式。
JFinalConfig只是暴露接口,配置信息最終保存在jFinal的靜態(tài)類com.jfinal.core.Config中。com.jfinal.core.Config類設(shè)計(jì)成不可以實(shí)例化,它定義的私有靜態(tài)成員變量可以保證唯一。JFinalConfig實(shí)現(xiàn)的接口就負(fù)責(zé)填充com.jfinal.core.Config成員變量。
在本文中會關(guān)注JFinalConfig的以下接口方法:
/** * Config route */ public abstract void configRoute(Routes me); /** * Config interceptor applied to all actions. */ public abstract void configInterceptor(Interceptors me); /** * Config handler */ public abstract void configHandler(Handlers me);1.3 com.jfinal.core.Config
在Config的成員變量中我們關(guān)注這幾個變量:
private static final Routes routes = new Routes(){public void config() {}}; private static final Interceptors interceptors = new Interceptors(); private static final Handlers handlers = new Handlers();
interceptors擁有所有的Interceptor,內(nèi)部結(jié)構(gòu)是List
handlers擁有所有的handler,內(nèi)部結(jié)構(gòu)是List
Routes定義了兩個容器,
private final Map> map = new HashMap >(); private final Map viewPathMap = new HashMap ();
對外提供了多個重載的add方法,用于增加路由,map和viewMap的鍵都是controllerKey。
關(guān)于Interceptor、Handler和Routes下文會繼續(xù)說明,我們先來看看自定義的JFinalConfig實(shí)現(xiàn)類com.app.common.Config做了什么事情。即是我們關(guān)注的JFinalConfig的抽象方法實(shí)現(xiàn)。
package com.app.common; public class Config extends JFinalConfig { @Override public void configConstant(Constants me) { //配置默認(rèn)View類型 } @Override public void configRoute(Routes me) { me.add("/api/user", UserController.class); me.add("/admin/user", ManagerController.class, "/admin"); me.add("/admin/index", IndexController.class, "/admin"); //... } @Override public void configPlugin(Plugins me) { //配置數(shù)據(jù)庫連接 //配置數(shù)據(jù)表和pojo映射 } @Override public void configInterceptor(Interceptors me) { //配置攔截器 } @Override public void configHandler(Handlers me) { //配置Handler //這里沒有配置,JFinal.init()方法也會添加一個ActionHandler } }
在configRoute實(shí)現(xiàn)中我們使用了兩種Routes.add()方法,向Routes添加了三個Controller。jFinal的路由是REST風(fēng)格的,這里
me.add("/api/user", UserController.class);的意思大概是請求/api/user時會交給UserController來處理。具體地看下文JFinalFilter的doFilter方法小節(jié)。
這里抽象實(shí)現(xiàn)方法什么時候被調(diào)用具體看JFinalFilter的init方法小節(jié)。
在進(jìn)入JFinalFilter的init和doFilter方法之前,我們將上面的提到的幾個概念梳理一下。
2.1 RoutesRoutes是jFinal的路由,有兩個路由映射的容器,請求路徑到Controller的映射和請求路徑到渲染頁面的映射。
Routes在項(xiàng)目中是作為com.jfinal.core.Config的成員變量出現(xiàn)的,負(fù)責(zé)維護(hù)jFinal項(xiàng)目的路由映射。整個jFinal項(xiàng)目只有一個com.jfinal.core.Config,作為靜態(tài)類可以保證它是唯一的,而它的靜態(tài)成員也是整個項(xiàng)目中唯一的。routes就是其中之一。
Routes提供了多個重載的add方法,我們來看看我使用到的其中兩個。
/** * Add route * @param controllerKey A key can find controller * @param controllerClass Controller Class * @param viewPath View path for this Controller */ public Routes add(String controllerKey, Class extends Controller> controllerClass, String viewPath) { //很多很多的corner check //處理controllerKey的前綴,前綴加SLASH / //處理viewPath的前綴和后綴,都加上SLASH //如果viewPath的根路徑baseViewPath不為空則在viewPath前拼接 map.put(controllerKey, controllerClass); viewPathMap.put(controllerKey, viewPath); return this;//為了鏈?zhǔn)綄懛?}
另外一個
public Routes add(String controllerkey, Class extends Controller> controllerClass) { return add(controllerkey, controllerClass, controllerkey); }
其實(shí)調(diào)用了上面的方法的。
public Routes add(String controllerKey, Class extends Controller> controllerClass, String viewPath) { }
一般使用過程中通過controllerKey找到Controller,這非常容易理解。而通過controllerKey在viewPathMap中找到viewPath,這個是用渲染頁面是使用的路徑,例如:
請求/api/user/edit執(zhí)行成功后渲染到/api/user/edit.jsp頁面。
一般我們定義controllery為/api/user,viewPath為/api/user/或者其他,而/edit和edit.jsp映射是約定好的。(但并不是直接映射的。)
最終的結(jié)果我們可以得到兩個配置好的map和viewPathMap。
與Routes同理,Interceptors也作為com.jfinal.core.Config的成員變量出現(xiàn)的,它本身是一個List容器,記錄的是項(xiàng)目的所有攔截器。在示例中com.app.common.Config并沒有設(shè)置攔截器,在實(shí)現(xiàn)的configInterceptor方法中并沒有做什么事情,如有需要我們可以調(diào)用Interceptors的add方法添加全局的攔截器。
final public class Interceptors { private final List2.3 HandlerinterceptorList = new ArrayList (); public Interceptors add(Interceptor globalInterceptor) { if (globalInterceptor != null) this.interceptorList.add(globalInterceptor); return this; } //... }
在com.jfinal.core.Config有一個成員變量handlers,記錄的是項(xiàng)目所有的Handler,可以向它添加Handler。在示例中com.app.common.Config實(shí)現(xiàn)的configHandler方法中也沒有做具體的配置。
Handler有一個成員變量nextHandler指向下一個Handler,這樣可以用鏈表形式將所有的Handler連接起來。Handler鏈表的頭節(jié)點(diǎn)最后保存在JFinal的handler變量,見JFinalFilter的init方法小節(jié)。這里先提一下如何獲得鏈表的頭節(jié)點(diǎn):在HandlerFacotry中提供的getHandler方法傳入原有的所有Handler和一個新的Handler,最終構(gòu)造一條Handler鏈,新的Handler被添加到鏈表的尾部,最終返回頭節(jié)點(diǎn)。
/** * Build handler chain */ public static Handler getHandler(ListhandlerList, Handler actionHandler) { Handler result = actionHandler; for (int i=handlerList.size()-1; i>=0; i--) { Handler temp = handlerList.get(i); temp.nextHandler = result; result = temp; } return result; }
Handler鏈的使用是在JFinalFilter的doFilter方法中,下文會提及。
2.4 ActionMappingActionMapping負(fù)責(zé)將Routes和Interceptors組織起來,整合后的結(jié)果存到在ActionMapping的mapping成員變量(Map
具體過程則是,遍歷Routes所有Controller、遍歷Controller所有method,將類級別(Controller)和方法(method)級別對應(yīng)的key或者名字連接起來作為鍵actionKey,將類級別(Controller)和方法(method)級別對應(yīng)的Interceptor整合計(jì)算后得到Action的攔截器數(shù)組actionInters。
最后用于實(shí)例化Action需要的變量如下所示:
new Action(controllerKey, actionKey, controllerClass, method, methodName, actionInters, routes.getViewPath(controllerKey));
具體的可以參照ActionMapping的buildActionMapping方法。
void buildActionMapping() { mapping.clear(); SetexcludedMethodName = buildExcludedMethodName(); InterceptorBuilder interceptorBuilder = new InterceptorBuilder(); Interceptor[] defaultInters = interceptors.getInterceptorArray(); interceptorBuilder.addToInterceptorsMap(defaultInters); for (Entry > entry : routes.getEntrySet()) { Class extends Controller> controllerClass = entry.getValue(); Interceptor[] controllerInters = interceptorBuilder.buildControllerInterceptors(controllerClass); Method[] methods = controllerClass.getMethods(); for (Method method : methods) { String methodName = method.getName(); if (!excludedMethodName.contains(methodName) && method.getParameterTypes().length == 0) { Interceptor[] methodInters = interceptorBuilder.buildMethodInterceptors(method); Interceptor[] actionInters = interceptorBuilder.buildActionInterceptors(defaultInters, controllerInters, controllerClass, methodInters, method); String controllerKey = entry.getKey(); ActionKey ak = method.getAnnotation(ActionKey.class); if (ak != null) { String actionKey = ak.value().trim(); if ("".equals(actionKey)) throw new IllegalArgumentException(controllerClass.getName() + "." + methodName + "(): The argument of ActionKey can not be blank."); if (!actionKey.startsWith(SLASH)) actionKey = SLASH + actionKey; if (mapping.containsKey(actionKey)) { warnning(actionKey, controllerClass, method); continue; } Action action = new Action(controllerKey, actionKey, controllerClass, method, methodName, actionInters, routes.getViewPath(controllerKey)); mapping.put(actionKey, action); } else if (methodName.equals("index")) { String actionKey = controllerKey; Action action = new Action(controllerKey, actionKey, controllerClass, method, methodName, actionInters, routes.getViewPath(controllerKey)); action = mapping.put(actionKey, action); if (action != null) { warnning(action.getActionKey(), action.getControllerClass(), action.getMethod()); } } else { String actionKey = controllerKey.equals(SLASH) ? SLASH + methodName : controllerKey + SLASH + methodName; if (mapping.containsKey(actionKey)) { warnning(actionKey, controllerClass, method); continue; } Action action = new Action(controllerKey, actionKey, controllerClass, method, methodName, actionInters, routes.getViewPath(controllerKey)); mapping.put(actionKey, action); } } } } // support url = controllerKey + urlParas with "/" of controllerKey Action actoin = mapping.get("/"); if (actoin != null) mapping.put("", actoin); }
這個方法會是整篇文章提到的最復(fù)雜的方法,所以這里全部列出。
主要的邏輯是拼接${controlerKey}/methodName作為actionKey,${controllerKey}類似/api/user是我們在JFinalConfig實(shí)現(xiàn)類中添加的。actionKey最后會和請求的URL比較,匹配時就返回其對應(yīng)的Action。拼接actionKey的過程中有兩個需要注意的地方,一個是Controller的方法不能有參數(shù),一個是如果方法名是index就將controllerKey作為actionKey,即是如果請求是/api/user最終調(diào)用的是UserController.index()。最后也做了請求是/的支持。
另外一個重要的是邏輯是整合計(jì)算Action的最終的攔截器數(shù)組actionInters。jFinal提供了Before注解的形式來在Controller類級別和method方法級別引入Interceptor,還有ClearInterceptor作為規(guī)則用于排除上層層次的Interceptor。這些細(xì)節(jié)就不展開了。
2.4 ActionMapping已經(jīng)提到了Action,這里提一下Action是怎么調(diào)用的。我們注意到實(shí)例化Action時傳入了很多參數(shù)。
new Action(controllerKey, actionKey, controllerClass, method, methodName, actionInters, routes.getViewPath(controllerKey));
其中controllerClass可以提供實(shí)例化一個Controller,methodName可以確定調(diào)用Controller的哪個方法,actionInters可以在調(diào)用Controller方法前執(zhí)行攔截過濾等,攔截過濾后再回到Action去調(diào)用真正的methodName方法。整個調(diào)用過程是ActionInvocation封裝完成的,具體細(xì)節(jié)就不展開了。
3 JFinalFilter init最后來看看兩個重要的流程。直接上代碼
public void init(FilterConfig filterConfig) throws ServletException { //實(shí)例化JFinalConfig實(shí)現(xiàn)類 createJFinalConfig(filterConfig.getInitParameter("configClass")); //配置初始化 //初始化Handler ActionMapping if (jfinal.init(jfinalConfig, filterConfig.getServletContext()) == false) throw new RuntimeException("JFinal init error!"); //Handler鏈頭節(jié)點(diǎn) handler = jfinal.getHandler(); constants = Config.getConstants(); encoding = constants.getEncoding(); jfinalConfig.afterJFinalStart(); String contextPath = filterConfig.getServletContext().getContextPath(); contextPathLength = (contextPath == null || "/".equals(contextPath) ? 0 : contextPath.length()); }
createJFinalConfig是JFinalFilter內(nèi)部方法,filterConfig.getInitParameter("configClass")是從web.xml獲得配置的JFinalConfig實(shí)現(xiàn)類,目的是實(shí)例化JFinalConfig。
private void createJFinalConfig(String configClass) { if (configClass == null) throw new RuntimeException("Please set configClass parameter of JFinalFilter in web.xml"); try { Object temp = Class.forName(configClass).newInstance(); if (temp instanceof JFinalConfig) jfinalConfig = (JFinalConfig)temp; else throw new RuntimeException("Can not create instance of class: " + configClass + ". Please check the config in web.xml"); } catch (InstantiationException e) { throw new RuntimeException("Can not create instance of class: " + configClass, e); } catch (IllegalAccessException e) { throw new RuntimeException("Can not create instance of class: " + configClass, e); } catch (ClassNotFoundException e) { throw new RuntimeException("Class not found: " + configClass + ". Please config it in web.xml", e); } }
接下來是調(diào)用
jfinal.init(jfinalConfig, filterConfig.getServletContext())
這部分是在JFinal類中完成的。
boolean init(JFinalConfig jfinalConfig, ServletContext servletContext) { this.servletContext = servletContext; this.contextPath = servletContext.getContextPath(); initPathUtil(); //調(diào)用JFinalConfig實(shí)現(xiàn)類的configXXX方法 Config.configJFinal(jfinalConfig); // start plugin and init logger factory in this method constants = Config.getConstants(); //初始化actionMapping initActionMapping(); //新建一個ActionHandler并且構(gòu)造一條Handler鏈并保存頭節(jié)點(diǎn) initHandler(); initRender(); initOreillyCos(); initI18n(); initTokenManager(); return true; }
這個方法中開始做整個項(xiàng)目的配置初始化,具體可以看Config.configJFinal(jfinalConfig)的實(shí)現(xiàn)。
/* * Config order: constant, route, plugin, interceptor, handler */ static void configJFinal(JFinalConfig jfinalConfig) { jfinalConfig.configConstant(constants); initLoggerFactory(); jfinalConfig.configRoute(routes); jfinalConfig.configPlugin(plugins); startPlugins(); // very important!!! jfinalConfig.configInterceptor(interceptors); jfinalConfig.configHandler(handlers); }
基本就是調(diào)用JFinalConfig的configXXX,具體如何做可以參考前面Routes、Interceptors和Handler小節(jié)。
接著來關(guān)注initActionMapping部分邏輯。
private void initActionMapping() { actionMapping = new ActionMapping(Config.getRoutes(), Config.getInterceptors()); actionMapping.buildActionMapping(); }
基本就是調(diào)用ActionMapping的buildActionMapping方法了,buildActionMapping可以參考前面ActionMapping小節(jié)。
最后關(guān)注initHandler部分邏輯。
private void initHandler() { Handler actionHandler = new ActionHandler(actionMapping, constants); handler = HandlerFactory.getHandler(Config.getHandlers().getHandlerList(), actionHandler); }
關(guān)于HandlerFactory的使用可以參考Handler小節(jié)。
執(zhí)行完JFinalFilter的init就為整個項(xiàng)目的路由解析做好了準(zhǔn)備了。
還是直接上代碼
public void doFilter(ServletRequest req, ServletResponse res, FilterChain chain) throws IOException, ServletException { HttpServletRequest request = (HttpServletRequest)req; HttpServletResponse response = (HttpServletResponse)res; request.setCharacterEncoding(encoding); //獲得請求URL String target = request.getRequestURI(); if (contextPathLength != 0) //切掉上下文路徑,contextPathLength是上下文路徑的長度 target = target.substring(contextPathLength); boolean[] isHandled = {false}; try { //Handler鏈調(diào)用 handler.handle(target, request, response, isHandled); } catch (Exception e) { if (log.isErrorEnabled()) { String qs = request.getQueryString(); log.error(qs == null ? target : target + "?" + qs, e); } } if (isHandled[0] == false) chain.doFilter(request, response); }
這里的handler是JFinal.initHanlder()方法獲得Handler鏈的頭節(jié)點(diǎn),如果整個項(xiàng)目沒有其他Handler,頭節(jié)點(diǎn)應(yīng)該是一個ActionHandler類型實(shí)例。
接下來看ActionHandler.handle方法
/** * handle * 1: Action action = actionMapping.getAction(target) * 2: new ActionInvocation(...).invoke() * 3: render(...) */ public final void handle(String target, HttpServletRequest request, HttpServletResponse response, boolean[] isHandled) { if (target.indexOf(".") != -1) { return ; } isHandled[0] = true; String[] urlPara = {null}; Action action = actionMapping.getAction(target, urlPara); if (action == null) { if (log.isWarnEnabled()) { String qs = request.getQueryString(); log.warn("404 Action Not Found: " + (qs == null ? target : target + "?" + qs)); } renderFactory.getErrorRender(404).setContext(request, response).render(); return ; } try { Controller controller = action.getControllerClass().newInstance(); controller.init(request, response, urlPara[0]); if (devMode) { boolean isMultipartRequest = ActionReporter.reportCommonRequest(controller, action); new ActionInvocation(action, controller).invoke(); if (isMultipartRequest) ActionReporter.reportMultipartRequest(controller, action); } else { new ActionInvocation(action, controller).invoke(); } Render render = controller.getRender(); if (render instanceof ActionRender) { String actionUrl = ((ActionRender)render).getActionUrl(); if (target.equals(actionUrl)) throw new RuntimeException("The forward action url is the same as before."); else handle(actionUrl, request, response, isHandled); return ; } if (render == null) render = renderFactory.getDefaultRender(action.getViewPath() + action.getMethodName()); render.setContext(request, response, action.getViewPath()).render(); } catch (RenderException e) { if (log.isErrorEnabled()) { String qs = request.getQueryString(); log.error(qs == null ? target : target + "?" + qs, e); } } catch (ActionException e) { int errorCode = e.getErrorCode(); if (errorCode == 404 && log.isWarnEnabled()) { String qs = request.getQueryString(); log.warn("404 Not Found: " + (qs == null ? target : target + "?" + qs)); } else if (errorCode == 401 && log.isWarnEnabled()) { String qs = request.getQueryString(); log.warn("401 Unauthorized: " + (qs == null ? target : target + "?" + qs)); } else if (errorCode == 403 && log.isWarnEnabled()) { String qs = request.getQueryString(); log.warn("403 Forbidden: " + (qs == null ? target : target + "?" + qs)); } else if (log.isErrorEnabled()) { String qs = request.getQueryString(); log.error(qs == null ? target : target + "?" + qs, e); } e.getErrorRender().setContext(request, response).render(); } catch (Exception e) { if (log.isErrorEnabled()) { String qs = request.getQueryString(); log.error(qs == null ? target : target + "?" + qs, e); } renderFactory.getErrorRender(500).setContext(request, response).render(); } }
render部分暫且不看。
從作者的代碼注釋中可以看出這個handle方法的主要邏輯。
我們就看其中兩個。
* 1: Action action = actionMapping.getAction(target) * 2: new ActionInvocation(...).invoke()
target是減去了contextPath部分的請求路徑,在ActionMapping.getAction(target)方法中將與ActionMapping維護(hù)的mapping表中的所有actionKey作比較,如果匹配就獲得一個Action。
看下實(shí)現(xiàn)代碼
/** * Support four types of url * 1: http://abc.com/controllerKey ---> 00 * 2: http://abc.com/controllerKey/para ---> 01 * 3: http://abc.com/controllerKey/method ---> 10 * 4: http://abc.com/controllerKey/method/para ---> 11 */ Action getAction(String url, String[] urlPara) { Action action = mapping.get(url); if (action != null) { return action; } // -------- int i = url.lastIndexOf(SLASH); if (i != -1) { action = mapping.get(url.substring(0, i)); urlPara[0] = url.substring(i + 1); } return action; }
簡單解釋下,這個方法支持四種形式的請求,見注釋。
首先嘗試mapping.get(url),
如果結(jié)果不為空,結(jié)合前面ActionMapping.buildActionMapping(),
我們知道這時/controllerKey或者/controllery/method匹配到了。
進(jìn)一步截取并嘗試mapping.get(url.substring(0,i))
即將/controllerKey/para和/controllerKey/method/para減去/para再執(zhí)行匹配。
para用urlPara[0]收集起來。
最后不管是否匹都配返回。
回到ActionHandler.handle()方法,用獲得的Action進(jìn)行調(diào)用處理請求。
new ActionInvocation(action, controller).invoke();
至此,jFinal的路由解析模塊就分析完了。
5 Thanks以上用于分析的jFinal版本是jfinal-1.8-bin-with-src.jar。
感謝轉(zhuǎn)神提供的案例
感謝豪的提點(diǎn)和幫助
支持一下文章作者吧!
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/65316.html
摘要:在中,通過對所有的類進(jìn)行過濾。在這個類中,均以成員變量的形式存在。中放置的是和的鍵值對。在中主要是調(diào)用了方法。 在jfinal中,通過JFinalFilter對所有的類進(jìn)行過濾。 以下是路由的調(diào)用關(guān)系(我在調(diào)用關(guān)系旁邊做了標(biāo)記,會貼出具體的代碼和解釋): -1- Config: Routes -2- Interceptors Handlers -3- publi...
摘要:是國產(chǎn)的框架,由五大部分組成。本文通過一個例子上手,旨在熟悉中各組件的用法。指的是表名,指的是主鍵數(shù)據(jù)庫連接池使用的是,還支持。默認(rèn)訪問方法,這點(diǎn)類似于如果之前有基礎(chǔ),上手會非常快。映射在上使用了校驗(yàn)攔截器,使用了權(quán)限攔截器。 JFinal是國產(chǎn)的MVC框架,由 Handler、Interceptor、Controller、Render、Plugin 五大部分組成。本文通過一個例子上手...
摘要:本文使用環(huán)境如何使用導(dǎo)入包將官網(wǎng)提供的包導(dǎo)入項(xiàng)目配置文件在項(xiàng)目配置文件中配置如下內(nèi)容即可生效生成日志文件運(yùn)行項(xiàng)目后將在根目錄下生成的文件。 本文使用環(huán)境 win7 Idea 14.1.4 jfinal 2.0 1.jfinal如何使用log4j a.導(dǎo)入jar包 將官網(wǎng)提供的log4j.jar包導(dǎo)入項(xiàng)目 b.配置文件 在項(xiàng)目配置文件(project/src/log4j.prope...
摘要:使用可以極度方便的使用,該插件不僅提供了豐富的,而且還同時支持多服務(wù)端。擁有超高的性能,豐富的數(shù)據(jù)結(jié)構(gòu),天然支持?jǐn)?shù)據(jù)持久化,是目前應(yīng)用非常廣泛的數(shù)據(jù)庫。 預(yù)設(shè) Ubuntu 上 安裝 redis 參見http://segmentfault.com/a/1190000004109484 概述 jfinal 2.0 中已集成RedisPlugin插件。 RedisPlugin 是支持 ...
閱讀 2673·2021-11-11 16:54
閱讀 3671·2021-08-16 10:46
閱讀 3451·2019-08-30 14:18
閱讀 3045·2019-08-30 14:01
閱讀 2731·2019-08-29 14:15
閱讀 2016·2019-08-29 11:31
閱讀 3093·2019-08-29 11:05
閱讀 2597·2019-08-26 11:54