spring boot定时任务接收邮件并且存储附件的方法讲解
在spring boot中写定时任务很简单,两个注解就可以实现。在启动类中加@EnableScheduling
,然后在你需要定时的方法上加上@Scheduled(cron="0 10 8 * * ?");
括号内为cron表达式。如下图:
接收邮件及其判断是否有附件,并且存储附件。
public class TimerTaskServiceImpl implements TimerTaskService { @Autowired private ParseTxtServiceImpl parseTxtService; /** * 接收邮件 */ @Override @Scheduled(cron="0 10 8 * * ?") public void timerTaskInfo(){ //邮件配置信息 String host=Constants.MAIL_HOST; String userName=Constants.MAIL_USER_NAME; String passWord=Constants.MAIL_PASS_WORD; //邮件配置类 Properties properties=new Properties(); //邮件配置缓存 Session session=Session.getDefaultInstance(properties); session.setDebug(true); String fileName=null; try { //邮件服务器的类型 Store store = session.getStore("pop3"); //连接邮箱服务器 store.connect(host,userName,passWord); // 获得用户的邮件帐户 Folder folder=store.getFolder("INBOX"); if (folder == null) { logger.info("获取邮箱文件信息为空"); } // 设置对邮件帐户的访问权限可以读写 folder.open(Folder.READ_WRITE); Calendar calendar=Calendar.getInstance(); calendar.add(Calendar.DATE,-1); Date mondayDate = calendar.getTime(); SearchTerm comparisonTermLe = new SentDateTerm(ComparisonTerm.GT, mondayDate); SearchTerm address=new SubjectTerm( "MU Report"); SearchTerm comparisonAndTerm = new AndTerm(address, comparisonTermLe); Message[] messages = folder.search(comparisonAndTerm); for(int i = 0 ; i < messages.length ; i++){ MimeMessage msg = (MimeMessage) messages[i]; //判断是否有附件 boolean isContainerAttachment = isContainAttachment(msg); if (isContainerAttachment) { //保存附件 fileName=saveAttachment(msg,Constants.STORAGE_FILE); //保存接收到的邮件并且收件箱删除邮件 msg.setFlag(Flags.Flag.DELETED, true); } if(!isContainerAttachment) { continue; } } folder.close(true); store.close(); parseTxtService.readTxt(fileName); }catch (NoSuchProviderException e){ loggerError.error("接收邮箱信息异常:{}",e); }catch (MessagingException e){ loggerError.error("连接邮箱服务器信息异常:{}",e); }catch (IOException e){ loggerError.error("接收邮箱信息解析异常:{}",e); }catch (IllegalStateException e){ loggerError.error("接收邮箱信息为空:{}",e); } } /** * 判断邮件中是否包含附件 * @return 邮件中存在附件返回true,不存在返回false * @throws MessagingException * @throws IOException */ public static boolean isContainAttachment(Part part) throws MessagingException, IOException { boolean flag = false; if (part.isMimeType(Constants.MULTI_PART)) { MimeMultipart multipart = (MimeMultipart) part.getContent(); int partCount = multipart.getCount(); for (int i = 0; i < partCount; i++) { BodyPart bodyPart = multipart.getBodyPart(i); String disp = bodyPart.getDisposition(); if (disp != null && (disp.equalsIgnoreCase(Part.ATTACHMENT) || disp.equalsIgnoreCase(Part.INLINE))) { flag = true; } else if (bodyPart.isMimeType(Constants.MULTI_PART)) { flag = isContainAttachment(bodyPart); } else { String contentType = bodyPart.getContentType(); if (contentType.indexOf(Constants.APPLICATION_CONTEXT) != -1) { flag = true; } if (contentType.indexOf(Constants.NAME_CONTEXT) != -1) { flag = true; } } if (flag){ break; } } } else if (part.isMimeType(Constants.MESSAGE_RFC)) { flag = isContainAttachment((Part)part.getContent()); } return flag; } /** * 保存附件 * @param part 邮件中多个组合体中的其中一个组合体 * @param destDir 附件保存目录 * @throws UnsupportedEncodingException * @throws MessagingException * @throws FileNotFoundException * @throws IOException */ public String saveAttachment(Part part, String destDir) throws UnsupportedEncodingException, MessagingException, FileNotFoundException, IOException { String fileName=null; if (part.isMimeType(Constants.MULTI_PART)) { Multipart multipart = (Multipart) part.getContent(); //复杂体邮件 //复杂体邮件包含多个邮件体 int partCount = multipart.getCount(); for (int i = 0; i < partCount; i++) { //获得复杂体邮件中其中一个邮件体 BodyPart bodyPart = multipart.getBodyPart(i); //某一个邮件体也有可能是由多个邮件体组成的复杂体 String disp = bodyPart.getDisposition(); if (disp != null && (disp.equalsIgnoreCase(Part.ATTACHMENT) || disp.equalsIgnoreCase (Part.INLINE))) { InputStream is = bodyPart.getInputStream(); saveFile(is, destDir, decodeText(bodyPart.getFileName())); fileName=decodeText(bodyPart.getFileName()); } else if (bodyPart.isMimeType(Constants.MULTI_PART)) { saveAttachment(bodyPart,destDir); } else { String contentType = bodyPart.getContentType(); if (contentType.indexOf(Constants.NAME_CONTEXT) != -1 || contentType.indexOf (Constants.APPLICATION_CONTEXT) != -1) { saveFile(bodyPart.getInputStream(), destDir, decodeText(bodyPart.getFileName())); fileName=decodeText(bodyPart.getFileName()); } } } } else if (part.isMimeType(Constants.MESSAGE_RFC)) { saveAttachment((Part) part.getContent(),destDir); } return fileName; } /** * 读取输入流中的数据保存至指定目录 * @param is 输入流 * @param fileName 文件名 * @param destDir 文件存储目录 * @throws FileNotFoundException * @throws IOException */ private void saveFile(InputStream is, String destDir, String fileName) throws FileNotFoundException, IOException { BufferedInputStream bis = new BufferedInputStream(is); if(fileName.contains(Constants.TXT_SUFIXX)) { BufferedOutputStream bos = new BufferedOutputStream( new FileOutputStream(new File(destDir + fileName))); int len = -1; while ((len = bis.read()) != -1) { bos.write(len); bos.flush(); } bos.close(); bis.close(); } } }
其中查询邮件的条件,你可以自行更改。
接收邮件服务器的配置
可能出现的bug
说说用模板可能会碰到的bug。
怎么回事呢?代码写了,看了好几遍也没错,就是运行就报错,在网上看了别人的代码拿过来还是报错,报错如下:
这个错误大概意思就是我的模板的html中每个标签都要是闭标签,要这种类型的<a></a>
,假如是<img xxx>
这种标签就会报错。
如下所示,最坑的方法就是修改的,而且以后html的标签都要是一对一对的,坑啊、
后来有找了很多资料,原来发现是这里,themeleaf默认应该是2.xx版本,这个版本解析标签都要是一对一对的,到了3.xx之后,就不需要这么麻烦了!
都是版本问题
总结
以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,谢谢大家对脚本之家的支持。如果你想了解更多相关内容请查看下面相关链接
- spring中通过ApplicationContext getBean获取注入对象的方法实例
- Springboot-dubbo-fescar 阿里分布式事务的实现方法
- 详解spring-boot下如何满足多生产环境中个性化定制功能
- Spring 中优雅的获取泛型信息的方法
- spring @Transactional 无效的解决方案
- activemq整合springboot使用方法(个人微信小程序用)
- Springboot项目打war包docker包找不到resource下静态资源的解决方案
- Spring自带的校验框架Validation的使用实例
- 详解Spring框架下向异步线程传递HttpServletRequest参数的坑
- Spring通过ApplicationContext主动获取bean的方法讲解
相关文章
阿里nacos+springboot+dubbo2.7.3统一处理异常的两种方式
本文主要介绍了阿里nacos+springboot+dubbo2.7.3统一处理异常的两种方式,文中根据实例编码详细介绍的十分详尽,具有一定的参考价值,感兴趣的小伙伴们可以参考一下2022-03-03springboot的application.yml配置port不生效的解决方案
这篇文章主要介绍了springboot的application.yml配置port不生效的解决方案,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教2024-07-07Java thread.isInterrupted() 返回值不确定结果分析解决
这篇文章主要介绍了Java thread.isInterrupted() 返回值不确定结果分析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习吧2022-12-12
最新评论