Java项目Freemark生成静态页面及语法

本文介绍如何使用Freemarker模板引擎结合Java程序生成HTML静态页面的方法,包括配置定时任务定期更新页面,以及如何在Eclipse中安装插件支持Freemarker语法高亮。

做门户网站有大量的页面 页面数据之多 每次请求都要查询数据库操作 性能 差  速度也慢的不得了  使用freemark生成静态页面

 

FreeMarker 是一个用Java编写的模板引擎,主要用来生成HTML Web页面,特别是基于MVC模式的应用程序。虽然FreeMarker具有一些编程的能力,但不像PHP,通常由Java程序准备要显示的数据,由 FreeMarker模板生成页面。 FreeMarker可以作为Web应用框架一个组件,但它与容器无关,在非Web应用程序环境也能工作的很好。 FreeMarker适合作为MVC的视图组件,还能在模板中使用JSP标记库。

ftl模板 + Map数据模型 = 输出html

首先编写使ftl模板可以生成html的代码  必需导入freemark包

public class CreateHTML {   
    private Configuration freemarker_cfg = null;   
  
    // private String sGeneFilePath = null;   
    // private String sGeneFileName = null;   
    // private String sTempPlateFilePath = null;   
  
    /**  
     * 创建多级目录  
     *   
     * @param path  
     *            String  
     * @return boolean 是否成功  
     */  
    private boolean creatDirs(String path) {   
        File aFile = new File(path);   
        if (!aFile.exists()) {   
            return aFile.mkdirs();   
        } else {   
            return true;   
        }   
    }   
  
    /**  
     * 生成静态文件.  
     *   
     * @param templateFileName  
     *            模板文件名,相对htmlskin路径,例如"/tpxw/view.ftl"  
     * @param propMap  
     *            用于处理模板的属性Object映射  
     * @param htmlFilePath  
     *            要生成的静态文件的路径,相对设置中的根路径,例如 "/tpxw/1/2005/4/"  
     * @param htmlFileName  
     *            要生成的文件名,例如 "1.htm"  
     * @param templateFilePath  
     *            模板路径  
     * @return boolean true代表生成文件成功  
     */  
    @SuppressWarnings("unchecked")   
    public void geneHtmlFile(String templateFileName, Map propMap,   
            String htmlFilePath, String htmlFileName, String templateFilePath) {   
  
        try {   
            Template t = this.getFreeMarkerCFG(templateFilePath).getTemplate(   
                    templateFileName);   
            // 如果根路径存在,则递归创建子目录   
            this.creatDirs(htmlFilePath);   
            File afile = new File(htmlFilePath + "/" + htmlFileName);   
            Writer out = new BufferedWriter(new OutputStreamWriter(   
                    new FileOutputStream(afile)));   
            t.process(propMap, out);   
            out.flush();   
            out.close();   
        } catch (TemplateException e) {   
            System.out.print(e.getMessage());   
        } catch (IOException e) {   
            System.out.print(e.getMessage());   
        } catch (Exception e) {   
            System.out.print(e.getMessage());   
        }   
    }   
  
    /**  
     *   
     * 获取freemarker的配置. freemarker本身支持classpath,目录和从ServletContext获取.  
     *   
     * @param templateFilePath  
     *            获取模板路径  
     * @return Configuration 返回freemaker的配置属性  
     * @throws Exception  
     */  
    private Configuration getFreeMarkerCFG(String templateFilePath)   
            throws Exception {   
        if (null == this.freemarker_cfg) {   
  
            this.freemarker_cfg = new Configuration();   
            try {   
                this.freemarker_cfg.setDirectoryForTemplateLoading(new File(   
                        templateFilePath));   
            } catch (Exception ex) {   
                throw ex;   
            }   
        }   
        return this.freemarker_cfg;   
    }   
  
}  

 

写一个定时器 定时生成静态页面

public class ContextListener implements ServletContextListener {   
  
    private static final long serialVersionUID = 1L;   
    private Timer timer = null;   
  
    public void contextDestroyed(ServletContextEvent event) {   
        timer.cancel();   
        event.getServletContext().log("定时器销毁");   
    }   
  
    public void contextInitialized(ServletContextEvent event) {   
        timer = new Timer(true);   
        event.getServletContext().log("定时器已启动");   
        // 6000  参数单位为毫秒  自动调用IndexTask类中的run方法   
        timer.schedule(new IndexTask(event), 0, 6000);   
    }   
} 

 

调用定时器中的程序  查询数据生成静态页面

