Smarty4j是一个开源的模板引擎。没错,它就是著名的php模板引擎之Java移植版。
它特点就是将模板文件或者字符串编译成java类直接执行,所以效率比一般的模板解释的方式处理要快。它发展较晚,所以没有 velocity、FreeMarker 有名,人气也比 php 版本的模板引擎差好多。
但是它很快!
本着怀疑一切的态度,我自己进行了实测。在渲染1000遍一个简单的页面时(只引入几种简单的数据类型), Smarty4j 耗时 16 毫秒,velocity 耗时 63 毫秒,FreeMarker 则用了 109 毫秒。渲染5000遍时,Smarty4j 耗时 172 毫秒,velocity 耗时 328 毫秒,FreeMarker 则用了 390 毫秒。(以上测试均为多次测试后取的平均值)
复杂页面耗时相差更大。所以把项目里的 velocity 或 FreeMarker 换成 Smarty4j 可以使你的应用提速不少。
已经有网友将 Smarty4j 提供了插件集成到 struts2 中了。那么能不能把它集成到国产的小巧的 Nutz MVC 框架中呢?
翻看了 Nutz 的用户手册后发现这是件非常容易的事情。
下面我们就开始吧:
1、首先要实现视图适配器
非常简单:
- /**
- * Smarty4j 视图适配器
- * @author QinerG(QinerG@gmail.com)
- */
- public class SmartyViewMaker implements ViewMaker {
- public View make(Ioc ioc, String type, String value) {
- if("st".equalsIgnoreCase(type)){
- return new SmartyView(value);
- }
- return null;
- }
- }
2、然后再实现具体的视图解析器
- /**
- * 使用 Smarty4j 模板生成页面
- * @author QinerG(QinerG@gmail.com)
- */
- public class SmartyView extends AbstractPathView implements View {
- private final String ext = ".html";
- private static Engine engine = new Engine();//加载模板引擎
- public SmartyView(String dest) {
- super(dest);
- engine.setTemplatePath("");
- engine.setDebug(true);
- }
- /* 渲染页面
- * @see org.nutz.mvc.View#render(javax.servlet.http.HttpServletRequest, javax.servlet.http.HttpServletResponse, java.lang.Object)
- */
- public void render(HttpServletRequest req, HttpServletResponse resp,
- Object obj) throws Throwable {
- if ("".equals(engine.getTemplatePath())) {
- String realPath = req.getSession().getServletContext().getRealPath("/");
- engine.setTemplatePath(realPath);
- }
- String path = evalPath(req, obj);
- // 空路径,采用默认规则
- if (Strings.isBlank(path)) {
- path = Mvcs.getRequestPath(req);
- path = "WEB-INF"
- + (path.startsWith("/") ? "" : "/")
- + Files.renameSuffix(path, ext);
- }
- // 绝对路径 : 以 '/' 开头的路径不增加 '/WEB-INF'
- else if (path.charAt(0) == '/') {
- if (!path.toLowerCase().endsWith(ext))
- path += ext;
- }
- // 包名形式的路径
- else {
- path = "WEB-INF/" + path.replace('.', '/') + ext;
- }
- Template template = engine.getTemplate(path);
- Context ctx = new Context(); // 生成数据容器对象
- ctx.set("obj", obj);
- ctx.set("request", req);
- ctx.set("response", resp);
- ctx.set("session", req.getSession());
- template.merge(ctx, resp.getWriter());
- }
- }
好了,完成!
那么具体怎么用呢?
首先在主模块上声明使用 Smarty4j 适配器
- @Views({SmartyViewMaker.class})
- public class MainModule { }
然后在action上声明模板路径即可,如:
- @At("/index")
- @Ok("st:st.index")
- public void index() {
这个action的模板将对应 WEB-INF/st/index.html 文件。
当然模板路径也可以放在 WEB-INF 外面,如:
@Ok("st:abc.bbc") 或 @Ok("st:/abc/bbc")
对应的模板路径为:
abc/bbc.html