使用java语言基于SMTP协议手写邮件客户端

1. 说明

电子邮件是互联网上常见的应用,他是互联网早期的产品,直至今日依然受到广大用户的喜爱(在中国可能因为文化背景不同,电子邮件只在办公的时候常用)。

电子邮件系统由以下几个部分组成:

  • 用户代理
  • 邮件服务器
  • 邮件传输协议

总所周知,目前市面上流行的电子邮箱有qq邮箱,163邮箱等,我们可以去申请一个qq邮箱或者163邮箱,原因是因为腾讯和网易提供了邮件服务器。

同时我们也知道我们不仅仅可以通过qq邮箱的官方客户端收发邮件,而且可以通过其他客户端登入qq邮箱,比如说网易邮箱大师等。这说明邮件服务商提供了邮件服务器,但是用户代理却不局限与该厂商,而用户代理是通过邮件传输协议与邮件服务器进行通信的,也就是说我们只要理解了邮件传输协议,了解一门网络编程语言,就可以动手实现我们自己的邮件客户端了。

那么我们开始实现吧;

2. SMTP协议

SMTP的全称是Simple Mail Transfer Protocol,简单邮件传输协议。顾名思义,这个协议十分的简单,通过对该协议的RFC文档的阅读,我们可以掌握该协议的基本内容,了解从用户代理与邮件服务器的通信规则。

3 准备工作

  1. 阅读SMTP协议的RFC文档
  2. 搭Maven环境
  3. 编写代码

4. SMTP协议精要

RFC文档当然是英文的,本来非常害怕,但是发现他的内容其实很少,所以边看文档边查词典读了两遍(千万别怕),下面是主要内容介绍。

SMTP协议分为标准SMTP协议和扩展SMTP协议,标准SMTP协议是1982年在RFC821 文档中定义的,而扩展SMTP协议是1995年在RFC1869 文档中定义的。扩展SMTP协议在标准 SMTP协议基础上的改动非常小,主要增加了邮件安全方面的认证功能,现在我们说的SMTP协议基本上都是扩展SMTP协议。

  • introduction介绍:

    主要介绍了SMTP扩展协议为消息传输代理提供了一个稳定的高效的基础(也就是说SMTP协议可以用来实现消息传输代理客户端,也急速邮件客户端)。然后说明了扩展的内容。

  • SMTP扩展协议的框架

    SMTP传输的是邮件对象,邮件对象包括封面和内容

    • 封面包括发件人的地址,多个收件人的地址和交付模式,使用一系列的协议单元发送。
    • 内容包括头部和主题两个部分,使用SMTP数据协议单元发送,头部包括一系列键值对,头部总是使用ASCII编码
  • SMTP协议包含得多指令,但是只需要用到以下指令就可以完成简单的邮件发送

    • ehlo <domain>

      如 ehlo zeng

      与SMTP协议建立连接后需要发送的第一条命令。

    • auth para

      设置验证方式,如auth login

    • mail from: <发送者邮箱>

      设置发送者邮箱,如mail from:<xxxx@qq.com>

    • rcpt to:<收件者邮箱>

      设置收件者邮箱,如rcpt to:<xxxx@qq.com>

    • data

      表示将要发送邮件的内容,这个命令后面的发送都是邮件内容

    • quit

      结束邮件发送

所有命令末尾都是回车换行

5. 源码