public class IndexTask extends TimerTask {   
       ServletContextEvent context;   
    WebApplicationContext wac;   
       BlogService blogService;   
    public IndexTask(ServletContextEvent event) {   
        context = event;   
        ServletContext application = context.getServletContext();   
        wac = WebApplicationContextUtils.getWebApplicationContext(application);   
  
    }   
  
public void run() {   
        try {   
              //这里使用spring框架   
              blogService = (BlogService) wac.getBean("blogService");   
                      Map root = new HashMap();   
          // 會員積分排行  查询数据存放在root Map中 ftl模板可以使用userAlbum   
         //blogService.findByAlbum是写好的dao查询方法   
              root.put("userAlbum", blogService.findByAlbum(   
                    "queryTopUserByscore"));   
                    CreateHTML chtml = new CreateHTML();   
            chtml.geneHtmlFile("index.ftl", root, context.getServletContext()   
                    .getRealPath("/"), "index.html", context   
                    .getServletContext().getRealPath(   
                            "/WEB-INF/templates"));   
    } catch (Exception e) {   
            e.printStackTrace();   
        } finally {   
        }   
    }  

 

然后写ftl模板 我把ftl存放在/WEB-INF/templates/ 目录下 新建index.ftl 写要迭代的数据 比如我这里存放的数据名称为userAlbum

<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN"    
"http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">   
<html xmlns="http://www.w3.org/1999/xhtml">   
 <div id="Layer4_right" class="left">   
            <div id="Layer4_right_bg1">   
                <span>个人·博客</span>   
                <a href="/album/searchAlbum.action>更多>></a>   
            </div>   
            <div id="Layer4_right_content">   
      这里使用#list  迭代数据和foreach差不多    使用了#if  #else判断数据的长度         
              <div id="Layer4_right_content_Gray">   
                    <ul>   
                        <#list userAlbum as pro>   
            <li>   
           <a href="/album/searchAlbumDesc.action?shnAlbum.albumid=${pro.albumid!}">   
             <img src="/upload/photo/smallimage/${pro.images!}" alt="相片"  
                     width="102" height="101" />   
            </a>   
        <span>专辑:<a href="/album/searchAlbum.action?albumid=${pro.albumid!}">   
             <#if (pro.title?length gt 5)>${pro.title[0..3]!}..<#else>${pro.title!}</#if>   
            </a>   
                  </li>   
        </#list>   
                    </ul>   
                </div>   
            </div>  

 

先来解释一下freemaker的基本语法了,
<# ... > 中存放所有freemaker的内容,之外的内容全部原样输出。
<@ ... /> 是函数调用
两个定界符内的内容中,第一个符号表示指令或者函数名,其后的跟随参数。freemaker提供的控制包括如下:
<#if condition><#elseif condition><#else> 条件判断
<#list hash_or_seq as var> 遍历hash表或者collection(freemaker称作sequence)的成员
<#macro name param1 param2 ... ><#nested param> 宏,无返回参数
<#function name param1 param2><#return val> 函数,有返回参数
var?member_function(...) 用函数对var进行转换,freemaker称为build-ins。实际内部实现类似member_function(var, ...)
stringA[M .. N] 取子字符串,类似substring(stringA, M, N)
{key:value, key2:value2 ...} 直接定义一个hash表
[item0, item1, item2 ...] 直接定义一个序列
hash0[key0] 存取hash表中key对应的元素
seq0[5] 存取序列指定下标的元素
<@function1 param0 param1 ... /> 调用函数function1
<@macro0 param0 param1 ; nest_param0 nest_param1 ...> nest_body <
> 调用宏,并处理宏的嵌套
<#assign var = value > 定义变量并初始化
<#local var = value> 在 macro 或者 function 中定义局部变量并初始化
<#global var = value > 定义全局变量并初始化
${var} 输出并替换为表达式的值
<#visit xmlnode> 调用macro匹配xmlnode本身及其子节点
<#recurse xmlnode> 调用macro匹配xmlnode的子节点

  <#include "index.ftl" encoding="GBK"> 包含另外一个ftl模板标签

 

 

eclipse 安装Freemark插件 支持语法加亮  加亮的ftl模板 如下图

 

 

 

我的eclipse版本是galileo

打开菜单项 Help -> install new software... .

