我们系统存在大量发送邮件的需求,项目使用的是Spring框架而JavaMail也能很好的跟Spring进行集成,由于发送邮件最好还是使用异步进行发送,所以这里就采用线程池+JavaMail进行邮件发送,下面看具体代码实现:

  Step1、引入JavaMail

<mail.version>1.4.7</mail.version>
<dependency>
<groupId>javax.mail</groupId>
<artifactId>mail</artifactId>
<version>${mail.version}</version>
</dependency>

  Step2、Spring配置文件中配置MailSender和线程池

 <!--MailSender-->
<bean id="javaMailSender" class="org.springframework.mail.javamail.JavaMailSenderImpl">
<property name="host" value="${email.host}"/>
<property name="username" value="${email.username}"/>
<property name="password" value="${email.password}"/>
<property name="javaMailProperties">
<props>
<prop key="mail.smtp.auth">true</prop>
<prop key="mail.smtp.timeout">25000</prop>
<!--
<prop key="mail.smtp.starttls.enable">true</prop>
<prop key="mail.smtp.socketFactory.class">javax.net.ssl.SSLSocketFactory</prop>
-->
</props>
</property>
</bean> <!--配置线程池-->
<bean id="taskExecutor" class="org.springframework.scheduling.concurrent.ThreadPoolTaskExecutor">
<!-- 核心线程数 -->
<property name="corePoolSize" value="${executor.corePoolSize}" />
<!-- 最大线程数 -->
<property name="maxPoolSize" value="${executor.maxPoolSize}" />
<!-- 最大队列数 -->
<property name="queueCapacity" value="${executor.queueCapacity}" />
<!-- 线程池维护线程所允许的空闲时间 -->
<property name="keepAliveSeconds" value="${executor.keepAliveSeconds}" />
</bean>

  Step3、新建邮件实体类,方便业务处理

import org.apache.commons.lang3.ArrayUtils;
import org.apache.commons.lang3.StringUtils;
import org.springframework.util.Assert; import javax.mail.internet.AddressException;
import java.util.Arrays; /**
* <p>
* <code>MailEntry</code>
* </p>
* Description:
* 邮件实体类
* @author jianzh5
* @version 2017/3/31 17:00
* @since 1.0
*/
public class MailEntry {
/**
* 收件人
*/
private String[] recipients;
/**
* 抄送人
*/
private String[] carbonCopy;
/**
* 主题
*/
private String subject;
/**
* 内容
*/
private String text; public String getSubject() {
return subject;
} public void setSubject(String subject) {
this.subject = subject;
} public String getText() {
return text;
} public void setText(String text) {
this.text = text;
} public String[] getRecipients() {
return recipients.clone();
} public void setRecipients(String[] recipients) throws AddressException {
Assert.isTrue(ArrayUtils.isNotEmpty(recipients),"收件人不能为空");
String[] addresses = new String[recipients.length];
for (int i = 0; i < addresses.length; i++) {
addresses[i] = parseAddress(recipients[i]);
}
this.recipients = addresses;
} public String[] getCarbonCopy() {
if(ArrayUtils.isNotEmpty(carbonCopy)){
return carbonCopy.clone();
}
return ArrayUtils.EMPTY_STRING_ARRAY;
} public void setCarbonCopy(String[] carbonCopy) throws AddressException {
String[] addresses = new String[carbonCopy.length];
for (int i = 0; i < addresses.length; i++) {
addresses[i] = parseAddress(carbonCopy[i]);
}
this.carbonCopy = addresses;
} /**
* @desc 只要设定邮件接受人的域账号即可自动增加企业后缀
* @author jianzh5
* @date 2017/4/1 13:43
* @param recipient 收件人域账号
* @return 带企业后缀的邮箱地址
* @throws AddressException
*/
private String parseAddress(String recipient) throws AddressException {
if(StringUtils.isEmpty(recipient)){
throw new AddressException("邮箱账号不能为空",recipient);
}
return recipient + "@company.com";
} @Override public String toString() {
return "MailEntry{" +
"recipients=" + Arrays.toString(recipients) +
", carbonCopy=" + Arrays.toString(carbonCopy) +
", subject='" + subject + '\'' +
", text='" + text + '\'' +
'}';
}
}

  说明:由于我们是使用企业邮箱发送邮件,而一般业务都只能取到公司用户名而非真正的邮箱地址,所以在上面实体类里进行了一次转换,如果能直接拿到邮箱地址的话上面的转换是不需要的。收件人和抄送人都是支持数组形式的参数,所以在这里直接使用数组作为收件人、抄送人属性,发送邮件的时候只要直接设置成数组参数即可。

  Step4、发送邮件业务层实现

