邮件模板

让我们来看看邮件模板的格式。模板是XML文件,它包含一个根元素和一系列根的子元素。根元素是。必要的子元素是, , 和 。可选的子元素是 , , 和 。如果你使用过邮件系统,那么你可以推导出这些元素实际包含的内容。可选的元素有多个实例,所以你可以为每种类型的接收者指定多个地址。我待会会在描述消息处理的时候来解释运行机制。以下是一个模板文件的例子。

rafe@rafe.us

someone@example.com

someoneelse@example.com

rafe@rafe.us

This is the subject

可定制的模板

属性文件的一个有用的特性是你可以使用MessageFormat 类用动态传入的值替代属性文件里的被指定参数。比如说,如果你需要在属性文件里指定errors,其中一个errors是file not found, 你可以这样写:

file.not.found.error=Error, could not find file {0}.

然后,在运行时刻,你这样使用MessageFormat:

ResourceBundle bundle = ResourceBundle.getBundle(

"MyProperties", currentLocale);

Object[] arguments    = { "some_file.txt" };

String newString      = MessageFormat.format(

bundle.getString("file.not.found.error"), arguments);

最后,newString 将包含Error, could not find file some_file.txt.我在这个系统里加入了类似的灵活性。 可以格式化所有的字符串,所以你可以在邮件模版的subject 和body元素里内嵌在属性文件使用的同样的令牌。

在某种情形下,你希望在发送邮件的时候插入个人化的信息。比如,你希望在邮件内容里或者订单的内容里包含收件人的姓。本系统使用MessageFormat 来处理邮件模版的内容和主题,从而解决这个问题。处理内容和主题的时候只使用一个参数数组。这样主题里可以包含令牌{0}, {2}, {3},  内容可以包含令牌{0}, {1}, {4} 。我之所以采用这种方式是因为在很多情形下主题和内容使用相同的参数,同时这种方式也简化了传递给EmailSender所需要的参数。

处理模版

创建完模版,下一步所要做的就是处理它。我们知道,现在有很多的XML处理包可供选择。Commons Digester是Jakarta的公共项目,最初是为了在Struts项目中快速方便的解析Struts的的配置文件而产生的。它提供了从XML文件里的元素到使用类似于XPath  语法的数据结构的映射。 好处在于为了从        XML文件里得到某个元素你不必用SAX一个节点一个节点的解析,也不必使用DOM处理树状数据结构。

下面这个方法从XML文件里读取数据,然后把数据拷贝到EmailTemplate对象中。

public static EmailTemplate getEmailTemplate(InputStream aStream)

{

Digester digester = new Digester();

digester.setValidating(false);

digester.addObjectCreate("email", EmailTemplate.class);

digester.addBeanPropertySetter("email/subject", "subject");

digester.addBeanPropertySetter("email/body", "body");

digester.addBeanPropertySetter("email/from", "from");

digester.addCallMethod("email/to", "addTo", 0);

digester.addCallMethod("email/cc", "addCc", 0);

digester.addCallMethod("email/bcc", "addBcc", 0);

try

{

return (EmailTemplate)digester.parse(aStream);

}

catch (IOException e)

{

logger.error("Error: ", e);

return null;

}

catch (SAXException e)

{

logger.error("Error: ", e);

return null;

}

}

让我们来逐行研究这段代码。Commons Digester工作的原理是由你来指定解析文件的一些规则。因为没有规范邮件模版的DTD文件,所以在指定处理规则之前,我将validating flag设定为false。开始处理文件的时候,我实例化Digester对象然后调用方法建立数据映射规则。首先,我调用addObjectCreate()方法来建立创建EmailTemplate对象的规则。email是XML模版文件的根元素。因此模版文件和EmailTemplate 对象一一对应。

我使用addBeanPropertySetter()来处理在模版文件中只出现一次的元素。这个方法有两个参数,元素的路径和要调用的赋值方法。在第一次调用的时候,我指定在文件中符合email/subject 模式的元素应该赋值给EmailTemplate 类的subject 。我们用 “/”来描速XML文件的内嵌关系。在这个例子中,符合subject模式的元素是email 子元素。为了提供更多的灵活性我们可以使用Wildcards。参考Commons Digester的JavaDoc 你可以了解详细的模式的构成方式。

使用赋值方法处理在模版文件中出现多次的元素是不可行的。我们使用addCallMethod()来处理这种情形,这个方法从元素中取值并且调用指定的方法。我使用这个方法有三个参数的版本,它们是:匹配的模式,调用的方法,调用方法所使用的参数数量。在例子的三种情形中第三个参数都是0,说明符合模式的元素是调用方法的唯一参数。在EmailTemplate类中我定义了三个方法:addTo(), addCc(),  addBcc(),这三个方法将模版文件中的收件人列表加入到模版类的收件人集合中。

