使用javamail发信过程中的一些问题及解决方法

今天在研究javamail发信的过程中,出现了一些小问题,现总结如下,以免后来者走些不必要的弯路,先把完整的能够正常运行的代码示例粘贴如下: 

import java.util.Properties;

import javax.activation.DataHandler;
import javax.activation.DataSource;
import javax.activation.FileDataSource;
import javax.mail.BodyPart;
import javax.mail.Message;
import javax.mail.Multipart;
import javax.mail.PasswordAuthentication;
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 javax.mail.internet.MimeUtility;

public class MailExample {

	public static void main(String args[]) throws Exception {
		//发件人使用发邮件的电子信箱服务器
		String host = "smtp.126.com";
		//发邮件的出发地(发件人的信箱)
		String from = "junJZ_2008@126.com";
		//发邮件的目的地(收件人信箱)
		String to = "junJZ_2008@163.com";

		// Get system properties
		Properties props = new Properties();

		// Setup mail server
		props.put("mail.smtp.host", host);

		//发送邮件时是否以授权方式访问邮件服务器
		props.put("mail.smtp.auth", "true");

		MyAuthenticator myauth = new MyAuthenticator("junJZ_2008@126.com", "XXX");
		// Get session
		Session session = Session.getDefaultInstance(props, myauth);

		//如果需要在发送邮件过程中监控mail命令的话,可以在发送前设置debug标志
		//session.setDebug(true);

		// Define message
		MimeMessage message = new MimeMessage(session);

		// Set the from address
		message.setFrom(new InternetAddress(from));

		// Set the to address
		message.addRecipient(Message.RecipientType.TO, new InternetAddress(to));

		// Set the subject
		message.setSubject("测试程序!");

		//用于存储邮件体(包括正文与附件)
		Multipart multipart = new MimeMultipart();

		//==============>>>>>以普通方式显示邮件体,并带附件
		//--创建邮件正文
		// Create the message part 
		BodyPart messageBodyPart = new MimeBodyPart();
		// Fill the message
		messageBodyPart.setText("这是用java写的发送电子邮件的测试程序!");
		//或以下方式设置文本也行
		//messageBodyPart.setContent("这是用java写的发送电子邮件的测试程序!", "text/plain");
		multipart.addBodyPart(messageBodyPart);

		//--创建邮件附件
		// Part two is attachment
		messageBodyPart = new MimeBodyPart();
		DataSource source = new FileDataSource("e:/tmp/test.gif");
		messageBodyPart.setDataHandler(new DataHandler(source));

		/*
		 * 邮件头(参见RFC822,RFC2047)只能包含US-ASCII字符。邮件头中任何包含非US-ASCII字符
		 * 的部分必须进行编码,使其只包含US-ASCII字符。所以使用java mail发送中文邮件必须经过编
		 * 码,否则别人收到你的邮件只能是乱码一堆。不过使用java mail 包的解决方法很简单,用它自
		 * 带的MimeUtility工具中encode开头的方法(如encodeText)对中文信息进行编码就可以了。
		 */
		//		messageBodyPart.setFileName("附件名");
		//让javamail决定用什么方式来编码 ,编码内容的字符集是系统字符集
		messageBodyPart.setFileName(MimeUtility.encodeText("附件名"));
		//使用指定的base64方式编码,并指定编码内容的字符集是GBK
		//messageBodyPart.setFileName(MimeUtility.encodeText("附件名", "GBK", "B"));
		multipart.addBodyPart(messageBodyPart);

		// Put parts in message
		message.setContent(multipart);
		//<<<<<<<==============

		//		//==============>>>>>以HTML方式显示邮件体,图片附在附件中
		//		// Create your new message part
		//		BodyPart messageBodyPart = new MimeBodyPart();
		//		String htmlText = "<H1>Hello</H1>" + "<img src=\"cid:memememe\">";
		//		messageBodyPart.setContent(htmlText, "text/html");
		//
		//		// Create a related multi-part to combine the parts
		//		multipart = new MimeMultipart("related");
		//		multipart.addBodyPart(messageBodyPart);
		//
		//		// Create part for the image
		//		messageBodyPart = new MimeBodyPart();
		//
		//		// Fetch the image and associate to part
		//		DataSource fds = new FileDataSource("e:/tmp/test.gif");
		//		messageBodyPart.setDataHandler(new DataHandler(fds));
		//		messageBodyPart.setHeader("Content-ID", "<memememe>");
		//
		//		// Add part to multi-part
		//		multipart.addBodyPart(messageBodyPart);
		//
		//		// Associate multi-part with message
		//		message.setContent(multipart);
		//		//<<<<<<<==============

		message.saveChanges(); // implicit with send()

		//Transport.send(message);

		/*
		 * 下面发送方法是一个很好的方法,尤其是在我们在同一个邮件服务器上发送多个邮件时。因为这时
		 * 我们将在连接邮件服务器后连续发送邮件,然后再关闭掉连接。Transport.send(message)这
		 * 个基本的方法是在每次调用时进行与邮件服务器的连接的,对于在同一个邮件服务器上发送多个邮
		 * 件来讲可谓低效的方式,所以采用下面的发送方式会好点
		 */
		Transport transport = session.getTransport("smtp");
		transport.connect(host, "junJZ_2008@126.com", "XXX");
		transport.sendMessage(message, message.getAllRecipients());
		transport.close();

	}
}