import org.apache.commons.lang3.ArrayUtils;
import org.springframework.core.task.TaskExecutor;
import org.springframework.mail.javamail.JavaMailSender;
import org.springframework.mail.javamail.MimeMessageHelper;
import org.springframework.stereotype.Service; import javax.annotation.Resource;
import javax.mail.MessagingException;
import javax.mail.internet.MimeMessage; /**
* <p>
* <code>MailServiceImpl</code>
* </p>
* Description:
* 邮件系统实现类
* @author jianzh5
* @version 2017/3/31 17:10
* @since 1.0
*/ @Service
public class MailServiceImpl implements IMailService{
private LoggerUtil logger = LoggerUtil.getLogger(this.getClass()); @Resource
private TaskExecutor taskExecutor;
@Resource
private JavaMailSender javaMailSender; @Override
public void sendMail(MailEntry mailEntry) {
MimeMessage message = javaMailSender.createMimeMessage();
try {
MimeMessageHelper helper = new MimeMessageHelper(message, true, "utf-8");
helper.setFrom("xxx@company.com");
helper.setTo(mailEntry.getRecipients()); //收件人
if(ArrayUtils.isNotEmpty(mailEntry.getCarbonCopy())){
helper.setCc(mailEntry.getCarbonCopy()); //抄送人
}
helper.setSubject(mailEntry.getSubject());
helper.setText(mailEntry.getText(),true);//设置为TRUE则可以使用Html标记
addSendMailTask(message);
} catch (MessagingException e) {
// e.printStackTrace();
logger.error("邮件转换异常,邮件详细信息为{}",e.getMessage());
} } /**
* @desc 使用多线程发送邮件
* @author jianzh5
* @date 2017/4/1 11:41
* @param message MimeMessage邮件封装类
*/
private void addSendMailTask(final MimeMessage message){
try{
taskExecutor.execute(new Runnable() {
@Override
public void run() {
javaMailSender.send(message);
}
});
}catch (Exception e){
logger.error("邮件发送异常,邮件详细信息为{}",e.getMessage());
} }
}

