0

0

JAVA+PHP+阿里云组件纯手工实现POP、SMTP、IMAP开发邮件服务器(2)

php中文网

php中文网

发布时间:2016-06-13 12:27:52

|

1451人浏览过

|

来源于php中文网

原创

JAVA+PHP+阿里云组件纯手工实现POP、SMTP、IMAP开发邮件服务器(二)

 java开发邮件服务器的接收模块

    用java建立socket服务端,监听端口25,实现smtp协议。即可完成邮件服务器的接收模块。

    这里要注意的是,SMTP协议其实可以分为两种。一种是你用手机、PC等客户端发邮件到邮件服务商的服务器的时候用的SMTP协议,这一类是需要登录验证的。一种是邮件服务商之间传递邮件的SMTP协议,此类协议是不需要登录的。比如你用Foxmail上你的QQ邮箱发送了一封邮件到163的邮箱。过程是这样的:

  1. 邮件从Foxmail通过SMTP协议发送到QQ邮箱的服务器。

  2. QQ邮箱的服务器通过SMTP协议将邮件投递到163的邮件服务器。

  3. 对方通过IMAP或者POP协议从163邮箱服务器拿到这封邮件。

    立即学习PHP免费学习笔记(深入)”;

 

    本文将实现的是不需要登录的SMTP协议。下面展示该协议:

C:telnet smtp.126.com 25   /* 以telnet方式连接126邮件服务器 */S:220 126.com Anti-spam GT for Coremail System (126com[071018]) /* 220为响应数字,其后的为欢迎信息,会应服务器不同而不同*/C:HELO smtp.126.com /* HELO 后用来填写返回域名(具体含义请参阅RFC821),但该命令并不检查后面的参数*/S:250 OKC: MAIL FROM: [email protected] /* 发送者邮箱 */S:250 … ./* “…”代表省略了一些可读信息 */C:RCPT TO: [email protected]/* 接收者邮箱 */S:250 … ./* “…”代表省略了一些可读信息 */C:DATA  /* 请求发送数据 */S:354 Enter mail, end with "." on a line by itselfC:Enjoy Protocol StudingC:.S:250 Message sentC:QUIT /* 退出连接 */S:221 Bye

    所以,我们建立Socket服务端,并监听25端口后,只要检测客户端发来的信息,并给出相应的回复,即可完成邮件的接收。具体java实现socket可以百度。下面解释几个重要的语法用法:

 

  1. MAIL FROM:这里后面跟的参数是发送者的邮箱。当你接收到邮件的正文(eml后缀的文件),并解析后,里面还有一个标签标记的是邮件的发件人。这两个参数理论上应该是一样的。当不一样的时候,QQ邮箱等会显示“该邮件由***代发”。所以我们接收邮件的时候,也要检测一下这两个邮件地址是否相同,避免有人恶意欺骗。另外,最重要的是SPF检查,我们下一篇单独介绍。SPF在反垃圾邮件方面的作用很大。比如MAIL FROM:[email protected]  ,我们就会去查cliyun.com的SPF记录,看看里面有没有包含当前连接我们socket服务端的客户端的IP地址,如果不包含,说明该客户端是欺骗我们的。

    Hika AI
    Hika AI

    Hika AI是一个免费的AI智能搜索引擎

    下载
  2. RCPT TO:这里标记的是收件人邮箱。当你完成邮件服务器的时候,你会发现网络上很多恶意的程序在大肆发送垃圾邮件,会给本来不存在的用户邮箱所在的服务器发送垃圾邮件。比如你的域名是cliyun.com,但是你的邮箱用户里,[email protected]??[email protected]?,应该把该邮件发送者的IP加入敏感列表,利用算法进行过滤。

  3. DATA:DATA之后,就是邮件EML正文,知道遇到单独一行的 . 结束。通常我们在解析这份EML文件之前,会把它本地保存一下,方便解析出问题后(比如有文字乱码),可以在调整代码后再次解析。

 

    邮件接收器的重点是EML文件的解析。通常使用javamail模块来解析,网上教程很多,但是很多都是不全的。因为一个完整的eml文件,含有很多信息。比如标题、正文、发件人、收件人、抄送、回复地址、附件等等。下面贴一段比较好的解析类:

