利用Java实现电子邮件的批量发送

JAVA MAIL是利用现有的邮件账户发送邮件的工具,比如说,我在网易注册一个邮箱账户,通过JAVA Mail的操控,我可以不亲自登录网易邮箱,让程序自动的使用网易邮箱发送邮件。这一机制被广泛的用在注册激活和垃圾邮件的发送等方面。进行下载,并将mail.jar添加到classpath即可。如果你使用的是JAVA EE SDK,则可以在C:glassfishv3glassfishmodulesmail.jar找到所需的jar包,同样需要添加的classpath。

  JAVA邮件发送的大致过程是这样的的:

  1、构建一个继承自javax.mail.Authenticator的具体类,并重写里面的getPasswordAuthentication()方法。此类是用作登录校验的,以确保你对该邮箱有发送邮件的权利。

  2、构建一个properties文件,该文件中存放SMTP服务器地址等参数。

  3、通过构建的properties文件和javax.mail.Authenticator具体类来创建一个javax.mail.Session。Session的创建,就相当于登录邮箱一样。剩下的自然就是新建邮件。

  4、构建邮件内容,一般是javax.mail.internet.MimeMessage对象,并指定发送人,收信人,主题,内容等等。

  5、使用javax.mail.Transport工具类发送邮件。

  下面是我封装的代码,注释也比较详细。呼呼~~

  1、首先是继承自javax.mail.Authenticator的一个具体类。getPasswordAuthentication()方法也就是构建一个PasswordAuthentication对象并返回,有点费解JAVA Mail这样的设计意图,可能是javax.mail.Authenticator为我们提供了附加的保证安全的验证措施吧。

  1. package com.mzule.simplemail;  
  2. import javax.mail.Authenticator;  
  3. import javax.mail.PasswordAuthentication;  
  4. /** 
  5. * 服务器邮箱登录验证 
  6. * @author MZULE 
  7. */ 
  8. public class MailAuthenticator extends Authenticator {  
  9. /** 
  10. * 用户名(登录邮箱) 
  11. */ 
  12. private String username;  
  13. /** 
  14. * 密码 
  15. */ 
  16. private String password;  
  17. /** 
  18. * 初始化邮箱和密码 
  19. * @param username 邮箱 
  20. * @param password 密码 
  21. */ 
  22. public MailAuthenticator(String username, String password) {  
  23. this.username = username;  
  24. this.password = password;  
  25. }  
  26. String getPassword() {  
  27. return password;  
  28. }  
  29. @Override 
  30. protected PasswordAuthentication getPasswordAuthentication() {  
  31. return new PasswordAuthentication(username, password);  
  32. }  
  33. String getUsername() {  
  34. return username;  
  35. }  
  36. public void setPassword(String password) {  
  37. this.password = password;  
  38. }  
  39. public void setUsername(String username) {  
  40. this.username = username;  
  41. }  
  42. }

 2、邮件发送类,剩下的步骤都是在这个类实现的。代码中的SimpleMail是封装了邮件主题和内容的一个POJO。觉得在一个方法参数中既包含主题又包含内容,不太合适,故重载了此方法。还有就是因为大多数邮箱的SMTP服务器地址都是可以通过邮箱地址算出来,简单起见,提供了一个不需要SMTP服务器地址的构造器。

  1. package com.mzule.simplemail;  
  2. import java.util.List;  
  3. import java.util.Properties;  
  4. import javax.mail.MessagingException;  
  5. import javax.mail.Session;  
  6. import javax.mail.Transport;  
  7. import javax.mail.internet.AddressException;  
  8. import javax.mail.internet.InternetAddress;  
  9. import javax.mail.internet.MimeMessage;  
  10. import javax.mail.internet.MimeMessage.RecipientType;  
  11. /**  
  12. * 简单邮件发送器,可单发,群发。  
  13. *  
  14. * @author MZULE  
  15. *  
  16. */ 
  17. public class SimpleMailSender {  
  18. /**  
  19. * 发送邮件的props文件  
  20. */ 
  21. private final transient Properties props = System.getProperties();  
  22. /**  
  23. * 邮件服务器登录验证  
  24. */ 
  25. private transient MailAuthenticator authenticator;  
  26. /**  
  27. * 邮箱session  
  28. */ 
  29. private transient Session session;  
  30. /**  
  31. * 初始化邮件发送器  
  32. *  
  33. * @param smtpHostName  
  34. * SMTP邮件服务器地址  
  35. * @param username  
  36. * 发送邮件的用户名(地址)  
  37. * @param password  
  38. * 发送邮件的密码  
  39. */ 
  40. public SimpleMailSender(final String smtpHostName, final String username,  
  41. final String password) {  
  42. init(username, password, smtpHostName);  
  43. }  
  44. /**  
  45. * 初始化邮件发送器  
  46. *  
  47. * @param username  
  48. * 发送邮件的用户名(地址),并以此解析SMTP服务器地址  
  49. * @param password  
  50. * 发送邮件的密码  
  51. */ 
  52. public SimpleMailSender(final String username, final String password) {  
  53. //通过邮箱地址解析出smtp服务器,对大多数邮箱都管用  
  54. final String smtpHostName = "smtp." + username.split("@")[1];  
  55. init(username, password, smtpHostName);  
  56. }  
  57. /**  
  58. * 初始化  
  59. *  
  60. * @param username  
  61. * 发送邮件的用户名(地址)  
  62. * @param password  
  63. * 密码  
  64. * @param smtpHostName  
  65. * SMTP主机地址  
  66. */ 
  67. private void init(String username, String password, String smtpHostName) {  
  68. // 初始化props  
  69. props.put("mail.smtp.auth", "true");  
  70. props.put("mail.smtp.host", smtpHostName);  
  71. // 验证  
  72. authenticator = new MailAuthenticator(username, password);  
  73. // 创建session  
  74. session = Session.getInstance(props, authenticator);  
  75. }  
  76. /**  
  77. * 发送邮件  
  78. *  
  79. * @param recipient  
  80. * 收件人邮箱地址  
  81. * @param subject  
  82. * 邮件主题  
  83. * @param content  
  84. * 邮件内容  
  85. * @throws AddressException  
  86. * @throws MessagingException  
  87. */ 
  88. public void send(String recipient, String subject, Object content)  
  89. throws AddressException, MessagingException {  
  90. // 创建mime类型邮件  
  91. final MimeMessage message = new MimeMessage(session);  
  92. // 设置发信人  
  93. message.setFrom(new InternetAddress(authenticator.getUsername()));  
  94. // 设置收件人  
  95. message.setRecipient(RecipientType.TO, new InternetAddress(recipient));  
  96. // 设置主题  
  97. message.setSubject(subject);  
  98. // 设置邮件内容  
  99. message.setContent(content.toString(), "text/html;charset=utf-8");  
  100. // 发送  
  101. Transport.send(message);  
  102. }  
  103. /**  
  104. * 群发邮件  
  105. *  
  106. * @param recipients  
  107. * 收件人们  
  108. * @param subject  
  109. * 主题  
  110. * @param content  
  111. * 内容  
  112. * @throws AddressException  
  113. * @throws MessagingException  
  114. */ 
  115. public void send(List<String> recipients, String subject, Object content)  
  116. throws AddressException, MessagingException {  
  117. // 创建mime类型邮件  
  118. final MimeMessage message = new MimeMessage(session);  
  119. // 设置发信人  
  120. message.setFrom(new InternetAddress(authenticator.getUsername()));  
  121. // 设置收件人们  
  122. final int num = recipients.size();  
  123.  InternetAddress[] addresses = new InternetAddress[num];  
  124. for (int i = 0; i <num; i++) {  
  125. addresses[i] = new InternetAddress(recipients.get(i));  
  126. }  
  127. message.setRecipients(RecipientType.TO, addresses);  
  128. // 设置主题  
  129. message.setSubject(subject);  
  130. // 设置邮件内容  
  131. message.setContent(content.toString(), "text/html;charset=utf-8");  
  132. // 发送  
  133. Transport.send(message);  
  134. }  
  135. /**  
  136. * 发送邮件  
  137. *  
  138. * @param recipient  
  139. * 收件人邮箱地址  
  140. * @param mail  
  141. * 邮件对象  
  142. * @throws AddressException  
  143. * @throws MessagingException  
  144.  */ 
  145. public void send(String recipient, SimpleMail mail)  
  146. throws AddressException, MessagingException {  
  147. send(recipient, mail.getSubject(), mail.getContent());  
  148. }  
  149. /**  
  150. * 群发邮件  
  151. *  
  152. * @param recipients  
  153. * 收件人们  
  154. * @param mail  
  155.  * 邮件对  
  156.  * @throws AddressException  
  157. * @throws MessagingException  
  158. */ 
  159. public void send(List<String> recipients, SimpleMail mail)  
  160. throws AddressException, MessagingException {  
  161. send(recipients, mail.getSubject(), mail.getContent());  
  162. }  
  163. }