Spring集成JavaMail并利用线程池发送邮件的更多相关文章

  1. spring boot / cloud (四) 自定义线程池以及异步处理@Async

    spring boot / cloud (四) 自定义线程池以及异步处理@Async 前言 什么是线程池? 线程池是一种多线程处理形式,处理过程中将任务添加到队列,然后在创建线程后自动启动这些任务.线 ...

  2. java利用线程池处理集合

    java利用线程池处理集合 2018年07月23日 17:21:19 衍夏成歌 阅读数:866   版权声明:本文为博主原创文章,未经博主允许不得转载. https://blog.csdn.net/s ...

  3. windows下利用线程池完成多任务的分配和运行

    在做项目的过程中有时候为了提升效率,用了多线程的方法来对任务进行分割和应用,后来发现,采用线程池的方法能更好的利用线程资源来计算任务,网上有很多关于如何运行线程池的例子,msdn上也给出了对应的例子: ...

  4. Python的并发并行[4] -> 并发[0] -> 利用线程池启动线程

    利用线程池启动线程 submit与map启动线程 利用两种方式分别启动线程,同时利用with上下文管理来对线程池进行控制 from concurrent.futures import ThreadPo ...

  5. 多个线程分别顺序交替打印一种不同字符abcdefg(已实现随便多少个线程打印多少个字符,利用线程池实现多线程)

    下面实现多线程顺序打印字符"abcdefg": 实现Runnable接口: /** * @author: rhyme * @date: 2019-08-17 14:39 * @to ...

  6. spring boot:使用多个线程池实现实现任务的线程池隔离(spring boot 2.3.2)

    一,为什么要使用多个线程池? 使用多个线程池,把相同的任务放到同一个线程池中,可以起到隔离的作用,避免有线程出错时影响到其他线程池,例如只有一个线程池时,有两种任务,下单,处理图片,如果线程池被处理图 ...

  7. spring boot:使用async异步线程池发送注册邮件(spring boot 2.3.1)

    一,为什么要使用async异步线程池? 1,在生产环境中,有一些需要延时处理的业务场景: 例如:发送电子邮件, 给手机发短信验证码 大数据量的查询统计 远程抓取数据等 这些场景占用时间较长,而用户又没 ...

  8. Spring Boot中如何配置线程池拒绝策略,妥善处理好溢出的任务

    通过之前三篇关于Spring Boot异步任务实现的博文,我们分别学会了用@Async创建异步任务.为异步任务配置线程池.使用多个线程池隔离不同的异步任务.今天这篇,我们继续对上面的知识进行完善和优化 ...

  9. Spring Boot配置公共的线程池

    内存资源很宝贵,线程池资源不宜过多的创建,同一个应用,尽量使用统一的线程池,并且相关参数需要设置适当,不造成资源的浪费,也不影响性能的提升. import java.util.concurrent.T ...

随机推荐

  1. 关于RTKLIB资料整理和学习

    最近要做一个关于rtk的项目,采用rtklib源码基础上进行移植,由于我在嵌入式方面和rtk方面都是小白,所以无论是嵌入式通信还是rtk都得从头学起.嵌入式方面打算用stm32进行移植,现在已经基本掌 ...

  2. POJ 3580 SuperMemo 伸展树

    题意: 维护一个序列,支持如下几种操作: ADD x y D:将区间\([x,y]\)的数加上\(D\) REVERSE x y:翻转区间\([x,y]\) REVOLVE x y T:将区间\([x ...

  3. Kafka安装和常用操作命令

    Kafka安装: 下载kafka_2.10-0.8.2.1 1.关闭防火墙 2.修改配置文件  server.properties broker.id=1log.dirs= /usr/kafka_2. ...

  4. gulp相关

    'use strict'; var gulp = require('gulp'), webserver = require('gulp-webserver'), //gulp服务器 connect = ...

  5. Pascal小游戏 随机函数

    一个被人写滥了的小程序,新手学习,Pascal By Chaobs 初学者可以用它来学习随机函数的运用,当然你完全可以自己写一个随机函数. var   player1,player2:longint; ...

  6. apache的/etc/httpd/conf/httpd.conf和/usr/local/apache2/conf/httpd.conf区别

    一.问题 centos系统用yum安装完apache后,重启后有时会失效,然后去网上找资料,发现有的说重启命令是这样的: /etc/init.d/httpd restart 而有的呢,说重启命令应该是 ...

  7. 把现有Unity3d游戏向Windows Phone 8.1移植(基础)

    最近在将一款现有的游戏向Windows Phone平台移植,暂时完成了一个小阶段,做一个总结. 开发环境: Windows 8.1 系统及以上,愿意的话,用Windows 10 尝鲜也可以. 微软账号 ...

  8. python之if测试

    (一)python的条件判断语句一般格式如下: if (条件1): (执行结果) elif(条件2): (执行结果) ..... else: (执行结果) 执行顺序为从上到下判断,若条件1不符合则进入 ...

  9. 【志银】MySQL命令总结

    ===0-MySQL密码设置===0.1-登入和进入MySQL数据库: 0.1.1-登入MySQL数据库:C:\Users\Administrator>mysql -u用户名 -hMySQL服务 ...

  10. CocoaPods 安装与使用

    1.如果之前已经安装过的 gem list --local | grep cocoapods 会看到如下输出: cocoapods (1.1.1)cocoapods-deintegrate (1.0. ...