class MyAuthenticator extends javax.mail.Authenticator {
	private String strUser;
	private String strPwd;

	public MyAuthenticator(String user, String password) {
		this.strUser = user;
		this.strPwd = password;
	}

	protected PasswordAuthentication getPasswordAuthentication() {
		return new PasswordAuthentication(strUser, strPwd);
	}
}
 

注意:上面的事例仅为使用163信箱时发送电子邮件的方法,因为使用的host为:smtp.163.com,如源代码 中:String host = "smtp.163.com";   //发件人使用发邮件的电子信箱服务器,如果使用其它的电子邮件发送,就必须在其 邮件服务器上查找相应的电子邮件服务器,例如搜狐就要使用smtp.sohu.com,具体情况具体对待,都可以从所使用的邮件服务器上获得的。如果没有 使用host ,也就是说,没有进行props.put("mail.smtp.host", host);设置,那么就会抛 javax.mail.MessagingException: Could not connect to SMTP host: localhost, port: 25; 的异常。当然了,如果你没有正确配置,这个异常仍然会被抛出的。


有些邮件服务系统是不需要验证发件人的授权的,所以可以很简单的使用
    Session session = Session.getDefaultInstance(props, null);
             而不必使用
    props.put("mail.smtp.auth", "true"); 
    MyAuthenticator myauth = new MyAuthenticator("你自己的电子信箱", "你自己的信箱密码");
    Session session = Session.getDefaultInstance(props, myauth);

就可以发送电子邮件了,这个多为一些企事业单位的内部电子信箱系统。
但是对于很多门户网站上的电邮系统,如:163,sohu,yahoo等等,如果仍然简单的这样使用就会抛

com.sun.mail.smtp.SMTPSendFailedException: 553 authentication is required,smtp8,wKjADxuAyCAfmnZE8BwtIA==.32705S2


 at com.sun.mail.smtp.SMTPTransport.issueSendCommand(SMTPTransport.java:1388)

 at com.sun.mail.smtp.SMTPTransport.mailFrom(SMTPTransport.java:959)

 at com.sun.mail.smtp.SMTPTransport.sendMessage(SMTPTransport.java:583)

 at javax.mail.Transport.send0(Transport.java:169)

 at javax.mail.Transport.send(Transport.java:98)

这样的异常,要求你必须进行授权校验,它的目的就是阻止他人任意乱发邮件,也算是为了减少垃圾邮件的出现吧。这时候,我们就要使用
    props.put("mail.smtp.auth", "true"); 
    MyAuthenticator myauth = new MyAuthenticator("你自己的电子信箱", "你自己的信箱密码");
    Session session = Session.getDefaultInstance(props, myauth);

这里还有一个特别注意的事情:在你使用Session.getDefaultInstance时,一定要 将    props.put("mail.smtp.auth", "true"); 置为true,它默认的是false,如果你没有做这一步,虽然 你使用了Session.getDefaultInstance(props, myauth);,你自己也确 实    MyAuthenticator myauth = new MyAuthenticator("你自己的电子信箱", "你自己的信箱密 码");但是它仍然会抛出
com.sun.mail.smtp.SMTPSendFailedException: 553 authentication is required,smtp8,wKjADxJA2SBrm3ZEFv0gIA==.40815S2


 at com.sun.mail.smtp.SMTPTransport.issueSendCommand(SMTPTransport.java:1388)

 at com.sun.mail.smtp.SMTPTransport.mailFrom(SMTPTransport.java:959)

 at com.sun.mail.smtp.SMTPTransport.sendMessage(SMTPTransport.java:583)

 at javax.mail.Transport.send0(Transport.java:169)

 at javax.mail.Transport.send(Transport.java:98)
这样的异常。我就在这一步费了好长时间,后来才发现了这个问题,很是郁闷。不过还好,总算解决了。

其实上面的做法只是比较简单的一种,也有很多其它的写法,如:
Properties props = System.getProperties();可以使用
Properties props = new Properties();来代替。


Transport.send(message);可以使用下面的代码来代替
      String username = "你的电子信箱用户名";
      String password = "你的电子信箱密码";
      message.saveChanges(); //    implicit with send()
      Transport transport = session.getTransport("smtp");
      transport.connect("mail.htf.com.cn", username, password);
      transport.sendMessage(message, message.getAllRecipients());
      transport.close();
这种方法在同时发送多封电子邮件时比较有用。


另附上使用org.apache.commons.mail进行发电子邮件的示例:

import org.apache.commons.mail.SimpleEmail;
import org.apache.commons.mail.*;

public class TestCommon {
  public TestCommon() {
  }
  public static void main(String[] args){
    SimpleEmail email = new SimpleEmail();
    email.setHostName("smtp.163.com");//设置使用发电子邮件的邮件服务器
    try {
      email.addTo("收件人信箱");
      email.setAuthentication("发件人信箱","发件人信箱密码");
      email.setFrom("发件人信箱");
      email.setSubject("Test apache.commons.mail message");
      email.setMsg("This is a simple test of commons-email");
      email.send();
    }
    catch (EmailException ex) {
      ex.printStackTrace();
    }
  }
} 

 

几个有用的链接:

使用java mail 包收发中文邮件的编码,解码问题以及解决方法


JavaMail API详解

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值