freemarker中Configuration的路徑設定問題
今天使用freemarker中Configuration的setClassForTemplateLoading方法遇到了載入模板目錄的一個小問題。
由於網上的其他論壇,部落格寫的有點亂,故記錄一下。
Freemarker是一個模板框架,主要是為了加快染速度而產生的。它與web容器無關,只要是關於模板生成一些程式碼的都可以使用它完成。
比如xml,Java程式碼的生成等。 其他類似的模板框架還有velocity。
本文主要講的是Freemarker的載入模板目錄問題,它的語法就不描述了。具體的語法可在官網下載參考手冊參考即可。
載入模板目錄方法
Freemarker提供了3種載入模板目錄的方法。 它使用Configuration類載入模板
3種方法分別是:
public void setClassForTemplateLoading(Class clazz, String pathPrefix);
public void setDirectoryForTemplateLoading(File dir) throws IOException;
public void setServletContextForTemplateLoading(Object servletContext, String path);
看名字也就知道了,分別基於類路徑、檔案系統以及Servlet Context。
第二,三種沒啥好說的。
第二種基於檔案系統。 比如載入/home/user/template下的模板檔案。
Configuration cfg = new Configuration(); cfg.setDirectoryForTemplateLoading(new File("/home/user/template")); cfg.getTemplate("Base.ftl");
這樣就獲得了/home/user/template/Base.ftl這個模板檔案。
第三種基於web project。 第二個引數是基於WebRoot下的。
比如: setServletContextForTemplateLoading(context, "/ftl") 就是 /WebRoot/ftl目錄。
第一種基於類路徑的方法有點小坑,其實看下原始碼程式碼就知道了。
比如 :
Configuration cfg = new Configuration(); cfg.setClassForTemplateLoading(FreemarkerUtil.class, "/template"); cfg.getTemplate("Base.ftl");
其實這個方法是根據類載入路徑來判斷的,最終會執行以下程式碼:
FreemarkerUtil.class.getClassLoader().getResource("/template/");
這裡注意一下第二個引數需要以 "/" 開頭。
其實我們看下原始碼就知道了:
可以看到,prefix如果最後1個字元不是 "/" 會預設加上。 但是第一個不是 "/" 字元 並不會自動加上。
最後獲得的URL方法:
我們可以看到URL是通過loaderClass.getResource(fullPath) 獲得的。
最終的具體生成程式碼是通過FMParser這個類生成的。 FMParser是使用JavaCC寫的,由於時間有限,就沒有具體研究了。