java实现简单邮件的发送以及常见问题

之前自己这边是有一个sendmail的组件,但是只能实现text的发送,今天逛博客,无意间发现了一个能够实现发送附件的senfmail功能的java实现,于是就进行了测试和实现,经过测试能够成功实现附件功能,希望通过发博的方式进行收藏,也把中间出现的问题和大家进行分享:

java实现简单邮件的发送以及常见问题

最近遇到个需求需要实现发送邮件的功能,以前做发送邮件功能都是有邮箱用户名密码,通过用户名密码连接对应的SMTP服务器来实现邮件的发送。但是这次用公司内部的邮箱,大体原理基本相同,但是给分配的邮箱并未提供密码,所以需要实现邮箱免密发送邮件。

1.首先在Mail工程下新建lib文件夹,导入javax.mail.jar包;

java mail 实现

import java.util.*;
import javax.mail.*;
import javax.mail.internet.*;

public class SendEmail {
    public static void main(String[] args) {
        // 收件人电子邮箱
        String to = "****@***.com";

        // 发件人电子邮箱
        String from = "****@***.com";

        // 指定发送邮件的主机
        String host = "";

        // 获取系统属性
        Properties properties = System.getProperties();

        // 设置邮件服务器
        properties.setProperty("mail.smtp.host", host);
        properties.setProperty("mail.smtp.auth", "false");
        properties.setProperty("mail.stmp.from", from);
        // ehlo被设置为false的时候连接邮件服务器不需要验证
        properties.setProperty("mail.smtp.ehlo", "false");

        // 获取默认session对象
        Session session = Session.getDefaultInstance(properties);
        // 开启debug模式
        session.setDebug(true);
        try {
            // 创建默认的 MimeMessage 对象
            MimeMessage message = new MimeMessage(session);

            // Set From: 头部头字段
            message.setFrom(new InternetAddress(from));

            // Set To: 头部头字段
            message.addRecipient(Message.RecipientType.TO, new InternetAddress(
                    to));

            // Set Subject: 头部头字段
            message.setSubject("This is the Subject Line!");

            // 设置消息体
            message.setText("This is actual message");

            // 发送消息
            Transport.send(message);
            System.out.println("Sent message successfully....");
        } catch (MessagingException mex) {
            mex.printStackTrace();
        }
    }
}

mail.smtp.ehlo
被设置为fales的时候连接邮件服务器不需要验证,如果不设置默认为true需要服务器需要验证。
mail.smtp.ehlo
为false的时候连接SMTP服务器是helo模式,为true的时候是ehlo模式。

2.在Mail项目下新建conf文件夹,新建file:mail-info.properties,进行配置;

commons-email 实现

commons-email其实是对java mail 的封装,更进一步方便使用了。
下面代码是实现了有用户名密码发送邮件。

import org.apache.commons.mail.DefaultAuthenticator;
import org.apache.commons.mail.Email;
import org.apache.commons.mail.EmailException;
import org.apache.commons.mail.SimpleEmail;

public class MailUtil {
    private static String _USER_NAME_ = "****@***.com";
    private static String _PASSWORD_ = "***";
    private static String _SMTP_ADDRESS_ = "smtp.com";
    private final static int _SMTP_PORT_SSL_ = 465;
    private static String from = "****@***.com";

    public static void sendMail(String toAddress, String subject, String content) {
        Email email = new SimpleEmail();
        email.setHostName(_SMTP_ADDRESS_);
        email.setSmtpPort(_SMTP_PORT_SSL_);
        email.setAuthenticator(new DefaultAuthenticator(_USER_NAME_, _PASSWORD_));
        // email.setSSLOnConnect(true);
        email.setSSL(true);// commons-mail-1.1支持的方法,1.4中使用setSSLOnConnect(true)代替
        try {
            email.setFrom(from);
            email.setSubject(subject);
            email.setMsg(content);
            email.addTo(toAddress);
            email.send();
        } catch (EmailException e) {
            e.printStackTrace();
        }
    }