  1. 点击 Add Update Site... , 输入 " FreeMarker " 作为名字以及 "- http://download.jboss.org/jbosstools/updates/stable/galileo/ 作为更新的地址
  2. 选中复选框 "FreeMarker"
  3. 点击下一步或完成按钮根据提示完成插件的安装

安装完毕后该插件自动关联*.ftl文件,你也可以在对话框中进行自行设置。

 

另一篇贴子写的很不错

http://clq9761.iteye.com/blog/1341534

【语音分离】基于平均谐波结构建模的无监督单声道音乐声源分离(Matlab代码实现)内容概要:本文介绍了基于平均谐波结构建模的无监督单声道音乐声源分离方法,并提供了相应的Matlab代码实现。该方法通过对音乐信号中的谐波结构进行建模,利用音源间的频率特征差异,实现对混合音频中不同乐器或人声成分的有效分离。整个过程无需标注数据,属于无监督学习范畴,适用于单通道录音场景下的语音与音乐分离任务。文中强调了算法的可复现性,并附带完整的仿真资源链接,便于读者学习与验证。; 适合人群:具备一定信号处理基础和Matlab编程能力的高校学生、科研人员及从事音频处理、语音识别等相关领域的工程师;尤其适合希望深入理解声源分离原理并进行算法仿真实践的研究者。; 使用场景及目标:①用于音乐音频中人声与伴奏的分离,或不同乐器之间的分离;②支持无监督条件下的语音处理研究,推动盲源分离技术的发展;③作为学术论文复现、课程项目开发或科研原型验证的技术参考。; 阅读建议:建议读者结合提供的Matlab代码与网盘资料同步运行调试,重点关注谐波建模与频谱分解的实现细节,同时可扩展学习盲源分离中的其他方法如独立成分分析(ICA)或非负矩阵分解(NMF),以加深对音频信号分离机制的理解。
内容概要:本文系统介绍了新能源汽车领域智能底盘技术的发展背景、演进历程、核心技术架构及创新形态。文章指出智能底盘作为智能汽车的核心执行层,通过线控化(X-By-Wire)和域控化实现驱动、制动、转向、悬架的精准主动控制,支撑高阶智能驾驶落地。技术发展历经机械、机电混合到智能三个阶段,当前以线控转向、线控制动、域控制器等为核心,并辅以传感器、车规级芯片、功能安全等配套技术。文中还重点探讨了“智能滑板底盘”这一创新形态,强调其高度集成化、模块化优势及其在成本、灵活性、空间利用等方面的潜力。最后通过“2025智能底盘先锋计划”的实车测试案例,展示了智能底盘在真实场景中的安全与性能表现,推动技术从研发走向市场验证。; 适合人群:汽车电子工程师、智能汽车研发人员、新能源汽车领域技术人员及对智能底盘技术感兴趣的从业者;具备一定汽车工程或控制系统基础知识的专业人士。; 使用场景及目标:①深入了解智能底盘的技术演进路径与系统架构;②掌握线控技术、域控制器、滑板底盘等关键技术原理与应用场景;③为智能汽车底盘研发、系统集成与技术创新提供理论支持与实践参考。; 阅读建议:建议结合实际车型和技术标准进行延伸学习,关注政策导向与行业测试动态,注重理论与实车验证相结合,全面理解智能底盘从技术构想到商业化落地的全过程。
【顶级EI复现】计及连锁故障传播路径的电力系统 N-k 多阶段双层优化及故障场景筛选模型(Matlab代码实现)内容概要:本文介绍了名为《【顶级EI复现】计及连锁故障传播路径的电力系统 N-k 多阶段双层优化及故障场景筛选模型(Matlab代码实现)》的技术资源,重点围绕电力系统中连锁故障的传播路径展开研究,提出了一种N-k多阶段双层优化模型,并结合故障场景筛选方法,用于提升电力系统在复杂故障条件下的安全性与鲁棒性。该模型通过Matlab代码实现,具备较强的工程应用价值和学术参考意义,适用于电力系统风险评估、脆弱性分析及预防控制策略设计等场景。文中还列举了大量相关的科研技术支持方向,涵盖智能优化算法、机器学习、路径规划、信号处理、电力系统管理等多个领域,展示了广泛的仿真与复现能力。; 适合人群:具备电力系统、自动化、电气工程等相关背景,熟悉Matlab编程,有一定科研基础的研究生、高校教师及工程技术人员。; 使用场景及目标:①用于电力系统连锁故障建模与风险评估研究;②支撑高水平论文(如EI/SCI)的模型复现与算法验证;③为电网安全分析、故障传播防控提供优化决策工具;④结合YALMIP等工具进行数学规划求解,提升科研效率。; 阅读建议:建议读者结合提供的网盘资源,下载完整代码与案例进行实践操作,重点关注双层优化结构与场景筛选逻辑的设计思路,同时可参考文档中提及的其他复现案例拓展研究视野。
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值