如何通过Java发送邮件来验证电子邮件地址是否真实存在

6
我们的Web应用程序会向输入其电子邮件ID的每个用户发送电子邮件。但是,我如何确保用户输入的电子邮件ID是有效的呢?实际上,当任何用户输入电子邮件ID时,我们会向他的电子邮件ID发送链接以激活帐户。我有一段发送电子邮件的代码。但是,即使电子邮件ID不存在,它也不会给我任何错误。请问您如何解决这个问题?如果电子邮件ID确实不存在,它应该给出一些错误信息。
以下是我的代码:
    package csv;
    import javax.mail.PasswordAuthentication;
    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.MessagingException;
    import javax.mail.Multipart;
    import javax.mail.Session;
    import javax.mail.Transport;
    import javax.mail.internet.AddressException;
    import javax.mail.internet.InternetAddress;
    import javax.mail.internet.MimeBodyPart;
    import javax.mail.internet.MimeMessage;
    import javax.mail.internet.MimeMultipart;

    public class email {

public void send(String recipeintEmail, 
        String subject, 
        String messageText,String[] attachments) 
        throws MessagingException, AddressException {
    /*
       It is a good practice to put this in a java.util.Properties 
       file and encrypt password. Scroll down 
       to comments below to see 
       how to use java.util.Properties in JSF context. 
    */
    String senderEmail = "our email address";
    String senderMailPassword = "password";
    String gmail = "smtp.gmail.com";

    Properties props = System.getProperties();

    props.put("mail.smtp.user", senderEmail);
    props.put("mail.smtp.host", "smtp.gmail.com");
    props.put("mail.smtp.port", "465");
    props.put("mail.smtp.starttls.enable", "true");
    props.put("mail.smtp.debug", "true");
    props.put("mail.smtp.auth", "true");
    props.put("mail.smtp.socketFactory.port", "465");
    props.put("mail.smtp.socketFactory.class", 
          "javax.net.ssl.SSLSocketFactory");
    props.put("mail.smtp.socketFactory.fallback", "false");

    // Required to avoid security exception.
    email.MyAuthenticator authentication = 
          new email.MyAuthenticator(senderEmail,senderMailPassword);
    Session session = 
          Session.getDefaultInstance(props,authentication);
    session.setDebug(true);

    MimeMessage message = new MimeMessage(session);

    BodyPart messageBodyPart = new MimeBodyPart();      
    messageBodyPart.setText(messageText);

    // Add message text
    Multipart multipart = new MimeMultipart();
    multipart.addBodyPart(messageBodyPart);

    // Attachments should reside in your server.
    // Example "c:\file.txt" or "/home/user/photo.jpg"

    for (int i=0; i < attachments.length; i++) {        

        messageBodyPart = new MimeBodyPart();       
        DataSource source = new FileDataSource(attachments[i]);
        messageBodyPart.setDataHandler(new DataHandler(source));
        messageBodyPart.setFileName(attachments [i]);          
        multipart.addBodyPart(messageBodyPart) ;  
    }



    message.setContent(multipart);                
    message.setSubject(subject);
    message.setFrom(new InternetAddress(senderEmail));
    message.addRecipient(Message.RecipientType.TO,
        new InternetAddress(recipeintEmail));

    Transport transport = session.getTransport("smtps");
    transport.connect(gmail,465, senderEmail, senderMailPassword);
    transport.sendMessage(message, message.getAllRecipients());

    transport.close();

}

private class MyAuthenticator extends javax.mail.Authenticator {
    String User;
    String Password;
    public MyAuthenticator (String user, String password) {
        User = user;
        Password = password;
    }

    @Override
    public PasswordAuthentication getPasswordAuthentication() {
        return new javax.mail.PasswordAuthentication(User, Password);
    }
}


public static void main(String args[]) throws MessagingException
{
    // email e=new email();
   // String at[]={"c:/COPYRIGHT.txt"};
  //  e.send("xyz@gmail.com", "hello","test"  )");
}

}
4个回答

4

没有绝对可靠的方法来做这件事。你可以尝试这篇博客文章中所述的步骤,但不保证适用于所有种类的邮件服务器/转发设置。

向用户发送激活密钥和URL,这将使提供有效电子邮件地址成为登录/使用你所提供内容的必要条件。


有没有办法处理像xyz@xyz.com这样的虚假邮件地址?上面的代码告诉我邮件已成功发送到xyz@xyz.com,即使它不是有效的。 - Kanchan Ruia
你可以进行的唯一快速验证是检查电子邮件格式 - 这意味着文本是否符合电子邮件ID的语法。否则,您将不得不依赖于博客文章中提到的机制,在某些情况下可能有效。 - ring bearer

3
您可以使用Java Mail API来验证电子邮件。
    try {
            //
            // Create InternetAddress object and validated the supplied
            // address which is this case is an email address.
            InternetAddress internetAddress = new InternetAddress(email);
            internetAddress.validate();
            isValid = true;
        } 
        catch (AddressException e) {
            System.out.println("You are in catch block -- Exception Occurred for: " + email);
        }

您可以从http://www.oracle.com/technetwork/java/index-138643.html下载Java邮件API。


2
如果您打算坚持通过发送电子邮件进行验证,我建议您非常清楚地说明,在允许用户访问其注册的内容之前,您将发送电子邮件验证链接,以减少虚假电子邮件地址的数量。我们已经进行了测试,并且在注册表单标签中做出一些小的更改,显著降低了无效电子邮件的比率。
需要注意的是,如果您使用发送电子邮件来验证电子邮件是否有效,并且向大量无效地址发送电子邮件,特别是在面向消费者的域名(例如 Yahoo、GMail、AOL 等)上,您可能会被标记为垃圾邮件发送者,因为硬退信(发送至无效地址)会影响您的声誉评分。我几个月前写了一篇博客文章,介绍了验证电子邮件的不同方法,包括免费和商业验证方式,可能会有用。

0

解决这个问题的一种方法是跟踪退回的邮件。但由于标准不统一,这种方法存在一些困难。

然而,做点什么总比什么都不做要好。有时候,知道用户是否收到了电子邮件并且没有回复,或者电子邮件地址本身是否不正确变得很重要。

您可能想要查看此链接: http://www.oracle.com/technetwork/java/faq-135477.html#bounce

请参阅来自链接的摘录:

问:当无法传递消息时,会返回失败消息。我如何检测这些“退回”的消息?

答:虽然有一个用于报告此类错误的互联网标准(多部分/报告MIME类型,请参见RFC1892),但它尚未广泛实施。 RFC1211深入讨论了这个问题,包括大量示例。

在互联网电子邮件中,特定邮箱或用户名的存在只能由将传递消息的最终服务器确定。消息可能通过多个中继服务器(无法检测错误)传递到达最终服务器。

通常情况下,当终端服务器检测到此类错误时,它会向原始消息的发送者返回一条指示失败原因的消息。有许多互联网标准涵盖此类传递状态通知,但大量服务器不支持这些新标准,而是使用特定技术返回此类失败消息。
这使得将“退回”的消息与导致问题的原始消息进行关联非常困难。(请注意,此问题完全独立于JavaMail。)JavaMail现在包括支持解析传递状态通知的功能;有关详细信息,请参见JavaMail软件包中的NOTES.txt文件。
处理此问题的方法和启发式技术有很多,但没有一种是完美的。其中一种技术是可变信封返回路径(Variable Envelope Return Paths),在http://cr.yp.to/proto/verp.txt中有描述。

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接