使用java语言基于SMTP协议手写邮件客户端 1. 说明
电子邮件是互联网上常见的应用,他是互联网早期的产品,直至今日依然受到广大用户的喜爱(在中国可能因为文化背景不同,电子邮件只在办公的时候常用)。
电子邮件系统由以下几个部分组成:
用户代理
邮件服务器
邮件传输协议
总所周知,目前市面上流行的电子邮箱有qq邮箱,163邮箱等,我们可以去申请一个qq邮箱或者163邮箱,原因是因为腾讯和网易提供了邮件服务器。
同时我们也知道我们不仅仅可以通过qq邮箱的官方客户端收发邮件,而且可以通过其他客户端登入qq邮箱,比如说网易邮箱大师等。这说明邮件服务商提供了邮件服务器,但是用户代理却不局限与该厂商,而用户代理是通过邮件传输协议与邮件服务器进行通信的,也就是说我们只要理解了邮件传输协议,了解一门网络编程语言,就可以动手实现我们自己的邮件客户端了。
那么我们开始实现吧;
2. SMTP协议SMTP的全称是Simple Mail Transfer Protocol,简单邮件传输协议。顾名思义,这个协议十分的简单,通过对该协议的RFC文档的阅读,我们可以掌握该协议的基本内容,了解从用户代理与邮件服务器的通信规则。
3 准备工作阅读SMTP协议的RFC文档
搭Maven环境
编写代码
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.题外话