邮件元素的六种类型的子元素的规则都被指定好之后,我开始解析这个文件。在这个例子中, 我传入getEmailTemplate 方法的输入参数InputStream 。parse方法可以解析File,SAX InputSource, InputStream,  Reader, 目标文件的URI。我使用InputStream。 由调用这个方法的代码取得XML文件并且把它转化为InputStream 。为了让这个方法更加通用,我可以用Object作为参数,并且在方法内部使用instanceof 来确定参数的类型,再用相应的方式来处理。

方法parse 抛出IOException 或者SAXException。把这些异常传给Log4J,由它来处理,返回null. 如果没有异常抛出, 将返回由Digester创建的EmailTemplate对象。

EmailTemplate类剩下的部分

getEmailTemplate()方法是类EmailTemplate的核心。其他的部分是一些属性值和一些辅助性的方法。有3个String 类型的属性值:内容,主题,寄件人地址,3个ArrayList属性值:to, CC, BCC 列表,这3个值都以String作为基本元素。还有相应的get,set和加入集合的方法。还有3个附加的方便的方法:getToAddresses(), getCcAddresses(), 和 getBccAddresses()。JavaMail接口需要InternetAddress 数组作为地址集合的参数,这些方法可以把对象的String数组转化为JavaMail接口需要的数组形式。

类EmailSender

当模版文件被解析成EmailTemplate对象,下一步就是发送邮件信息。EmailSender 类包含一个静态的,重载的方法-sendEmail()。 这个方法可以通过很多种方式调用,所有的方式都是对下面这个完全参数方法的一个引用:

public static void sendEmail(

String aTo,

EmailTemplate aTemplate,

String[] aArgs)

参数不需要过多的解释。第一个是邮件的发送地址。你可以在邮件模版里指定很多接收人地址,但是在运行时刻,大多数情况下,系统只需要一个接收人。比如说,你发送一封密码提醒的邮件,只需要指定申请密码的用户的邮件地址。在邮件模版里指定的收件人列表在某种情况下适用:作为测试,系统需要发送邮件到特定收件人列表或者发送时需要包含特定收件人列表。比如说,假设一个系统每当订单提交的时候需要通过一封邮件触发一个workflow,在这种情形下邮件模版种特定的接收人地址是有意义的。

第二个参数是EmailTemplate自身。第三个参数是MessageFormat解析邮件主题和内容所需要的参数集。由调用这个方法的代码来创建个性化邮件模版所需要的信息数组。也有其他申明的方法简化了这个方法的调用(所以你可以在不指定收件人,或者在没有参数的情况下调用这个方法)。

方法内部由使用JavaMail发送邮件所需要的一系列调用组成。我觉得使用JavaMail会造成许多冗余,我们来具体看一下。首先,我要通过检测来确定EmailTemplate是否为空。如果为空,什么都不能做。设定的第一步是使用SMTP server的设置创建一个Properties对象(Hashtable)。我把SMTP server的设置设定在 文件里,所以我把这个值从属性文件里读出来然后放到我创建的properties对象里去。

接着我创建了一个JavaMail Session 对象传入Properties 对象。Session对象在创建MimeMessage对象的时候需要。这个是我待会要做的。然后我将From:的值指定到传入参数EmailTemplate对象的相应栏位。下一步我把To:的值设定到我构建的消息中。这里会有一些技巧,因为用户可以传入To: 地址,同时邮件模版里也包含一些To:地址。问题在于JavaMail 喜欢使用数组描速地址列表,所以由我来决定接收人列表的有多大,然后构建传入的参数。

因为CC: BCC:的地址必须在模版里指定,我们可以直接来处理它们。我使用EmailTemplate类里的方法把其他的收件人加入到消息里。就像我开始提到的,我使用MessageFormat解析处理邮件主题和内容的方法所需要的参数集。做完之后,我把新的主题拷贝到消息主体里。如此处理消息的内容。剩下的就是调用Transport.send()并且传入MimeMessage 对象。

使用这个系统

我刚才已经解释了系统的运作原理,现在我来解释如何通过 servlet来使用它,在其他程序里调用的方式是类似的。以下是代码:

// Grab the email template.

InputStream template =

getServlet()

.getServletConfig()

.getServletContext()

.getResourceAsStream(

"/WEB-INF/email/registrationNotification.xml");

EmailTemplate notification = EmailTemplate.getEmailTemplate(template);

// Create the section of the email containing the actual user data.

String[] args = { "Rafe" };

EmailSender.sendEmail("rafe@rafe.us", notification, args);

使用这个系统的第一步是把你的XML模版文件转化成InputStream。 因为我使用的是servlet,我从ServletContext取得这个文件。当然还有其他的方式取得这个文件,但是在servlet环境里,这种方式很好用。我只用把InputStream 传给刚才所描述的EmailTemplate.getEmailTemplate()方法就可以了。下一步,建立个性化邮件所需要的参数数组,然后调用方法EmailSender.sendEmail()。

更多