    public static void main(String[] args) {
        // test
        String subject = "测试邮件";
        String content = "test test test。。。测试内容。。。这是从MailUtil.java发来的。";
        String to = "****@***.com";
        MailUtil.sendMail(to, subject, content);
    }
}
mail.host=smtp.126.com
mail.port=25
mail.username=zhanglincumt@126.com
mail.password=******
mail.from=zhanglincumt@126.com
mail.to=1447363383@qq.com

常见问题

3.新建一个util的package,里面放读取配置文件的AppContext.java和保存临时配置文件的SystemContains.java;

一、运行过程抛出异常

1、Exception in thread “main” java.lang.NoClassDefFoundError:
com/sun/mail/util/LineInputStream

原因:此异常多出现在J2EE版本的邮件服务中,因为J2EE原有jar中包含用于邮件收发的mail.jar,只是版本可能比较低。

解决方法:手动删除J2EE版本里的JavaMail相关的jar包(mail.jar),导入自己下载的新版本mail.jar和activation.jar。

2、Exception in thread “main” java.lang.NoClassDefFoundError:
javax/activation/DataSource

原因:没有导入activation.jar,编译时异常,运行时如果没有使用附件功能的话会正常运行。

解决方法:导入activation.jar

package util;

import java.io.IOException;
import java.io.InputStream;
import java.util.Properties;
/**
 * 读取配置文件
 * 2014年12月31日
 */
public enum AppContext {

    INSTANCE;

    public Properties configuration = new Properties();

    public void init() {
        InputStream is = this.getClass().getResourceAsStream("/mail-info.properties");
        if (is != null) {
            try {
                this.configuration.clear();
                this.configuration.load(is);
            } catch (IOException e) {
            } finally {
                try {
                    is.close();
                } catch (Throwable t) {}
            }
        }
    }

    public String getConfigValue(String key) {
          return this.configuration.getProperty(key);
    }

}

package util;
/**
 * 临时保存配置文件信息
 */
public class SystemConstants {
    //info
    public static final String MAILHOST = AppContext.INSTANCE.getConfigValue("mail.host");
    public static final String MAILPORT = AppContext.INSTANCE.getConfigValue("mail.port");
    public static final String MAILUSERNAME = AppContext.INSTANCE.getConfigValue("mail.username");
    public static final String MAILPASSWORD = AppContext.INSTANCE.getConfigValue("mail.password");
    public static final String MAILFROM = AppContext.INSTANCE.getConfigValue("mail.from");
    public static final String MAILTO = AppContext.INSTANCE.getConfigValue("mail.to");
}

二、发送过程出现错误,一般不是代码问题

1、’550 5.1.1 test@163.com: Recipient address rejected: User unknown
in virtual mailbox table’,

你发送的帐号在邮件系统内不存在,请检查你的邮件地址是否输入有误

2、’554 5.7.1 test@163.com: Sender address rejected: Access denied’,

你的邮箱是内部帐号,你所发送的地址不在授权域之内

3、’553 5.7.1 test@163.com澳门新葡亰3522平台游戏,: Sender address rejected: not logged in’,

发送邮件需要SMTP身份验证,你的帐号SMTP身份验证部分没有设置正确,请检查配置

4、’454 4.7.1 <unknown[...]>: Client host rejected:
Access denied’

你的IP被管理员屏蔽,请检查你部的机器是否被感染病毒自动发送邮件

4.在mail包下新建发送文件类JavaMail.java和压缩解压缩类ZIP.java;

package mail;

import java.io.File;

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

import util.AppContext;
import util.SystemConstants;
/**
 * send email
 * 2014年12月31日
 */

public class JavaMail {
    /**
     * Message对象将存储我们实际发送的电子邮件信息,
     * Message对象被作为一个MimeMessage对象来创建并且需要知道应当选择哪一个JavaMail session。
     */
    private MimeMessage message;

    /**
     * Session类代表JavaMail中的一个邮件会话。
     * 每一个基于JavaMail的应用程序至少有一个Session(可以有任意多的Session)。
     * 
     * JavaMail需要Properties来创建一个session对象。
     * 寻找"mail.smtp.host"    属性值就是发送邮件的主机
     * 寻找"mail.smtp.auth"    身份验证,目前免费邮件服务器都需要这一项
     */
    private Session session;

