## 1接收Email协议类型
接收Email:收件人通过MUA软件把邮件从MDA抓取到本地计算机的过程。

1.1 POP3

从MUA到MDA使用最广泛的是协议是POP3
* Post Office Protocol version 3,建立在tcp之上的协议
* 标准端口110
* 整个会话加密,加密端口995

1.2 IMAP

与POP3的区别:IMAP在本地的所有操作都会自动同步到服务器上,允许用户自己创建文件夹
* Internet Mail Access Protocol
* 标准端口143
* 加密端口993

2如何接收Email

  • 使用Java Mail API,实现邮件的接收
  • 无需关心POP3协议原理
    //创建session
Session session = Session.getInstance(props, null);
//创建Store对象,代表用户在邮件服务器上的整个存储
Store store = new POP3SSLStore(session, url);
//获取Folder对象,最常用的是收件箱INBOX
Folder folder = store.getFolder("INBOX");
//获取Message:返回收件箱所有的邮件,并循环处理message对象
Message[] messages = folder.getMessages();
for(Message message:messages){
...
}
package com.feiyangedu.sample.pop3;

import java.io.IOException;
import java.util.ArrayList;
import java.util.List;
import java.util.Properties; import javax.mail.Address;
import javax.mail.Authenticator;
import javax.mail.BodyPart;
import javax.mail.Flags;
import javax.mail.Folder;
import javax.mail.Message;
import javax.mail.MessagingException;
import javax.mail.Multipart;
import javax.mail.Part;
import javax.mail.PasswordAuthentication;
import javax.mail.Session;
import javax.mail.Store;
import javax.mail.URLName;
import javax.mail.internet.InternetAddress;
import javax.mail.internet.MimeMessage;
import javax.mail.internet.MimeMessage.RecipientType;
import javax.mail.internet.MimeUtility; import com.sun.mail.pop3.POP3SSLStore; public class Pop3 { final String popHost;
final String username;
final String password;
final boolean debug; public Pop3(String popHost, String username, String password) {
this.popHost = popHost;
this.username = username;
this.password = password;
this.debug = true;
} public static void main(String[] args) throws Exception {
Pop3 pop = new Pop3("pop.qq.com","123456789@qq.com","pop3授权码");//注意,不能用手机号,要用qq号
Folder folder = null;
Store store = null;
try {
store = pop.createSSLStore(); //获取到Store对象后,就可以获取Folder对象
folder = store.getFolder("INBOX");
folder.open(Folder.READ_WRITE);
System.out.println("Total messages: " + folder.getMessageCount()); //邮件数量
System.out.println("New messages: " + folder.getNewMessageCount()); //新邮件数量
System.out.println("Unread messages: " + folder.getUnreadMessageCount()); //未读邮件数量
System.out.println("Deleted messages: " + folder.getDeletedMessageCount()); //已删除的邮件数量
Message[] messages = folder.getMessages();
for (Message message : messages) {
printMessage((MimeMessage) message);
}
} finally {
if (folder != null) {
try {
folder.close(true);
} catch (MessagingException e) {
e.printStackTrace();
}
}
if (store != null) {
try {
store.close();
} catch (MessagingException e) {
e.printStackTrace();
}
}
}
}
//创建Store对象
public Store createSSLStore() throws MessagingException {
Properties props = new Properties();
props.setProperty("mail.store.protocol", "pop3");
props.setProperty("mail.pop3.port", "995"); // 主机端口号
props.setProperty("mail.pop3.host", this.popHost);// POP3主机名
// 启动SSL:
props.put("mail.smtp.socketFactory.class", "javax.net.ssl.SSLSocketFactory");
props.put("mail.smtp.socketFactory.port", "995");
URLName url = new URLName("pop3", this.popHost, 995, "", this.username, this.password);
Session session = Session.getInstance(props, null);
session.setDebug(this.debug); // 显示调试信息
Store store = new POP3SSLStore(session, url);
store.connect(); //连接到服务器
return store;
} Session createTLSStore() {
Properties props = new Properties();
props.put("mail.smtp.host", this.popHost); // POP3主机名
props.put("mail.smtp.port", "587"); // 主机端口号
props.put("mail.smtp.auth", "true"); // 是否需要用户认证
props.put("mail.smtp.starttls.enable", "true"); // 启用TLS加密
Session session = Session.getInstance(props, new Authenticator() {
protected PasswordAuthentication getPasswordAuthentication() {
return new PasswordAuthentication(Pop3.this.username, Pop3.this.password);
}
});
session.setDebug(this.debug); // 显示调试信息
return session;
} Session createInsecureStore() {
Properties props = new Properties();
props.put("mail.smtp.host", this.popHost); // POP3主机名
props.put("mail.smtp.port", "25"); // 主机端口号
props.put("mail.smtp.auth", "true"); // 是否需要用户认证
Session session = Session.getInstance(props, new Authenticator() {
protected PasswordAuthentication getPasswordAuthentication() {
return new PasswordAuthentication(Pop3.this.username, Pop3.this.password);
}
});
session.setDebug(this.debug); // 显示调试信息
return session;
} static void printMessage(MimeMessage msg) throws IOException, MessagingException {
System.out.println("--------------------");
System.out.println("Subject: " + MimeUtility.decodeText(msg.getSubject())); //邮件的主题
System.out.println("From: " + getFrom(msg)); //发件人
System.out.println("To: " + getTo(msg)); //收件人
System.out.println("Sent: " + msg.getSentDate().toString()); //发送时间
System.out.println("Seen: " + msg.getFlags().contains(Flags.Flag.SEEN)); //是否已读
System.out.println("Priority: " + getPriority(msg)); //优先级
System.out.println("Size: " + msg.getSize() / 1024 + "kb"); //邮件大小
System.out.println("Body: " + getBody(msg)); //邮件正文
System.out.println("--------------------");
System.out.println();
} static String getFrom(MimeMessage msg) throws IOException, MessagingException {
Address[] froms = msg.getFrom();
return addressToString(froms[0]);
} static String getTo(MimeMessage msg) throws MessagingException, IOException {
// 使用 msg.getAllRecipients() 获取所有收件人
Address[] tos = msg.getRecipients(RecipientType.TO);
List<String> list = new ArrayList<>();
for (Address to : tos) {
list.add(addressToString(to));
}
return String.join(", ", list);
} static String addressToString(Address addr) throws IOException {
InternetAddress address = (InternetAddress) addr;
String personal = address.getPersonal();
return personal == null ? address.getAddress()
: (MimeUtility.decodeText(personal) + " <" + address.getAddress() + ">");
} static String getPriority(MimeMessage msg) throws MessagingException {
String priority = "Normal";
String[] headers = msg.getHeader("X-Priority");
if (headers != null) {
String header = headers[0];
if ("1".equals(header) || "high".equalsIgnoreCase(header)) {
priority = "High";
} else if ("5".equals(header) || "low".equalsIgnoreCase(header)) {
priority = "Low";
}
}
return priority;
} static String getBody(Part part) throws MessagingException, IOException {
if (part.isMimeType("text/*")) {//如果邮件的正文是text开头,是一个文本邮件或html邮件
return part.getContent().toString();
}
if (part.isMimeType("multipart/*")) { //如果是以multipart开头,那么他是一个带附件的邮件,还需要递归的把附件的内容读出来
Multipart multipart = (Multipart) part.getContent();
for (int i = 0; i < multipart.getCount(); i++) {
BodyPart bodyPart = multipart.getBodyPart(i);
String body = getBody(bodyPart);
if (!body.isEmpty()) {
return body;
}
}
}
return "";
}
}

