`

关于Freemarker生成静态html文件及中文乱码的问题

    博客分类:
  • Java
阅读更多
转自:http://blog.csdn.net/it_man/archive/2009/01/17/3808697.aspx

先看生成静态html文件:
FreeMarker允许Java servlet保持图形设计同应用程序逻辑的分离,这是通过在模板中密封HTML完成的。模板用servlet提供的数据动态地生成 HTML。模板语言是强大的直观的,编译器速度快,输出接近静态HTML页面的速度。

FreeMarker是一个模板引擎,一个基于模板生成文本输出的通用工具,使用纯Java编写
FreeMarker被设计用来生成HTML Web页面,特别是基于MVC模式的应用程序
虽然FreeMarker具有一些编程的能力,但通常由Java程序准备要显示的数据,由FreeMarker生成页面,通过模板显示准备的数据
FreeMarker不是一个Web应用框架,而适合作为Web应用框架一个组件
FreeMarker与容器无关,因为它并不知道HTTP或Servlet;FreeMarker同样可以应用于非Web应用程序环境
FreeMarker更适合作为Model2框架(如Struts)的视图组件,你也可以在模板中使用JSP标记库
FreeMarker是免费的

模板:
 

<html>
<head>
<title>查看文章: ${newsitem.title} </title>
</head>
<body>
<table width="100%" border="0" cellpadding="0" cellspacing="0" bgcolor="#FFFFFF">
<tr><td>
<table width="95%" border="0" align="center" cellpadding="2" cellspacing="6" >
	<tr> 
      <td height="10" align="left" colspan=2 ></td>
    </tr>
    <tr>
     	  <td align="left" width="538" >
     	  <strong>${newsitem.title}</strong> ( ${newsitem.addtime} )
          </td>
          <td align="right">
			<a href="index.jsp">返回</a>
			&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;
          </td>
    </tr>
    <tr>                      
         <td align="left" valign=top colspan=2>
         <hr align="left"  width="95%" size="1" noshade color="#cc0000" >
         </td>
    </tr>
    <tr>                      
      <td colspan=2>${newsitem.showContent}
	  </td>
    </tr>
</table>            
            
<br>
</td></tr>
</table>
</body>

</html>

 


代码:


import java.io.*;
import java.util.HashMap;
import java.util.Map;

import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import freemarker.template.*;
/*
* Created on 2005-4-7
 *
 */
 
/**
 * 测试FreeMarker.
 *
 * @author scud
 *
 */
public class FreeMarkerTest
{
 
        private final Log logger = LogFactory.getLog(getClass());
        private Configuration freemarker_cfg = null;
        public static void main(String[] args)
        {
            //@todo 自己的一个类
            NewsItem aItem = null;
           
            //@todo 装入新闻
            //NewsItem = loadNewsItem(1);
               
            FreeMarkerTest test = new FreeMarkerTest();
           
            Map root = new HashMap();
            root.put("newsitem", aItem);
            String sGeneFilePath = "/tpxw/";
            String sFileName = "1.htm";
            boolean bOK = test.geneHtmlFile("/tpxw/view.ftl",root, sGeneFilePath,sFileName);
        }
       
 
        /**
         * 获取freemarker的配置. freemarker本身支持classpath,目录和从ServletContext获取.
         */
        protected Configuration getFreeMarkerCFG()
        {
            if (null == freemarker_cfg)
            {
                // Initialize the FreeMarker configuration;
                // - Create a configuration instance
                freemarker_cfg = new Configuration();
 
                // - FreeMarker支持多种模板装载方式,可以查看API文档,都很简单:路径,根据Servlet上下文,classpath等等
               
                //htmlskin是放在classpath下的一个目录
                freemarker_cfg.setClassForTemplateLoading(this.getClass(), "/htmlskin");
            }
            return freemarker_cfg;
        }
 
        /**
         * 生成静态文件.
         *
         * @param templateFileName 模板文件名,相对htmlskin路径,例如"/tpxw/view.ftl"
         * @param propMap 用于处理模板的属性Object映射
         * @param htmlFilePath 要生成的静态文件的路径,相对设置中的根路径,例如 "/tpxw/1/2005/4/"
         * @param htmlFileName 要生成的文件名,例如 "1.htm"
         */
        public boolean geneHtmlFile(String templateFileName,Map propMap, String htmlFilePath,String htmlFileName )
        {
               //@todo 从配置中取得要静态文件存放的根路径:需要改为自己的属性类调用
            String sRootDir = "e:/webtest/htmlfile" ;
           
            try
            {
                Template t = getFreeMarkerCFG().getTemplate(templateFileName);
 
                //如果根路径存在,则递归创建子目录
                creatDirs(sRootDir,htmlFilePath);
               
                File afile = new File(sRootDir +"/" +htmlFilePath + "/" + htmlFileName);
                Writer out = new BufferedWriter(new OutputStreamWriter(new FileOutputStream(afile)));
                t.process(propMap, out);
            }
            catch (TemplateException e)
            {
                logger.error("Error while processing FreeMarker template " + templateFileName,e);
                return false;
            }
            catch (IOException e)
            {
                logger.error("Error while generate Static Html File " + htmlFileName,e);
                return false;
            }
            return true;
        }
       
       
        /**
         * 创建多级目录
         *
         * @param aParentDir String
         * @param aSubDir  以 / 开头
         * @return boolean 是否成功
         */
        public static boolean creatDirs(String aParentDir, String aSubDir)
        {
            File aFile = new File(aParentDir);
            if (aFile.exists())
            {
                File aSubFile = new File(aParentDir + aSubDir);
                if (!aSubFile.exists())
                {
                    return aSubFile.mkdirs();
                }
                else
                {
                    return true;
                }
            }
            else
            {
                return false;
            }
        }   
 
 
   
}


编码的问题

这里说的是编码的问题。项目使用的都是UTF-8编码,生成的文件在UTF-8编码下察看是乱码,而GBK正常(后来发现因为我用的中文操作系统所以用GBK查看正常)。
      当然我把Freemarker的配置都改成了UTF-8,我的模版文件也是UTF-8编码的。下面是原来的代码
   
public void setTemplatePath(Resource templatePath) {
        this.templatePath = templatePath;
        //设置freemarker的参数
        freemarkerCfg = new Configuration();
        try {
            freemarkerCfg.setDirectoryForTemplateLoading(this.templatePath.getFile());
            freemarkerCfg.setObjectWrapper(new DefaultObjectWrapper());

            //解决freemarker模板读取后出现乱码的问题
            freemarkerCfg.setDefaultEncoding("UTF-8");
        } catch (IOException ex) {
            throw new SystemException("No Directory found,please check you config.");
        }
    }
    /**
     * 生成静态文件
     * @param templateFileName 模版名称eg:(biz/order.ftl)
     * @param propMap 用于处理模板的属性Object映射 
     * @param htmlFilePath 要生成的静态文件的路径,相对设置中的根路径,例如 "/biz/2006/5/" 
     * @param htmlFileName 要生成的文件名,例如 "123.htm" 
     * @return
     */
    private boolean buildHtml(String templateFileName,Map propMap, String htmlFilePath,String htmlFileName){
        try {
            Template template = freemarkerCfg.getTemplate(templateFileName);
            template.setEncoding("UTF-8");
            //创建生成文件目录
            creatDirs(buildPath.getFilename(),htmlFilePath);
            File htmlFile = new File(buildPath + htmlFilePath + htmlFileName);
            Writer out = new BufferedWriter(new OutputStreamWriter(new FileOutputStream(htmlFile)));
            template.process(propMap,out);
            out.flush();
            return true;
        } catch (TemplateException ex){
            log.error("Build Error"+templateFileName,ex);
            return false;
        } catch (IOException e) {
            log.error("Build Error"+templateFileName,e);
            return false;
        }
        
    }


下面是修改之后的代码
   
/**
     * 生成静态文件
     * @param templateFileName 模版名称eg:(biz/order.ftl)
     * @param propMap 用于处理模板的属性Object映射 
     * @param htmlFilePath 要生成的静态文件的路径,相对设置中的根路径,例如 "/biz/2006/5/" 
     * @param htmlFileName 要生成的文件名,例如 "123.htm" 
     * @return
     */
    private boolean buildHtml(String templateFileName,Map propMap, String htmlFilePath,String htmlFileName){
        try {
            Template template = freemarkerCfg.getTemplate(templateFileName);
            template.setEncoding("UTF-8");
            //创建生成文件目录
            creatDirs(buildPath.getFilename(),htmlFilePath);
            File htmlFile = new File(buildPath + htmlFilePath + htmlFileName);
            Writer out = new BufferedWriter(new OutputStreamWriter(new FileOutputStream(htmlFile),"UTF-8"));
            template.process(propMap,out);
            out.flush();
            return true;
        } catch (TemplateException ex){
            log.error("Build Error"+templateFileName,ex);
            return false;
        } catch (IOException e) {
            log.error("Build Error"+templateFileName,e);
            return false;
        }
        
    }


原因就在于OutputStreamWriter的不同构造方法

OutputStreamWriter(OutputStream out)
          创建使用默认字符编码的 OutputStreamWriter。
OutputStreamWriter(OutputStream out, String charsetName)
          创建使用指定字符集的 OutputStreamWriter。

这个是中文JDK的文档说明,刚开始我使用默认的构造函数,所以使用了系统默认的编码,GBK,所以在生成静态文件的时候把UTF-8内容用GBK编码写入了,所以在UTF-8下浏览就有问题。

还有关于修改模版文件同样也要注意这个问题。
   
public String loadTemplate(String templateName) {
        StringBuffer sb = new StringBuffer();
        try {
            File file = new File(templatePath+"/"+templateName);
            BufferedReader reader = new BufferedReader(new InputStreamReader(new FileInputStream(file),"UTF-8"));
            String line = reader.readLine();
            while(line != null)    {
                sb.append(line);
                sb.append("\r\n");
                line = reader.readLine();
            }
            reader.close();
        } catch (IOException e) {
            throw new SystemException("Loading template Error:",e);
        }
        return sb.toString();
    }
    public void saveTemplate(String templateName, String templateContent) {
        try {
            File file = new File(templatePath + "/" + templateName);
            Writer out = new BufferedWriter(new OutputStreamWriter(new FileOutputStream(file),"UTF-8"));
            out.write(templateContent);
            out.flush();
            //扔出templatesave事件
            TemplateSaveEvent evt = new TemplateSaveEvent();
            evt.setTemplateName(templateName);
            dispatchTemplateEvent(evt);
        } catch (IOException e) {
            throw new SystemException("Write template Error",e);
        }
    }
分享到:
评论

相关推荐

    apache-freemarker-2.3.27-incubating-bin.tar.gz

    在描述中提到的“生成的word内容中有&等特殊字符也不需要转义处理”,这指的是FreeMarker引擎在处理模板时,能够自动处理特殊字符,避免了在生成Word文档时出现乱码或解析错误的问题。 FreeMarker的核心概念包括: ...

    用freemarker导出word

    1. 创建模板文件:使用`.ftl`扩展名创建一个Freemarker模板,模板文件中包含静态文本和动态占位符(如`${}`)。动态占位符用于插入数据模型中的值。 2. 设计模板结构:模板应按照Word文档的结构设计,例如标题、段落...

    FreemarkerWord.rar

    在生成Word文档时,保持文件的UTF-8编码可以避免字符乱码问题,保证内容的可读性。 在压缩包内的“WordText”可能是一个或者多个示例模板文件,或者是用于测试导出功能的文本文件。开发者可以查看这些文件,了解...

    Spring boot web 访问

    总结来说,"Spring Boot web 访问"涵盖了创建一个基础的Spring Boot Web应用,配置访问静态资源,使用Spring Boot DevTools进行快速开发,以及处理中文乱码问题。这些都是Spring Boot入门开发者需要掌握的基本技能,...

    Java用freemarker导出word例子

    `Freemarker`最初是为Web应用设计的,但现在也被广泛应用于生成静态文档、报告和其他类型的文本输出。 `Freemarker`是一个基于模板的系统,开发者可以创建一个FTL(FreeMarker Template Language)文件,其中包含...

    struts2+freemarker+log4j

    注意可能出现的乱码问题,这可能是由于字符编码设置不一致导致的,检查项目中各文件的编码设置,以及服务器的字符集配置,确保其与你的系统环境匹配。 通过这个项目,你可以学习到如何整合这三个强大的工具,理解...

    spring整合freemarker发送邮件例子

    Freemarker是一个模板引擎,它允许开发者通过模板语言将静态模板和动态数据结合,生成HTML或其他文本格式的输出。在Spring中,我们可以使用`FreeMarkerConfigurer`配置Freemarker,并设置模板目录,以便于加载和解析...

    SpringBoot+MybatisPlus+代码生成器整合示例

    * src/main/resources:项目的资源目录,用于存放项目的配置文件和静态资源。 pom.xml文件: 在pom.xml文件中,我们定义了项目的依赖关系和插件配置。例如,我们使用了SpringBoot的starter-parent作为项目的父项目...

    javapms门户管理系统 v1.4 beta

    9、修正评论数量不更新,评论“顶”数量保存不了的问题10、修正静态页生成功能,提升生成效率;11、修正广告模板逻辑判断的问题;12、修正栏目图片无法删除的问题;13、新增图片裁剪功能;14、新

    springboot中文参考指南

    - **模板引擎**:支持Thymeleaf、Freemarker等,用于生成动态HTML页面。 - **静态资源处理**:默认情况下,SpringBoot会处理`/static/`、`/public/`、`/resources/`目录下的静态资源。 5. **数据访问** - **JPA...

    Java Web之高级应用,以ppt的形式呈现

    Keytool是Java提供的工具,用于管理Keystore,创建和管理密钥对,导入和导出证书,以及生成CSR文件以申请服务器证书。 接着,自动登录和单点登录(SSO)是提高用户体验的重要功能。自动登录通常通过Cookie实现,...

    Java Web之高级应用

    8. **字符乱码问题**:Web开发中常见的字符编码问题,包括HTTP请求和响应的编码、HTML页面的charset声明、数据库存储和读取时的编码转换等。解决这类问题通常需要确保整个数据链路的编码一致,如设置正确的Content-...

    spirngmvc+hibernate

    描述中提到的“编码格式utf-8”表明该项目遵循UTF-8编码标准,这是Web开发中的常用字符编码,支持全球多种语言,避免了字符乱码问题。 **JSP(JavaServer Pages)** 和 **FreeMarker** 都是常见的视图技术,它们...

    windchill学习2

    在Web开发中,例如FreeMarker或Velocity,模板引擎允许开发者将动态数据插入到静态模板中,生成最终的HTML页面。 通过学习这些材料,你可以深入了解Windchill如何与Web开发技术协同工作,提升你的Java Web技能,并...

Global site tag (gtag.js) - Google Analytics