import java.io.BufferedInputStream;import java.io.BufferedOutputStream;import java.io.File;import java.io.FileOutputStream;import java.io.InputStream;import java.text.SimpleDateFormat;import java.util.Date;import java.util.Properties;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.Session;import javax.mail.Store;import javax.mail.internet.InternetAddress;import javax.mail.internet.MimeMessage;import javax.mail.internet.MimeUtility; /** * @author yh *  */public class ShowMail {     private MimeMessage mimeMessage = null;    private String saveAttachPath = ""; // 附件下载后的存放目录    private StringBuffer bodyText = new StringBuffer(); // 存放邮件内容的StringBuffer对象    private String dateFormat = "yy-MM-dd HH:mm"; // 默认的日前显示格式     /**     * 构造函数,初始化一个MimeMessage对象     */    public ShowMail() {    }     public ShowMail(MimeMessage mimeMessage) {        this.mimeMessage = mimeMessage;        System.out.println("创建一个ReceiveEmail对象....");    }     public void setMimeMessage(MimeMessage mimeMessage) {        this.mimeMessage = mimeMessage;        System.out.println("设置一个MimeMessage对象...");    }     /**     *  * 获得发件人的地址和姓名       */    public String getFrom() throws Exception {        InternetAddress address[] = (InternetAddress[]) mimeMessage.getFrom();        String from = address[0].getAddress();        if (from == null) {            from = "";            System.out.println("无法知道发送者.");        }        String personal = address[0].getPersonal();         if (personal == null) {            personal = "";            System.out.println("无法知道发送者的姓名.");        }         String fromAddr = null;        if (personal != null || from != null) {            fromAddr = personal + "<" + from + ">";            System.out.println("发送者是:" + fromAddr);        } else {            System.out.println("无法获得发送者信息.");        }        return fromAddr;    }     /**     *  * 获得邮件的收件人,抄送,和密送的地址和姓名,根据所传递的参数的不同     *  * "to"----收件人 "cc"---抄送人地址 "bcc"---密送人地址       */    public String getMailAddress(String type) throws Exception {        String mailAddr = "";        String addType = type.toUpperCase();         InternetAddress[] address = null;        if (addType.equals("TO") || addType.equals("CC")                || addType.equals("BCC")) {             if (addType.equals("TO")) {                address = (InternetAddress[]) mimeMessage                        .getRecipients(Message.RecipientType.TO);            } else if (addType.equals("CC")) {                address = (InternetAddress[]) mimeMessage                        .getRecipients(Message.RecipientType.CC);            } else {                address = (InternetAddress[]) mimeMessage                        .getRecipients(Message.RecipientType.BCC);            }             if (address != null) {                for (int i = 0; i < address.length; i++) {                    String emailAddr = address[i].getAddress();                    if (emailAddr == null) {                        emailAddr = "";                    } else {                        System.out.println("转换之前的emailAddr: " + emailAddr);                        emailAddr = MimeUtility.decodeText(emailAddr);                        System.out.println("转换之后的emailAddr: " + emailAddr);                    }                    String personal = address[i].getPersonal();                    if (personal == null) {                        personal = "";                    } else {                        System.out.println("转换之前的personal: " + personal);                        personal = MimeUtility.decodeText(personal);                        System.out.println("转换之后的personal: " + personal);                    }                    String compositeto = personal + "<" + emailAddr + ">";                    System.out.println("完整的邮件地址:" + compositeto);                    mailAddr += "," + compositeto;                }                mailAddr = mailAddr.substring(1);            }        } else {            throw new Exception("错误的电子邮件类型!");        }        return mailAddr;    }     /**     *  * 获得邮件主题       */    public String getSubject() throws MessagingException {        String subject = "";        try {            System.out.println("转换前的subject:" + mimeMessage.getSubject());            subject = MimeUtility.decodeText(mimeMessage.getSubject());            System.out.println("转换后的subject: " + mimeMessage.getSubject());            if (subject == null) {                subject = "";            }        } catch (Exception exce) {            exce.printStackTrace();        }        return subject;    }     /**     *  * 获得邮件发送日期       */    public String getSentDate() throws Exception {        Date sentDate = mimeMessage.getSentDate();        System.out.println("发送日期 原始类型: " + dateFormat);        SimpleDateFormat format = new SimpleDateFormat(dateFormat);        String strSentDate = format.format(sentDate);        System.out.println("发送日期 可读类型: " + strSentDate);        return strSentDate;    }     /**     *  * 获得邮件正文内容       */    public String getBodyText() {        return bodyText.toString();    }     /**     *   * 解析邮件,把得到的邮件内容保存到一个StringBuffer对象中,解析邮件     *   * 主要是根据MimeType类型的不同执行不同的操作,一步一步的解析        */     public void getMailContent(Part part) throws Exception {         String contentType = part.getContentType();        // 获得邮件的MimeType类型        System.out.println("邮件的MimeType类型: " + contentType);         int nameIndex = contentType.indexOf("name");         boolean conName = false;         if (nameIndex != -1) {            conName = true;        }         System.out.println("邮件内容的类型: " + contentType);         if (part.isMimeType("text/plain") && conName == false) {            // text/plain 类型            bodyText.append((String) part.getContent());        } else if (part.isMimeType("text/html") && conName == false) {            // text/html 类型            bodyText.append((String) part.getContent());        } else if (part.isMimeType("multipart/*")) {            // multipart/*            Multipart multipart = (Multipart) part.getContent();            int counts = multipart.getCount();            for (int i = 0; i < counts; i++) {                getMailContent(multipart.getBodyPart(i));            }        } else if (part.isMimeType("message/rfc822")) {            // message/rfc822            getMailContent((Part) part.getContent());        } else {         }    }     /**     *   * 判断此邮件是否需要回执,如果需要回执返回"true",否则返回"false"       */    public boolean getReplySign() throws MessagingException {         boolean replySign = false;         String needReply[] = mimeMessage                .getHeader("Disposition-Notification-To");         if (needReply != null) {            replySign = true;        }        if (replySign) {            System.out.println("该邮件需要回复");        } else {            System.out.println("该邮件不需要回复");        }        return replySign;    }     /**     * 获得此邮件的Message-ID        */    public String getMessageId() throws MessagingException {        String messageID = mimeMessage.getMessageID();        System.out.println("邮件ID: " + messageID);        return messageID;    }     /**     * 判断此邮件是否已读,如果未读返回false,反之返回true     */    public boolean isNew() throws MessagingException {        boolean isNew = false;        Flags flags = ((Message) mimeMessage).getFlags();        Flags.Flag[] flag = flags.getSystemFlags();        System.out.println("flags的长度: " + flag.length);        for (int i = 0; i < flag.length; i++) {            if (flag[i] == Flags.Flag.SEEN) {                isNew = true;                System.out.println("seen email...");                // break;            }        }        return isNew;    }     /**     * 判断此邮件是否包含附件     */    public boolean isContainAttach(Part part) throws Exception {        boolean attachFlag = false;        // String contentType = part.getContentType();        if (part.isMimeType("multipart/*")) {            Multipart mp = (Multipart) part.getContent();            for (int i = 0; i < mp.getCount(); i++) {                BodyPart mPart = mp.getBodyPart(i);                String disposition = mPart.getDisposition();                if ((disposition != null)                        && ((disposition.equals(Part.ATTACHMENT)) || (disposition                                .equals(Part.INLINE))))                    attachFlag = true;                else if (mPart.isMimeType("multipart/*")) {                    attachFlag = isContainAttach((Part) mPart);                } else {                    String conType = mPart.getContentType();                     if (conType.toLowerCase().indexOf("application") != -1)                        attachFlag = true;                    if (conType.toLowerCase().indexOf("name") != -1)                        attachFlag = true;                }            }        } else if (part.isMimeType("message/rfc822")) {            attachFlag = isContainAttach((Part) part.getContent());        }        return attachFlag;    }     /**     *  * 保存附件       */     public void saveAttachMent(Part part) throws Exception {        String fileName = "";        if (part.isMimeType("multipart/*")) {            Multipart mp = (Multipart) part.getContent();            for (int i = 0; i < mp.getCount(); i++) {                BodyPart mPart = mp.getBodyPart(i);                String disposition = mPart.getDisposition();                if ((disposition != null)                        && ((disposition.equals(Part.ATTACHMENT)) || (disposition                                .equals(Part.INLINE)))) {                    fileName = mPart.getFileName();                    if (fileName.toLowerCase().indexOf("gb2312") != -1) {                        fileName = MimeUtility.decodeText(fileName);                    }                    saveFile(fileName, mPart.getInputStream());                } else if (mPart.isMimeType("multipart/*")) {                    saveAttachMent(mPart);                } else {                    fileName = mPart.getFileName();                    if ((fileName != null)                            && (fileName.toLowerCase().indexOf("GB2312") != -1)) {                        fileName = MimeUtility.decodeText(fileName);                        saveFile(fileName, mPart.getInputStream());                    }                }            }        } else if (part.isMimeType("message/rfc822")) {            saveAttachMent((Part) part.getContent());        }    }     /**     * 设置附件存放路径     */    public void setAttachPath(String attachPath) {        this.saveAttachPath = attachPath;    }     /**     *  * 设置日期显示格式       */    public void setDateFormat(String format) throws Exception {        this.dateFormat = format;    }     /**     *  * 获得附件存放路径       */    public String getAttachPath() {        return saveAttachPath;    }     /**     *  * 真正的保存附件到指定目录里       */    private void saveFile(String fileName, InputStream in) throws Exception {        String osName = System.getProperty("os.name");        String storeDir = getAttachPath();        String separator = "";        if (osName == null) {            osName = "";        }        if (osName.toLowerCase().indexOf("win") != -1) {            separator = "\\";            if (storeDir == null || storeDir.equals(""))                storeDir = "c:\\tmp";        } else {            separator = "/";            storeDir = "/tmp";        }        File storeFile = new File(storeDir + separator + fileName);        System.out.println("附件的保存地址: " + storeFile.toString());        // for(int i=0;storefile.exists();i++){        // storefile = new File(storedir+separator+fileName+i);        // }        BufferedOutputStream bos = null;        BufferedInputStream bis = null;         try {            bos = new BufferedOutputStream(new FileOutputStream(storeFile));            bis = new BufferedInputStream(in);            int c;            while ((c = bis.read()) != -1) {                bos.write(c);                bos.flush();            }        } catch (Exception exception) {            exception.printStackTrace();            throw new Exception("文件保存失败!");        } finally {            bos.close();            bis.close();        }    }     /**     * ReceiveEmail类测试     */    public static void main(String args[]) throws Exception {        String host = "pop.sina.com";        String username = "***";        String password = "***";         Properties props = new Properties();        Session session = Session.getDefaultInstance(props, null);         Store store = session.getStore("pop3");        store.connect(host, username, password);         Folder folder = store.getFolder("INBOX");        folder.open(Folder.READ_ONLY);        Message message[] = folder.getMessages();        System.out.println("邮件数量: " + message.length);        ShowMail re = null;         for (int i = 0; i < message.length; i++) {            re = new ShowMail((MimeMessage) message[i]);            System.out.println("邮件 " + i + " 主题: " + re.getSubject());            System.out.println("邮件 " + i + " 发送时间: " + re.getSentDate());            System.out.println("邮件 " + i + " 是否需要回复: " + re.getReplySign());            System.out.println("邮件 " + i + " 是否已读: " + re.isNew());            System.out.println("邮件 " + i + " 是否包含附件: "                    + re.isContainAttach((Part) message[i]));            System.out.println("邮件 " + i + " 发送人地址: " + re.getFrom());            System.out                    .println("邮件 " + i + " 收信人地址: " + re.getMailAddress("to"));            System.out.println("邮件 " + i + " 抄送: " + re.getMailAddress("cc"));            System.out.println("邮件 " + i + " 暗抄: " + re.getMailAddress("bcc"));            re.setDateFormat("yy年MM月dd日 HH:mm");            System.out.println("邮件 " + i + " 发送时间: " + re.getSentDate());            System.out.println("邮件 " + i + " 邮件ID: " + re.getMessageId());            re.getMailContent((Part) message[i]);            System.out.println("邮件 " + i + " 正文内容: \r\n" + re.getBodyText());            re.setAttachPath("e:\\");            re.saveAttachMent((Part) message[i]);        }    }}

 

    解析完EML文件,将信息存入数据库后,邮件服务器的接收器就完成工作了。实际生产环境中,为了项目的健壮性,接收器会有很多个,比如颗粒云邮箱的邮件接收器有15台,一封邮件发过来,怎么知道该投递到哪台服务器呢?这是取决于域名的MX记录的。比如qq.com的MX记录里有很多台邮件接收服务器,他们每个的优先级不同。一般都是根据优先级由高到低投递邮件。当一台投递失败后,就切换到另一台。如何通过命令行来查看一个域名的MX记录呢?

QQ截图20151127195216.jpg

这部分是邮件服务器的发件器需要做的工作。我们会在后面的博文中,介绍如何用JAVA检测邮件地址的MX记录。

原文地址:JAVA+PHP+阿里云组件纯手工实现POP、SMTP、IMAP开发邮件服务器(二)

相关文章

PHP速学教程(入门到精通)
PHP速学教程(入门到精通)

PHP怎么学习?PHP怎么入门?PHP在哪学?PHP怎么学才快?不用担心,这里为大家提供了PHP速学教程(入门到精通),有需要的小伙伴保存下载就能学习啦!

下载

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
包子漫画网页版入口与全集阅读指南_正版免费漫画快速访问方法
包子漫画网页版入口与全集阅读指南_正版免费漫画快速访问方法

本专题汇总了包子漫画官网和网页版入口,提供最新章节抢先看方法、正版免费阅读指南,以及稳定访问方式,帮助用户快速直达包子漫画页面,无广告畅享全集漫画内容。

47

2026.02.10

MC.JS网页版快速畅玩指南_MC.JS官网在线入口及免安装体验方法
MC.JS网页版快速畅玩指南_MC.JS官网在线入口及免安装体验方法

本专题汇总了MC.JS官网入口和网页版快速畅玩方法,提供免安装访问、不同版本(1.8.8、1.12.8)在线体验指南,以及正版网页端操作说明,帮助玩家轻松进入MC.JS世界,实现即时畅玩与高效体验。

34

2026.02.10

谷歌邮箱网页版登录与注册全指南_Gmail账号快速访问与安全操作教程
谷歌邮箱网页版登录与注册全指南_Gmail账号快速访问与安全操作教程

本专题汇总了谷歌邮箱网页版的最新登录入口和注册方法,详细提供官方账号快速访问方式、网页版操作教程及安全登录技巧,帮助用户轻松管理Gmail邮箱账户,实现高效、安全的邮箱使用体验。

25

2026.02.10

铁路12306订票与退改全攻略_高效购票与座位选取技巧
铁路12306订票与退改全攻略_高效购票与座位选取技巧

本专题全面汇总铁路12306订票、退票、改签及候补订单操作技巧,提供车厢座位分布参考、抢票攻略和高铁安检注意事项,帮助新手用户快速掌握高效购票与退改流程,提高出行效率和体验。

31

2026.02.10

TensorFlow2深度学习模型实战与优化
TensorFlow2深度学习模型实战与优化

本专题面向 AI 与数据科学开发者,系统讲解 TensorFlow 2 框架下深度学习模型的构建、训练、调优与部署。内容包括神经网络基础、卷积神经网络、循环神经网络、优化算法及模型性能提升技巧。通过实战项目演示,帮助开发者掌握从模型设计到上线的完整流程。

0

2026.02.10

Vue3组合式API与组件开发实战
Vue3组合式API与组件开发实战

本专题讲解 Vue 3 组合式 API 的核心概念与应用技巧,深入分析响应式系统、生命周期管理、组件设计与复用策略。通过完整项目案例,指导前端开发者实现高性能、结构清晰的 Vue 应用,提升开发效率与代码可维护性。

4

2026.02.10

Go语言微服务架构与gRPC实战
Go语言微服务架构与gRPC实战

本专题面向有 Go 基础的开发者,系统讲解微服务架构设计与 gRPC 的高效应用。内容涵盖服务拆分、RPC 通信、负载均衡、错误处理、服务注册与发现等关键技术。通过实战案例,帮助开发者搭建高性能、可扩展的 Go 微服务系统。

1

2026.02.10

React 18状态管理与Hooks高级实践
React 18状态管理与Hooks高级实践

本专题专注于 React 18 的高级开发技术,详细讲解 useState、useEffect、useReducer、useContext 等 Hooks 的使用技巧,以及 Redux、Zustand 等状态管理工具的集成与优化方法。通过真实案例,帮助前端开发者构建可维护、性能优良的现代 React 应用。

4

2026.02.10

Node.js后端开发与Express框架实践
Node.js后端开发与Express框架实践

本专题针对初中级 Node.js 开发者,系统讲解如何使用 Express 框架搭建高性能后端服务。内容包括路由设计、中间件开发、数据库集成、API 安全与异常处理,以及 RESTful API 的设计与优化。通过实际项目演示,帮助开发者快速掌握 Node.js 后端开发流程。

2

2026.02.10

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
深入剖析redis教程
深入剖析redis教程

共55课时 | 8.2万人学习

Redis中文开发手册
Redis中文开发手册

共0课时 | 0人学习

麦子学院深入浅出 redis 视频教程
麦子学院深入浅出 redis 视频教程

共20课时 | 4.5万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号