JavaMail是API 是一个标准的Java扩展,它是J2EE的范畴,在J2EE开发过程中可能会需要用到这个API。在学习JavaMail之前,有必须要对现在的互联网的邮件协议进行有个大体的了解。
邮件协议
在Internet中,常用的邮件操作相关的协议有3个—SMTP、POP、IMAP。
简单邮件传输协议(simple mail transferprotocol,SMTP),这个协议是邮件服务器之间相互发送邮件的协议,也是客户端利用该协议向邮件服务器端发送邮件的协议。一般一个邮件首先会被传送到某一个邮件服务器,再被邮件服务器分发到一个或多个目标邮件服务器。
邮局协议第3版(postoffice protocol version 3,POP3),协议主要用于从邮件服务器检索以得到新的邮件,大量的客户机从邮件服务器接收邮件就是使用POP3协议。
因特网消息访问协议(internet messager accessprotocol,IMAP),该协议比POP3功能更加强大,它可在接收邮件时,把邮件保存在邮件服务器中,既可在服务器中保留邮件也可把邮件下载
安装与配置JavaMail
由于JavaMail是一个扩展的部分,要进行发送接收邮件,需要两个包:
一个是JavaMail,这个包含了对SMTP、POP3、IMAP提供了支持,封装了电子邮件功能中的邮件对象、发送功能、身份认证、接收等。当前最新的版本是1.5
一个是JAF(JavaBeans Activation Framework),主要用来描述和显示邮件中的相关内容的,当前最新的版本是1.1.1
具体所需要的包,可以在本文的附件中直接下载。
邮件发送与接收
JavaMail包中的类比较多,主要用到的有会话类、地址类、邮件类、邮件发送类、邮件接收类和邮件文件夹类这些常用的类。
会话类(Session),主要用来创建邮件对象、实现邮件对象中数据的封装并可指定邮件服务器认证的客户端属性。它表示程序与某一个邮件服务器即将建立通信,在建立的过程可以进行口令认证。
地址类(Address),这个地址类主要是表示邮件发送人和邮件接收人的地址,一般主要用的是InternetAddress。
邮件类(Message),邮件消息的主要类,它包含了邮件中的所有部分,继承自一个接口Part,一般在使用的过程中直接是利用它的子类MimeMessage
邮件发送类(Transport),一般是从一个会话中获取一个邮件发送类的实例,将已经写好的邮件利用SMTP协议发送到指定的邮件服务器,在发送的过程中,首先根据指定口令连接到邮件服务器,再发送邮件。
邮件接收类(Store),这个其实就是邮件服务器中的存储库,里面放着所有的邮件文件夹
邮件文件夹类(Folder),该文件夹就是消息的具体所在文件夹,默认的邮件均在INBOX文件中。
发送邮件
基本步骤:
1 利用Properties来设置Session,一般主要设置两个mail.smtp.host和mail.smtp.auth,第一个主要是设置邮件服务器名,第二个是设置口令true或者false
2 利用Session.getInstance(Properties)启动一个与邮件服务器的连接
3 根据获取的Session来传建一个消息Message
4 定义消息的发信人地址InternetAddress和消息的收信人地址。
5 设置消息发送的主题和内容
6 利用Message.saveChanges()来存储填写的邮件信息
7 根据Session.getTransport("smtp")获取邮件发送类
8 利用发送人的用户名和密码连接到指定的邮件服务器
9 将该消息发送
注意:发送消息最重要的是要正确的找到发送消息的邮件服务器名,至于收信人的邮箱无所谓,可以是任意正确的邮件,譬如发送人是163邮件,可以发送给搜狐邮箱,新浪邮箱,QQ邮箱。
示例代码:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
|
package
whut.mailsender;
import
java.util.Properties;
import
javax.activation.DataHandler;
import
javax.activation.FileDataSource;
import
javax.mail.Address;
import
javax.mail.BodyPart;
import
javax.mail.Message;
import
javax.mail.Multipart;
import
javax.mail.Session;
import
javax.mail.Transport;
import
javax.mail.internet.InternetAddress;
import
javax.mail.internet.MimeBodyPart;
import
javax.mail.internet.MimeMessage;
import
javax.mail.internet.MimeMultipart;
import
whut.mailreceiver.MailAuthenticator;
public
class
SimpleSMTPSender {
public
static
void
main(String[] args) {
try
{
Properties props=
new
Properties();
//传递一个邮件服务器名smtp.163.com
//mail.smtp.host代表是发信人所在的邮箱服务器名
props.put(
"mail.smtp.host"
,
"smtp.163.com"
);
props.put(
"mail.smtp.auth"
,
true
);
//对于发送邮件,只需要保证发送人所在的邮件服务器正确打开就可以了
//收信人的邮箱可以是任意地址,如@163.com,@qq.com,@126.com
//创建一个程序与邮件服务器的通信
Session mailConnection=Session.getInstance(props,
null
);
Message msg=
new
MimeMessage(mailConnection);
//创建一个要输入用户名和指令的
//Session mailConnection=Session.getInstance(props,new MailAuthenticator());
//设置发送人和接受人
/*
* 群发邮件的方法
* StringBuffer buffer=new StringBuffer();
* buffer.append("[email protected],")
* buffer.append("[email protected]")
* String all=buffer.toString();
* Address[] allre=InternetAddress.parse(all);
* msg.setRecipient(Message.RecipientType.TO, allre);
*/
msg.setFrom(sender);
msg.setRecipient(Message.RecipientType.TO, receiver);
msg.setSubject(
"You must comply"
);
//msg.setContent("Hello", "text/plain");
//下面是模拟发送带附件的邮件
//新建一个MimeMultipart对象用来存放多个BodyPart对象
Multipart mtp=
new
MimeMultipart();
//------设置信件文本内容------
//新建一个存放信件内容的BodyPart对象
BodyPart mdp=
new
MimeBodyPart();
//给BodyPart对象设置内容和格式/编码方式
mdp.setContent(
"hello"
,
"text/html;charset=gb2312"
);
//将含有信件内容的BodyPart加入到MimeMultipart对象中
mtp.addBodyPart(mdp);
//设置信件的附件(用本地机上的文件作为附件)
mdp=
new
MimeBodyPart();
FileDataSource fds=
new
FileDataSource(
"f:/webservice.doc"
);
DataHandler dh=
new
DataHandler(fds);
mdp.setFileName(
"webservice.doc"
);
//可以和原文件名不一致
mdp.setDataHandler(dh);
mtp.addBodyPart(mdp);
//把mtp作为消息对象的内容
msg.setContent(mtp);
//以上为发送带附件的方式
//先进行存储邮件
msg.saveChanges();
Transport trans=mailConnection.getTransport(
"smtp"
);
String pw=
""
;
//邮件服务器名,用户名,密码
trans.connect(
"smtp.163.com"
, username, pw);
trans.sendMessage(msg, msg.getAllRecipients());
trans.close();
}
catch
(Exception e)
{
System.err.println(e);
}
finally
{
System.exit(
0
);
}
}
}
|
接受邮件
基本步骤:
1 利用Properties创建一个属性,不需要设置任何属性,之间传递Session使用
2 Session.getDefaultInstance()获取一个邮件会话
3 使用该会话向某种提供者请求一个存储库,ss.getStore("pop3");获取一个Store
4 存储库向指定的邮件服务器建立连接
5 通过getFolder("INBOX"),获取该存储库中INBOX文件夹
6 打开INBOX文件夹
7 消息处理
8 关闭文件夹
9 关闭存储库
注意:从服务器返回的邮件有可能在发送者或者接受者出现中文乱码,这里进行了乱码处理
示例代码:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
|
package
whut.mailreceiver;
import
java.io.InputStreamReader;
import
java.io.Reader;
import
java.text.SimpleDateFormat;
import
java.util.Date;
import
java.util.Enumeration;
import
java.util.Properties;
import
javax.mail.Flags;
import
javax.mail.Folder;
import
javax.mail.Header;
import
javax.mail.Message;
import
javax.mail.Session;
import
javax.mail.Store;
import
javax.mail.internet.InternetAddress;
import
javax.mail.internet.MimeUtility;
//利用POP3来读取邮件
//主要用来检测消息Message的基本信息,如发送者,收信者,时间
public
class
POP3Client {
public
static
void
main(String[] args) {
Properties props = System.getProperties();
String host =
"pop3.163.com"
;
String password =
"312313121"
;
String provider =
"pop3"
;
try
{
// 连接到POP3服务器
Session ss = Session.getDefaultInstance(props,
null
);
// 向回话"请求"一个某种提供者的存储库,是一个POP3提供者
Store store = ss.getStore(provider);
// 连接存储库,从而可以打开存储库中的文件夹,此时是面向IMAP的
store.connect(host, username, password);
// 打开文件夹,此时是关闭的,只能对其进行删除或重命名,无法获取关闭文件夹的信息
// 从存储库的默认文件夹INBOX中读取邮件
Folder inbox = store.getFolder(
"INBOX"
);
if
(inbox ==
null
) {
System.out.println(
"NO INBOX"
);
System.exit(
1
);
}
// 打开文件夹,读取信息
inbox.open(Folder.READ_ONLY);
System.out.println(
"TOTAL EMAIL:"
+ inbox.getMessageCount());
// 获取邮件服务器中的邮件
Message[] messages = inbox.getMessages();
for
(
int
i =
0
; i < messages.length; i++) {
System.out.println(
"------------Message--"
+ (i +
1
)
+
"------------"
);
// 解析地址为字符串
String from = InternetAddress.toString(messages[i].getFrom());
if
(from !=
null
) {
String cin = getChineseFrom(from);
System.out.println(
"From:"
+ cin);
}
String replyTo = InternetAddress.toString(messages[i]
.getReplyTo());
if
(replyTo !=
null
)
{
String rest = getChineseFrom(replyTo);
System.out.println(
"Reply To"
+ rest);
}
String to = InternetAddress.toString(messages[i]
.getRecipients(Message.RecipientType.TO));
if
(to !=
null
) {
String tos = getChineseFrom(to);
System.out.println(
"To:"
+ tos);
}
String subject = messages[i].getSubject();
if
(subject !=
null
)
System.out.println(
"Subject:"
+ subject);
SimpleDateFormat sdf =
new
SimpleDateFormat(
"yyyy-MM-dd HH:mm:ss"
);
Date sent = messages[i].getSentDate();
if
(sent !=
null
)
System.out.println(
"Sent Date:"
+ sdf.format(sent));
Date ress = messages[i].getReceivedDate();
if
(ress !=
null
)
System.out.println(
"Receive Date:"
+ sdf.format(ress));
// 显示消息的所有首部信息
// Enumeration headers=messages[i].getAllHeaders();
// while(headers.hasMoreElements())
// {
// Header h=(Header)headers.nextElement();
// String res=h.getName();
// String val=h.getValue();
// System.out.println(res+":"+val);
// }
// System.out.println();
// 读取消息主题部分
// Object content=messages[i].getContent();
// System.out.println(content);
// 根据指定的编码格式读出内容
// Reader read=new
// InputStreamReader(messages[i].getInputStream());
// int a=0;
// while((a=read.read())!=-1)
// {
// System.out.print((char)a);
// }
// 获取该消息的类型
// String type=messages[i].getContentType();
// String
// sender=InternetAddress.toString(messages[i].getFrom());
// System.out.println("Sender:"+sender);
// System.out.println("Content-type:"+type);
}
// 关闭连接,但不删除服务器上的消息
// false代表不是删除
inbox.close(
false
);
store.close();
}
catch
(Exception e) {
System.err.println(e);
}
}
// 解决中文乱码问题
public
static
String getChineseFrom(String res) {
String from = res;
try
{
if
(from.startsWith(
"=?GB"
) || from.startsWith(
"=?gb"
)
|| from.startsWith(
"=?UTF"
)) {
from = MimeUtility.decodeText(from);
}
else
{
from =
new
String(from.getBytes(
"ISO8859_1"
),
"GBK"
);
}
}
catch
(Exception e) {
e.printStackTrace();
}
return
from;
}
}
|
一般在读取邮件的时候会有个口令验证,可以在访问的过程中设置用于用户口令输入提示框:此时要修改一下Session的获取方法,传递一个继承了Authenticator,连接到POP3服务器,当提供者需要用户名和密码的时候,则会回调Authenticator的子类的getPasswordAuthentication必,同时须在connect中口令字段为null才能行
Session ss=Session.getDefaultInstance(props, new MailAuthenticator());
//向回话请求一个某种提供者的存储库,是一个POP3提供者
Store store=ss.getStore(provider);
//连接存储库,从而可以打开存储库中的文件夹,此时是面向IMAP的store.connect(host, null, null);
口令弹出框代码如下:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
|
package
whut.mailreceiver;
import
java.awt.Container;
import
java.awt.GridLayout;
import
java.awt.event.ActionEvent;
import
java.awt.event.ActionListener;
import
javax.mail.Authenticator;
import
javax.mail.PasswordAuthentication;
import
javax.swing.JButton;
import
javax.swing.JDialog;
import
javax.swing.JFrame;
import
javax.swing.JLabel;
import
javax.swing.JPanel;
import
javax.swing.JPasswordField;
import
javax.swing.JTextField;
//口令验证
public
class
MailAuthenticator
extends
Authenticator {
private
JDialog passwordDialog=
new
JDialog(
new
JFrame(),
true
);
private
JLabel mainLabel=
new
JLabel(
"Please enter your user name and password:"
);
private
JLabel userLabel=
new
JLabel(
"User name:"
);
private
JLabel passwordLabel=
new
JLabel(
"Password:"
);
private
JTextField usernameField=
new
JTextField(
20
);
private
JPasswordField passwordField=
new
JPasswordField(
20
);
private
JButton okButton=
new
JButton(
"OK"
);
public
MailAuthenticator()
{
this
(
""
);
}
public
MailAuthenticator(String username)
{
Container pane=passwordDialog.getContentPane();
pane.setLayout(
new
GridLayout(
4
,
1
));
pane.add(mainLabel);
JPanel p2=
new
JPanel();
p2.add(userLabel);
p2.add(usernameField);
pane.add(p2);
JPanel p3=
new
JPanel();
p3.add(passwordLabel);
p3.add(passwordField);
pane.add(p3);
JPanel p4=
new
JPanel();
p4.add(okButton);
pane.add(p4);
passwordDialog.setSize(
400
,
200
);
ActionListener al=
new
HideDialog();
okButton.addActionListener(al);
usernameField.addActionListener(al);
passwordField.addActionListener(al);
}
class
HideDialog
implements
ActionListener
{
public
void
actionPerformed(ActionEvent e)
{
passwordDialog.hide();
}
}
@Override
protected
PasswordAuthentication getPasswordAuthentication() {
// TODO Auto-generated method stub
passwordDialog.show();
//getPassword是返回的字符数组
String username=usernameField.getText();
String password=
new
String(passwordField.getPassword());
passwordField.setText(
""
);
return
new
PasswordAuthentication(username,password);
}
}
|
邮件传输中的几个类说明
Addres类,一般直接使用它的子类InternetAddress邮箱的地址,有两个特别重要的方法toString(),y用于将地址数组转换为字符串,并且多个地址以逗号隔开,parse()用于将以逗号隔开的地址解析为地址数组,这个可以为群发邮件用。
URLName类,这个目的是为了构造一个不需要该模式的协议处理器,它不会尝试连接,可以用于方便的标识文件夹和非标准的URL的存储库
Message类,主要是消息的抽象类,一般使用其子类MimeMessage,可以通过它来获取邮件的发送者,接受者,时间,主题,内容等邮件首部,查看邮件查收情况。
Part接口,一般包括了部分属性的方法、首部的方法、和内容的方法。实际运用中都是通过它的子类来调用的,可以根据属性来判断一个邮件是否含有附件。利用getAllHeaders()获得邮件的所有首部名和值,它返回的是Enumeration。
查看邮件的内容,查看MIME内容的类型是getContentType(),如返回"text/plain;charset="GBK",
一般通过getContent()来查看返回具体的内容,这里利用了JAF来自动的调整类型为java可识别的类型,当是text/plain的时候,返回是String,内容是多部分的时候,返回的是Multipart对象。如果是其他JAF也会将其转换为Java适当的对象。如果类型不能识别,则返回一个InputStream。
写入邮件的内容,一般文本型直接调用setText();也可以使用setContent(string,chartype);当要发送多个部分的时候,就需要setContent(Multipart p);每一个部分必须是MimeBodyPart,然后将其添加到MimeMultipart.在邮件的传输过程中,如果想要传输文件的话,必须将数据包装到BodyPart中,然后再添加到Multipart 中。
Folder文件,一般都是通过Session、Store或另一个Folder获得,一般获取的文件夹并不确保一定存在,要利用exists()判断。对于文件夹的一些操作必须要注意,检索和获取文件夹信息,必须要打开文件夹上才能操作。删除或者重命名文件夹,只能在文件夹关闭的时候完成。
题外:
国内常见的几个免费邮件服务器名如下:
网易免费邮箱:发送服务器:smtp.163.com 接收服务器:pop.163.com
新浪免费邮箱:发送服务器:smtp.sina.com.cn 接收服务器:pop3.sina.com.cn
搜狐邮箱:发送服务器:smtp.sohu.com 接收服务器:pop3.sohu.com
常见乱码处理方式
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
|
package
whut.mailsender;
import
javax.mail.Part;
import
javax.mail.internet.MimeUtility;
import
sun.misc.BASE64Decoder;
public
class
StringUtil {
//发信人,收信人,回执人邮件中有中文处理乱码,res为获取的地址
//http默认的编码方式为ISO8859_1
//对含有中文的发送地址,使用MimeUtility.decodeTex方法
//对其他则把地址从ISO8859_1编码转换成gbk编码
public
static
String getChineseFrom(String res) {
String from = res;
try
{
if
(from.startsWith(
"=?GB"
) || from.startsWith(
"=?gb"
)
|| from.startsWith(
"=?UTF"
)) {
from = MimeUtility.decodeText(from);
}
else
{
from =
new
String(from.getBytes(
"ISO8859_1"
),
"GBK"
);
}
}
catch
(Exception e) {
e.printStackTrace();
}
return
from;
}
//转换为GBK编码
public
static
String toChinese(String strvalue) {
try
{
if
(strvalue ==
null
)
return
null
;
else
{
strvalue =
new
String(strvalue.getBytes(
"ISO8859_1"
),
"GBK"
);
return
strvalue;
}
}
catch
(Exception e) {
return
null
;
}
}
//接收邮件时,获取某个邮件的中文附件名,出现乱码
//对于用base64编码过的中文,则采用base64解码,
//否则对附件名进行ISO8859_1到gbk的编码转换
public
static
String getFileChinese(Part part)
throws
Exception {
String temp = part.getFileName();
// part为Part实例
if
((temp.startsWith(
"=?GBK?"
) && temp.endsWith(
"?="
))
|| (temp.startsWith(
"=?gbk?b?"
) && temp.endsWith(
"?="
))) {
temp = StringUtil.getFromBASE64(temp.substring(
8
, temp.indexOf(
"?="
) -
1
));
}
else
{
temp = StringUtil.toChinese(temp);
}
return
temp;
}
public
static
String getFromBASE64(String s) {
if
(s ==
null
)
return
null
;
BASE64Decoder decoder =
new
BASE64Decoder();
try
{
byte
[] b = decoder.decodeBuffer(s);
return
new
String(b);
}
catch
(Exception e) {
return
null
;
}
}
}
|