    /***
     * 邮件是既可以被发送也可以被受到。JavaMail使用了两个不同的类来完成这两个功能:Transport 和 Store。 
     * Transport 是用来发送信息的,而Store用来收信。对于这的教程我们只需要用到Transport对象。
     */
    private Transport transport;

    /*
     * 初始化方法
     */
    public JavaMail(boolean debug) {
        session = Session.getInstance(AppContext.INSTANCE.configuration);
        session.setDebug(debug);//开启后有调试信息
        message = new MimeMessage(session);
    }

    /**
     * 发送邮件
     * @param subject     邮件主题
     * @param sendHtml    邮件内容
     */
    public void sendEmail(String subject, String sendHtml) {
        try {
            // 发件人
            //InternetAddress from = new InternetAddress(sender_username);
            // 下面这个是设置发送人的Nick name
            InternetAddress from = new InternetAddress(MimeUtility.encodeWord("zhanglincc")+" <"+SystemConstants.MAILUSERNAME+">");
            message.setFrom(from);

            // 收件人
            InternetAddress to = new InternetAddress(SystemConstants.MAILTO);
            message.setRecipient(Message.RecipientType.TO, to);//还可以有CC、BCC

            // 邮件主题
            message.setSubject(subject);

            String content = sendHtml.toString();

            // 邮件内容,也可以使纯文本"text/plain"
            message.setContent(content, "text/html;charset=UTF-8");

            // 保存邮件
            message.saveChanges();

            // smtp验证,就是你用来发邮件的邮箱用户名密码            
            transport = session.getTransport("smtp");
            transport.connect(SystemConstants.MAILHOST, SystemConstants.MAILUSERNAME, SystemConstants.MAILPASSWORD);

            // 发送
            transport.sendMessage(message, message.getAllRecipients());

            System.out.println("send success!");
        } catch (Exception e) {
            e.printStackTrace();
        }finally {
            if(transport!=null){
                try {
                    transport.close();
                } catch (MessagingException e) {
                    e.printStackTrace();
                }
            }
        }
    }

    /**
     * 发送邮件
     * @param subject 邮件主题
     * @param sendHtml  邮件内容
     * @param attachment  附件
     */
    public void sendEmail(String subject, String sendHtml, String filePath) {
        File file = new File(filePath);
        if(file.exists()){
            try {
                // 发件人
                InternetAddress from = new InternetAddress(SystemConstants.MAILUSERNAME);
                message.setFrom(from);

                String [] strings = SystemConstants.MAILTO.split(";");
                Address [] receiverAddress = new Address[strings.length];

                for(int i = 0;i<strings.length;i++){
                    receiverAddress[i]=new InternetAddress(strings[i]);
                }

                message.setRecipients(Message.RecipientType.TO, receiverAddress);

                // 邮件主题
                message.setSubject(subject);

                // 向multipart对象中添加邮件的各个部分内容,包括文本内容和附件
                Multipart multipart = new MimeMultipart();

                // 添加邮件正文
                BodyPart contentPart = new MimeBodyPart();
                contentPart.setContent(sendHtml, "text/html;charset=UTF-8");
                multipart.addBodyPart(contentPart);

                //如果所给的路径是目录就压缩
                if(file.isDirectory()){
                    File fileTemp = new File(file.getParentFile().getAbsoluteFile()+"\"+file.getName()+".zip");
                    System.out.println(fileTemp.getAbsolutePath());
                    ZIP.zipFiles(file.listFiles(),fileTemp);
                    file = fileTemp;
                    System.out.println(file.isFile());
                }
                // 添加附件的内容
                if (file.isFile()) {
                    BodyPart attachmentBodyPart = new MimeBodyPart();
                    DataSource source = new FileDataSource(file);
                    attachmentBodyPart.setDataHandler(new DataHandler(source));

                    // 网上流传的解决文件名乱码的方法,其实用MimeUtility.encodeWord就可以很方便的搞定
                    // 这里很重要,通过下面的Base64编码的转换可以保证你的中文附件标题名在发送时不会变成乱码
                    //sun.misc.BASE64Encoder enc = new sun.misc.BASE64Encoder();
                    //messageBodyPart.setFileName("=?GBK?B?" + enc.encode(attachment.getName().getBytes()) + "?=");

                    //MimeUtility.encodeWord可以避免文件名乱码
                    attachmentBodyPart.setFileName(MimeUtility.encodeWord(file.getName()));
                    multipart.addBodyPart(attachmentBodyPart);
                }

                // 将multipart对象放到message中
                message.setContent(multipart);
                // 保存邮件
                message.saveChanges();

                transport = session.getTransport("smtp");
                // smtp验证,就是你用来发邮件的邮箱用户名密码
                transport.connect(SystemConstants.MAILHOST, SystemConstants.MAILUSERNAME, SystemConstants.MAILPASSWORD);
                // 发送
                transport.sendMessage(message, message.getAllRecipients());

                System.out.println("send success!");
            } catch (Exception e) {
                e.printStackTrace();
            } finally {
                if (transport != null) {
                    try {
                        transport.close();
                    } catch (MessagingException e) {
                        e.printStackTrace();
                    }
                }
            }
        }else{
            System.out.println("附件地址有误  ");
            System.exit(-1);
        }
    }

}

package mail;

import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.util.Enumeration;
import java.util.zip.ZipEntry;
import java.util.zip.ZipFile;
import java.util.zip.ZipOutputStream;

/**
 * 压缩文件、解压压缩文件
 * 2014年12月31日
 */
public class ZIP {