import com.sun.xml.internal.messaging.saaj.util.Base64;
import lombok.Data; import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.io.PrintWriter;
import java.net.Socket; /**
* @author zeng
*/
public class MyEmailClient {
public static void main(String[] args) throws IOException {
//敏感信息。。
Token token=new Token("",25,"","");
Socket socket=null;
PrintWriter printWriter=null;
BufferedReader br=null; try {
//1. 连接smtp邮箱服务器
socket=new Socket(token.getAddress(),token.getPort());
printWriter=new PrintWriter(socket.getOutputStream(),true);
br=new BufferedReader(new InputStreamReader(socket.getInputStream()));
//2. 第一条命令 ehlo
printWriter.println("ehlo zeng");
System.out.println(br.readLine());
//3. 发送,auth
printWriter.println("auth login");
System.out.println(br.readLine());
//4. 用户名和密码
printWriter.println(token.getUserName());
printWriter.println(token.getPassWord());
//会有一大串信息返回,如果最后返回235 Authentication successful则成功
String temp=null;
while ((temp=br.readLine())!=null){
System.out.println(temp);
if ("235 Authentication successful".equals(temp)){
break;
}
}
System.out.println("认证成功"); //设置发件人和收件人,敏感信息
String sentUser="";
String recUser="";
printWriter.println("mail from:<"+sentUser+">");
System.out.println(br.readLine());
printWriter.println("rcpt to:<"+recUser+">");
System.out.println(br.readLine()); //设置data
printWriter.println("data");
System.out.println(br.readLine()); //设置邮件主题
printWriter.println("subject:test");
printWriter.println("from:"+sentUser);
printWriter.println("to:"+recUser);
//设置邮件格式
printWriter.println("Content-Type: text/plain;charset=\"utf8\"");
printWriter.println();
//邮件正文
printWriter.println("来自java手写smtp邮件客户端");
printWriter.println(".");
printWriter.print("");
System.out.println(br.readLine()); //退出
printWriter.println("rset");
System.out.println(br.readLine());
printWriter.println("quit");
System.out.println(br.readLine()); } catch (IOException e) {
e.printStackTrace();
}finally {
//释放连接
socket.close();
printWriter.close();
br.close();
}
}
}
@Data
class Token{
String address;
Integer port;
String userName;
String passWord; Token(String address, Integer port, String userName, String passWord) {
this.address = address;
this.port = port;
this.userName = new String(Base64.encode(userName.getBytes()));
this.passWord = new String(Base64.encode(passWord.getBytes()));
}
}

6.题外话

之所以写这个的原因是自己把计网考完后阅读了《计算机网络 自顶向下方法》这本书,该书应用层协议的课后习题就有一个实现邮件客户端,当时看到这个题目的时候,感觉不可思议,因为之前课堂上学计网的时候都是一些理论的知识,真没想过自己动手写代码。之前在写Spring代码的时候也用过mail相关的类,所以自己也决定通过读RFC文档去实现一个自己的邮件客户端,以帮助我发现更多的乐趣。

最新文章

  1. jmeter之连接mysql和SQL Server配置
  2. cocos2d-js 学习笔记 --安装调试(2)
  3. 8-05分支结构CASE..END
  4. gfw列表
  5. C# 读写十六进制bin 文件
  6. [转]比较Jmeter、Grinder和JAVA多线程本身压力测试所带来的性能开销
  7. nginx proxy超时报错 upstream timed out (110: Connec...
  8. php concurrence
  9. undefined reference to `switch_dev_unregister&#39;
  10. 【机器学习算法-python实现】svm支持向量机(1)—理论知识介绍
  11. Qt 5 如何修改打包好的应用程序图标
  12. Android 首次进入应用时加载引导界面
  13. NFS挂载故障卡死的问题
  14. 求逆序对常用的两种算法 ----归并排 &amp; 树状数组
  15. day2——两数相加
  16. alias重命名命令
  17. BZOJ5018:[SNOI2017]英雄联盟(背包DP)
  18. Django框架下的小人物--Cookie
  19. Update SSM agent to each EC2 via Bat and bash script
  20. [C/C++] static在C和C++中的用法和区别

热门文章

  1. 使用scratchbox2建立交叉编译环境
  2. 实现dropdownList 无刷新
  3. ASP.NET MVC控制器Controller中参数
  4. Delphi6/7 中XML 文档的应用
  5. 关于这次KPL春季决赛的感悟
  6. 使用nodejs-koa2-mysql-sequelize-jwt 实现项目api接口
  7. LCN自动补偿
  8. Storm 学习之路(八)—— Storm集成HDFS和HBase
  9. javaWeb 概念介绍
  10. Confluence5.6.6安装和破解