3、调用上面的邮箱发送器,可以构建一个工厂类,工厂类可以封装创建的过程,所以通过读配置文件获取邮箱用户名,密码都会变得十分方便。下面的代码是我在写观察者模式的时候写的,只是简单演示了工厂类。

  1.  package com.mzule.dp.observer.factory;  
  2. import com.mzule.dp.observer.constant.MailSenderType;  
  3. import com.mzule.simplemail.SimpleMailSender;  
  4. /**  
  5. * 发件箱工厂  
  6. *  
  7. * @author MZULE  
  8. *  
  9. */ 
  10. public class MailSenderFactory {  
  11. /**  
  12. * 服务邮箱  
  13. */ 
  14. private static SimpleMailSender serviceSms = null;  
  15. /**  
  16. * 获取邮箱  
  17. *  
  18. * @param type 邮箱类型  
  19. * @return 符合类型的邮箱  
  20. */ 
  21. public static SimpleMailSender getSender(MailSenderType type) {  
  22. if (type == MailSenderType.SERVICE) {  
  23. if (serviceSms == null) {  
  24. serviceSms = new SimpleMailSender("invisible@126.com",  
  25. "hidden");  
  26. }  
  27. return serviceSms;  
  28. }  
  29. return null;  
  30. }  
  31. }

  4、发送邮件,还是观察者模式DEMO里面的代码,呼呼。

  1. package com.mzule.dp.observer.observer;  
  2. import java.util.ArrayList;  
  3. import java.util.List;  
  4. import java.util.Observable;  
  5. import java.util.Observer;  
  6. import javax.mail.MessagingException;  
  7. import javax.mail.internet.AddressException;  
  8. import com.mzule.dp.observer.constant.MailSenderType;  
  9. import com.mzule.dp.observer.factory.MailSenderFactory;  
  10. import com.mzule.dp.observer.po.Product;  
  11. import com.mzule.simplemail.SimpleMailSender;  
  12. public class ProductPriceObserver implements Observer {  
  13. @Override 
  14. public void update(Observable obj, Object arg) {  
  15. Product product = null;  
  16. if (obj instanceof Product) {  
  17. product = (Product) obj;  
  18. }  
  19. if (arg instanceof Float) {  
  20. Float price = (Float) arg;  
  21.  Float decrease = product.getPrice() - price;  
  22. if (decrease >0) {  
  23. // 发送邮件  
  24. SimpleMailSender sms = MailSenderFactory  
  25. .getSender(MailSenderType.SERVICE);  
  26. List<String> recipients = new ArrayList<String>();  
  27. recipients.add("invisible@qq.com");  
  28. recipients.add("invisible@gmail.com");  
  29. try {  
  30. for (String recipient : recipients) {  
  31. sms.send(recipient, "价格变动", "您关注的物品" 
  32. + product.getName() + "降价了,由" 
  33. + product.getPrice() + "元降到" + price + "元,降幅达" 
  34. + decrease + "元人民币。赶快购物吧。");  
  35. }  
  36. catch (AddressException e) {  
  37. e.printStackTrace();  
  38. catch (MessagingException e) {  
  39. e.printStackTrace();  
  40. }  
  41. }  
  42. }  
  43. }  
  44. }

本文出自seven的测试人生公众号最新内容请见作者的GitHub页:http://qaseven.github.io/

时间: 2024-08-04 02:29:37

利用Java实现电子邮件的批量发送的相关文章

利用Java实现串口全双工通讯 (转载)

利用Java实现串口全双工通讯 (投稿) Qingye Jiang (John)SMTH ID: qyjohnE-mail : qjiang@tsinghua.edu 一个嵌入式系统通常需要通过串口与其主控系统进行全双工通讯,譬如一个流水线控制系统需要不断的接受从主控系统发送来的查询和控制信息,并将执行结果或查询结果发送回主控系统.本文介绍了一个简单的通过串口实现全双工通讯的Java类库,该类库大大的简化了对串口进行操作的过程. 本类库主要包括:SerialBean.java (与其他应用程序的

利用Java Beans在应用程序中创建组件

JavaBeans模块使开发人员能够创建称之为组件的软件单元(也就是我们熟知的beans).你可以把beans加载在更复杂的组件.Java小型应用程序(applets)或应用程序上.JavaBeans广泛应用于IDE应用程序,使你能够很容易地可视化组合组件并动态修改它的属性. Beans是动态的,允许你改变和定制它.在JavaBean Builder Tool里的设计模块中,你能利用 Bean的属性窗口来可视化定制和保存(持久)bean.你也可以从toolbox中选择一个bean,把它拖拽到窗体

利用java socket 写的一个联机的五子棋游戏,服务器端和客户端的问题,大神求教啊。。

问题描述 利用java socket 写的一个联机的五子棋游戏,服务器端和客户端的问题,大神求教啊.. 利用java socket 写的一个联机的五子棋游戏,一个服务器端的程序和两个玩家的客户端程序,可不可以都运行在同一台主机上啊? 我运行服务器端和其中一个玩家的客户端程序时,正常.但是运行第二个玩家的客户端程序时,就出现了 Java.net.BindExecption Address already in use: JVM_Blind. 的异常.怎么办? 还有我打算客户端把下的棋子的对象传给服

利用Java获取文件名、类名、方法名和行号的方法小结_java

大家都知道,在C语言中,我们可以通过宏FILE. __LINE__来获取文件名和行号,而在Java语言中,则可以通过StackTraceElement类来获取文件名.类名.方法名.行号,具体代码如下: public static int getLineNumber( ){ StackTraceElement[] stackTrace = new Throwable().getStackTrace(); return stackTrace[1].getLineNumber( ); } public

利用java实现UDP(二)

问题描述 本帖子接<利用java实现UDP>这个帖子:客户端:UDPClientBean.javapackageUDPClient;importjava.io.*;importjava.net.*;classUDPClientBean{privateDatagramSocketdSocket;privateintServerPort;privateintClientPort;privateInetAddressServerIP;privateInetAddressClientIP;privat

利用java实现UDP

问题描述 网络课设作业,利用java来实现UDP,做了个小程序,第一次运行完全正确,等到第二次运行,就出现了问题,每次点"发送"按钮,都会停止在那里,没有错误提示,不知道是运算速度慢,结果还没出来,还是肿么一个情况,请大神给予解答哈--因每个帖子最多长度为10000个字符,所以分两个帖子发.呜呜呜--服务器端:UDPServerBean.javapackageUDPServer;importjava.io.*;importjava.net.*;publicclassUDPServerB

java struts2-怎样利用Java 中的struts2框架实现数据库中用户登录功能?

问题描述 怎样利用Java 中的struts2框架实现数据库中用户登录功能? 在Action中LoginAction怎样写? 配置文件中怎样写? 总体实现能够使数据库中已经存在的用户凭自己的密码与用户名登录成功呢?数据库是Oracle数据库.

利用Java实现zip压缩/解压缩

压缩 利用Java实现zip压缩/解压缩(作者: 2000年07月06日 13:30) 由于网络带宽有限,所以数据文件的压缩有利于数据在Internet上的快速传输,同时也节 省服务器的外存空间. Java 1.1实现了I/O数据流与网络数据流的单一接口,因此数据的压缩.网络传输和解 压缩的实现比较容易,下面介绍利用ZipEntry.ZipInputStream和ZipOutputStream三个Java 类实现zip数据压缩方式的编程方法. zip压缩文件结构:一个zip文件由多个entry组

Php利用java解析xml

xml 作者: 雨伞首先如果你对JAVA不感兴趣或者说你的主机不支持JAVA或者你也不打算学习的话请不要浪费时间在这里. 请先安装JAVA的执行环境与PHP结合,具体参考http://www.phpx.com/happy/thr78795.html 或者下载http://www.javax.org/download/php_java.rar里面我写有个readme.txt说明文档,这个压缩包是个简单的例子. 要下载我的JAVA源码请到http://www.javax.org/download/J