这个系统还可以更多的优化,有两个比较明显的需要改善的地方:系统应该同时支持纯文本和HTML;支持附件。创建这种类型的信息需要使用类型javax.mail.MimeMultipart。还有在何处存储附件和如何指定附件的问题。在我的系统里,我没有在模版文件里处理附件,因为我的附件是在邮件发送的时候创建的。

Rafe Colburn 是一个Java开发工程师,同时也是一名计算机图书的作者,他使用过Perl,CGI, HTML, JAVA

posted on 2006-02-28 09:53 黑咖啡 阅读(469) 评论(3)  编辑  收藏 所属分类: tec

java构建xml参数_Java中使用XML创建EMAIL模板相关推荐

  1. java xml 合并_Java中合并XML文档的设计与实现

    为了读写XML文件,需要导入如下JAVA包,"//"后为注释说明,笔者的环境是JDK1.3.1,在JDK 1.4.0中测试也通过. Import java.io. *; //Jav ...

  2. java 不定长参数_java中什么是不定长参数?

    java中的不定长参数 不定长度参数,就是没有规定长度的参数. 不定长参数方法的语法如下:返回值 方法名(参数类型...参数名称) 在参数列表中使用"..."形式定义不定长参数,其 ...

  3. java对xml解析_Java中对xml的解析

    // 1.通过DocumentBuilderFactor创建解析工厂 DocumentBuilderFactory builderFactory = DocumentBuilderFactory .n ...

  4. java xml 合并_Java中合并XML文档的合并

    Private Boolean dupliate (Document doc_dup, Element father, Element son) throws Exception { Boolean ...

  5. java 什么是参数_java中参数是什么?

    java中参数是什么?以下文字资料是由(历史新知网www.lishixinzhi.com)小编为大家搜集整理后发布的内容,让我们赶快一起来看一下吧! java中参数是什么? 参数分为实参和形参.方法名 ...

  6. java 什么是参数_java中的参数是什么?

    本篇文章介绍了Java中什么是参数,以及参数的两种类型及他们之间的关系,希望对学习Java开发的同学有帮助. java中的参数是什么? 参数分两种:一种叫形式参数,一种叫实际参数 方法名后面括号里面用 ...

  7. java lambda做参数_Java中的Lambda参数

    我的追求是如何在函数中将lambda表达式作为参数? 我的意图是,当我执行sum方法时,方法accept中的参数被对象延续使用并使用(是吗?),我的疑问是我如何做才能使该对象延续成为使用value1 ...

  8. java map传入参数_JAVA中map中参数的添加修改

    Map以按键/数值对的形式存储数据,和数组非常相似,在数组中存在的索引,它们本身也是对象. Map的接口 Map---实现Map Map.Entry--Map的内部类,描述Map中的按键/数值对. S ...

  9. java 异常处理发生异常_Java中的异常处理

    java 异常处理发生异常 Exception Handling in Java is a very interesting topic. Exception is an error event th ...

最新文章

  1. torch量化的流程
  2. 【C++ Primer学习笔记】第1章:快速入门
  3. 泰勒公式推导过程_论泰勒级数在机器学习家庭中的地位
  4. leetcode 105. 从前序与中序遍历序列构造二叉树
  5. Linux环境下静态库的生成和使用 (.a文件)
  6. 数据分析-pands分析美国选民对总统的喜好(python实现)
  7. 计算机多媒体设计徽章,酷毙了:Hackaday将会议徽章设计成一台可编程的电脑
  8. FastDFS服务器搭建
  9. mac gcc安装_16_超级小白Mac Pro下安装superset遇见的坑
  10. java中jscrollpane_java中JScrollPane问题
  11. java类转换异常,java.lang.ClassCastException: java.lang.Integer cannot be cast to java.lang.Long
  12. 在 IE 中使用 Windows 窗体控件
  13. 【Gym-101908 L】Subway Lines【树上两条路径交】
  14. Pop猫 回收站图标
  15. 被面试官问到项目中的难点?是时候对自己的项目进行总结了(记一次项目问题总结)
  16. 客运综合管理系统项目解析-安全检查(模块)-车辆安检情况查询
  17. UVM m_sequencer 和 p_sequencer
  18. us排名2019计算机专业排名,2019 USNews美国大学计算机专业排名
  19. Windows 10最新原版镜像(Version 21H1)
  20. Android百度地图雷达效果,地图导航实测:百度地图路线雷达圈粉“老司机”

热门文章

  1. 接待员如何向客人upsell_客房留言卡也能收获好评,看看高情商酒店如何做的?...
  2. python cli_click python cli 开发包
  3. python 测试端口连通_Python语言 实现端口连通性检测
  4. QString转char*的问题
  5. pthread_cond_wait的spurious wakeup问题
  6. char 转wchar_t 及wchar_t转char
  7. java 匿名内部类 百科_java匿名内部类具体概念是什么,在什么地方用到?
  8. 一元三次方程重根判别式_许兴华——关于复数集中解一元二次方程的问题
  9. 【转】添加web引用和添加服务引用有什么区别?
  10. C++11 FAQ中文版:std::function 和 std::bind