3总结

Java接收Email:

  • 用maven引入javamail依赖
  • 确定POP3服务器信息:域名/端口/使用明文/SSL
  • 调用相关API接收Email(无需关心底层TCP Socket连接)
  • 设置debug模式可以查看通信详细内容(便于排查错误)

廖雪峰Java13网络编程-2Email编程-2接收Email的更多相关文章

  1. 廖雪峰Java13网络编程-2Email编程-1发送email

    1.邮件发送 1.1传统邮件发送: 传统的邮件是通过邮局投递,从一个邮局到另一个邮局,最终到达用户的邮箱. 1.2电子邮件发送: 与传统邮件类似,它是从用户电脑的邮件软件(如outlook)发送到邮件 ...

  2. 廖雪峰Java13网络编程-1Socket编程-2TCP编程

    1. Socket 在开发网络应用程序的时候,会遇到Socket这个概念. Socket是一个抽象概念,一个应用程序通过一个Socket来建立一个远程连接,而Socket内部通过TCP/IP协议把数据 ...

  3. 廖雪峰Java13网络编程-1Socket编程-5UDP编程

    1. UDP编程: 不需要建立连接 可以直接发送和接收数据 1.1 客户端 DatagramSocket sock = new DatagramSocket(){} sock.connect(addr ...

  4. 廖雪峰Java13网络编程-1Socket编程-3TCP多线程编程

    TCP多线程编程 一个ServerSocket可以和多个客户端同时建立连接,所以一个Server可以同时与多个客户端建立好的Socket进行双向通信. 因此服务器端,当我们打开一个Socket以后,通 ...

  5. 廖雪峰Java13网络编程-3其他-2RMI远程调用

    1.RMI远程调用: Remote Method Invocation 目的:把一个接口方法暴露给远程 示例: 定义一个接口Clock,它有一个方法能够获取当前的时间,并编写一个实现类,来实现这个接口 ...

  6. 廖雪峰Java13网络编程-1Socket编程-1网络编程概念

    1.计算机网络 1.1 什么是计算机网络? 两台或更多计算机组成的网络 同一网络内的任意2台计算机都可以直接通信 所有计算机必须遵循同一种网络协议 1.2 什么是互联网 互联网是网络的网络 互联网采用 ...

  7. 廖雪峰Java13网络编程-3其他-1HTTP编程

    1.HTTP协议: Hyper Text Transfer Protocol:超文本传输协议 基于TCP协议之上的请求/响应协议 目前使用最广泛的高级协议 * 使用浏览器浏览网页和服务器交互使用的就是 ...

  8. 廖雪峰Java1-3流程控制-9break、continue

    break跳出循环 int sum=0; for(int i=1; ;i++) { sum =sum + i; if(i == 100) { break; } } System.out.println ...

  9. 廖雪峰Java1-3流程控制-6 do-while循环

    do-while循环 do-while先执行循环,再判断条件. 条件满足时继续循环:条件不满足时退出:至少循环1次 int sum =0; int n = 1; do{ sum = sum + n; ...

随机推荐

  1. C++之引用与符号“&”

    一.&的意思: 1.取地址符,这时候它用于数据的前面,比如int a=&b; 2.C++里还使用&作为引用符,如果你确认程序是标准的C而非C++的话,那么可以排除是引用了.引用 ...

  2. 用python写的自动转发邮件信息模板

    # -*- coding:utf-8 -*- #加密东西 def decrypt(key, s): c = bytearray(str(s).encode("gbk")) n = ...

  3. AtCoder ABC 130F Minimum Bounding Box

    题目链接:https://atcoder.jp/contests/abc130/tasks/abc130_f 题目大意 给定地图上 N 个点的坐标和移动方向,它们会以每秒 1 个单位的速度移动,设 A ...

  4. Mybatis 使用的 9 种设计模式,真是太有用了~

    Java技术栈 ) {      name = fullname.substring(0, delim);      children = fullname.substring(delim + 1); ...

  5. ios position:fixed 上滑下拉抖动

    ios position:fixed 上滑下拉抖动 最近呢遇到一个ios的兼容问题,界面是需要一个头底部的固定的效果,用的position:fixed定位布局,写完测试发现安卓手机正常的,按时ios上 ...

  6. 2019-8-31-dotnet-判断程序当前使用管理员运行降低权使用普通权限运行

    title author date CreateTime categories dotnet 判断程序当前使用管理员运行降低权使用普通权限运行 lindexi 2019-08-31 16:55:58 ...

  7. 随笔记录 grub引导故障修复 2019.8.7

    系统备份: [root@localhost ~]# mkdir /abc [root@localhost ~]# mount /dev/sdb1 /abc [root@localhost ~]# dd ...

  8. ReentrantLock与synchronized的区别

    1.与synchronized相比,ReentrantLock提供了更多,更加全面的功能,具备更强的扩展性.例如:时间锁等候,可中断锁等候,锁投票. 2.ReentrantLock还提供了条件Cond ...

  9. mysql 监控及优化——转载自http://www.cnblogs.com/suansuan/

    1.Mysql连接数 Mysql默认最大连接数为100. 设置Mysql的最大连接数,在Mysql的配置文件中增加: max_connections = 1000   #Mysql的最大连接数,默认如 ...

  10. 初识Qgis

    折腾了一天,qgis终于能在跟了自己8年的本本上顺利打开了,官网先后下载了3.8和3.4版本的都出现了同样的问题,"could not load qgis_app.dll",goo ...