    /**
     * 功能:压缩多个文件成一个zip文件
     * @param srcfile:源文件列表
     * @param zipfile:压缩后的文件
     */
    public static void zipFiles(File[] srcfile, File zipfile) {
        byte[] buf = new byte[1024];
        try {
            // ZipOutputStream类:完成文件或文件夹的压缩
            ZipOutputStream out = new ZipOutputStream(new FileOutputStream(zipfile));
            for (int i = 0; i < srcfile.length; i++) {
                FileInputStream in = new FileInputStream(srcfile[i]);
                out.putNextEntry(new ZipEntry(srcfile[i].getName()));
                int len;
                while ((len = in.read(buf)) > 0) {
                    out.write(buf, 0, len);
                }
                out.closeEntry();
                in.close();
            }
            out.close();
            System.out.println("压缩完成.");
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

    /**
     * 功能:解压缩
     * @param zipfile:需要解压缩的文件
     * @param descDir:解压后的目标目录
     * @throws IOException
     */
    @SuppressWarnings("rawtypes")
    public static void unZipFiles(File zipfile, String descDir) {
        File file = new File(descDir);
        if (!file.exists()) {
            try {
                file.mkdir();
            } catch (Exception e) {
                e.printStackTrace();
            }
        }
        try {
            ZipFile zf = new ZipFile(zipfile);
            for (Enumeration entries = zf.entries(); entries.hasMoreElements();) {
                ZipEntry entry = (ZipEntry) entries.nextElement();
                String zipEntryName = entry.getName();
                InputStream in = zf.getInputStream(entry);
                OutputStream out = new FileOutputStream(descDir + zipEntryName);
                byte[] buf1 = new byte[1024];
                int len;
                while ((len = in.read(buf1)) > 0) {
                    out.write(buf1, 0, len);
                }
                in.close();
                out.close();
                System.out.println("解压缩完成.");
            }

            zf.close();
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

    public static void main(String[] args) {
        // 需要解压缩的文件
        File file = new File("D:\test");
        File filenew = new File("D:\test.zip");
        zipFiles(file.listFiles(), filenew);

        // 解压后的目标目录
        String dir = "D:\workspace\";
        unZipFiles(filenew, dir);
    }
}

5.新建测试类进行测试;

package main;

import util.AppContext;
import mail.JavaMail;

public class Main {
    public static void main(String[] args) {
        AppContext.INSTANCE.init();
        JavaMail mail = new JavaMail(false);
        mail.sendEmail("来自lynnzhangcc的邮件", "我是邮件内容");
        mail.sendEmail("测试附件为文件夹的邮件主题", "测试附件为文件夹的邮件内容","D:\test");
        mail.sendEmail("测试附件为文件的邮件主题", "测试附件文件的邮件内容","D:\test");
    }
}

如果运行正常,console会打印send success!的语句。