编写SMTP协议邮箱发信(QQ邮箱为例)

时间:2020-12-26 10:13:33

语言:C/C++  
编程软件:VS2015
字符集:ASCII码
主要使用:Winsock套接字库
了解E-Mail协议:http://blog.csdn.net/aaron133/article/details/78365503
POP3邮箱收信(163邮箱为例):http://blog.csdn.net/aaron133/article/details/78413701

说明:这里拿QQ邮箱做为例子,其他126、163邮箱也一样可以,大同小异。

首先你要知道QQ提供的用于管理邮箱SMTP服务器监听端口和服务器域名或IP地址。


一、如何查询QQ的SMTP服务器:


QQ提供的SMTP服务器:http://service.mail.qq.com/cgi-bin/help?subtype=1&&id=28&&no=167

从上面可以知道:

SMTP服务器:smtp.qq.com

监听端口号:465、587.(使用25端口也行


二、打开QQ邮箱的SMTP/POP3服务:


打开QQ邮箱-->设置-->账户-->找到以下选项:

POP3/IMAP/SMTP/Exchange/CardDAV/CalDAV服务

点击开启“POP3/SMTP服务”选项.

这时,QQ会提供一个字符串密钥(如:QKBE KQQW UDVC BJAB),这个密钥充当着登录密码的作用.

如果你获取密钥后,修改了QQ密码,那么密钥会无效,那就要重新获取.

如果你早已打开了该SMTP服务,但没有密钥,关闭重新打开服务.

得到密钥:QKBE KQQW UDVC BJAB  实际使用密钥时:qkbekqqwudvcbjab


三、上传QQ账户和密钥时,要转换成“Base64编码”再上传给QQ的SMTP服务器:

(关于Base64编码的概念:http://blog.csdn.net/aaron133/article/details/78352525)


使用以下的EncodeBase64函数即可将它们转换成Base64编码:

struct Base64Date6    
{    
    unsigned int d4 : 6;      
    unsigned int d3 : 6;    
    unsigned int d2 : 6;    
    unsigned int d1 : 6;    
};    
//转换为Base64编码函数  void EncodeBase64(char *dbuf, char *buf128, int len)   {   //参数1:接收新Base64编码的缓冲区   参数2:要转换的字符串缓冲区  参数3:要转换的字符串长度    struct Base64Date6 *ddd = NULL;      int           i = 0;      char          buf[256] = { 0 };      char          *tmp = NULL;      char          cc = '\0';        memset(buf, 0, 256);      strcpy_s(buf, 256, buf128);      for (i = 1; i <= len / 3; i++)      {          tmp = buf + (i - 1) * 3;          cc = tmp[2];          tmp[2] = tmp[0];          tmp[0] = cc;          ddd = (struct Base64Date6 *)tmp;          dbuf[(i - 1) * 4 + 0] = ConvertToBase64((unsigned int)ddd->d1);          dbuf[(i - 1) * 4 + 1] = ConvertToBase64((unsigned int)ddd->d2);          dbuf[(i - 1) * 4 + 2] = ConvertToBase64((unsigned int)ddd->d3);          dbuf[(i - 1) * 4 + 3] = ConvertToBase64((unsigned int)ddd->d4);      }      if (len % 3 == 1)      {          tmp = buf + (i - 1) * 3;          cc = tmp[2];          tmp[2] = tmp[0];          tmp[0] = cc;          ddd = (struct Base64Date6 *)tmp;          dbuf[(i - 1) * 4 + 0] = ConvertToBase64((unsigned int)ddd->d1);          dbuf[(i - 1) * 4 + 1] = ConvertToBase64((unsigned int)ddd->d2);          dbuf[(i - 1) * 4 + 2] = '=';          dbuf[(i - 1) * 4 + 3] = '=';      }      if (len % 3 == 2)      {          tmp = buf + (i - 1) * 3;          cc = tmp[2];          tmp[2] = tmp[0];          tmp[0] = cc;          ddd = (struct Base64Date6 *)tmp;          dbuf[(i - 1) * 4 + 0] = ConvertToBase64((unsigned int)ddd->d1);          dbuf[(i - 1) * 4 + 1] = ConvertToBase64((unsigned int)ddd->d2);          dbuf[(i - 1) * 4 + 2] = ConvertToBase64((unsigned int)ddd->d3);          dbuf[(i - 1) * 4 + 3] = '=';      }      return;  }  //辅助计算的函数  char ConvertToBase64(char uc)  {      if (uc < 26)      {          return 'A' + uc;      }      if (uc < 52)      {          return 'a' + (uc - 26);      }      if (uc < 62)      {          return '0' + (uc - 52);      }      if (uc == 62)      {          return '+';      }      return '/';  }  

四、开始编写:


通信步骤:
1、连接QQ的SMTP服务器  //2开始,每一个命令都要接收一个服务器响应码
2、编辑邮件内容和标题格式:
   From:<发件人>\r\n(只是用于显示
   To:<收件人>\r\n以;分割,只是用于显示
   Subject:主题\r\n\r\n"
   内容
3、发送EHLO命令标明身份
4、发送STARTTLS命令标明使用安全传输层协议(TLS)
5、发送AUTH LOGIN命令使用邮箱登录
6、上传QQ账号和邮箱密钥(Base64编码)
7、发送MAIL FROM命令指定发件人,就是当前QQ邮箱.
8、发送RCPT TO命令指定一个或多个收件人
9、发送DATA命令准备开始发送邮件内容
10、发送内容,内容以\r\n.\r\n结束
11、发送QUIT命令退出

头文件:

#pragma once  
#include <Windows.h>
#include <stdio.h>
#include <stdlib.h>
#include <iostream>
using namespace std;
#pragma comment(lib, "Ws2_32.lib")

源文件:

#include "头文件"  

//将要发送的邮件内容 (这里的From和To只用于在邮件上显示的)
char EmailContents[] = "From:<3093575@qq.com>\r\n" //这里只是说明发件人(QQ会检查,标记为垃圾邮件而发送失败)
"To:<109333576@qq.com>;<519194462@qq.com>\r\n" //这里只是说明收件人(可以乱填的)
"Subject:第一封测试邮件\r\n\r\n" //邮件标题
"Hello World, Hello Email!"; //邮件内容

//上面的EncodeBase64函数这里就不列出来了
//上面的ConvertToBase64函数这里就不列出来了
int main()
{
system("color 4e");
char buf[1500];
char login[128];
char pass[128];
ZeroMemory(buf, 1500);
ZeroMemory(login, 128);
ZeroMemory(pass, 128);
//加载Winsock库
WSADATA WSAData;
WSAStartup(MAKEWORD(2, 2), &WSAData);
//创建TCP套接字
SOCKET sockfd = socket(AF_INET, SOCK_STREAM, 0);
SOCKADDR_IN dest;
//端口25
dest.sin_port = htons(25);//587也可以
dest.sin_family = AF_INET;
//获取smtp.qq.com服务器域名所代表的IP地址
hostent* hptr = gethostbyname("smtp.qq.com");
memcpy(&dest.sin_addr.S_un.S_addr, hptr->h_addr_list[0], hptr->h_length);
//连接腾讯的邮件SMTP服务器
int ok = connect(sockfd, (SOCKADDR*)&dest, sizeof(SOCKADDR));
if (ok != 0)
exit(0);
//用于向服务器标明用户身份,可以为发件人的服务器域名或者主机名
sprintf_s(buf, 1500, "EHLO Aaron-PC\r\n");
send(sockfd, buf, strlen(buf), 0);
//接收服务器响应
ZeroMemory(buf, 1500);
recv(sockfd, buf, 1500, 0);
cout << "QQ SMTP Server Say: \r\n" << buf << endl;
//开启安全传输层协议(TLS)
//这里的STARTTLS空格的后面要加一些字符才行,这里不是很懂,但这就可以了.
sprintf_s(buf, 1500, "STARTTLS a\r\n");
send(sockfd, buf, strlen(buf), 0);
//接收服务器响应
ZeroMemory(buf, 1500);
recv(sockfd, buf, 1500, 0);
cout << "QQ SMTP Server Say: \r\n" << buf << endl;
//邮箱认证,发送该命令后依次发送邮箱账号和密码(账号密码均使用Base64编码)
sprintf_s(buf, 1500, "AUTH LOGIN\r\n");
send(sockfd, buf, strlen(buf), 0);
//接收服务器响应
ZeroMemory(buf, 1500);
recv(sockfd, buf, 1500, 0);
cout << "Auth Login Receive:\r\n " << buf << endl;
//发送QQ账号
ZeroMemory(buf, 1500);
sprintf_s(buf, 1500, "3093575@qq.com");//你的邮箱账号
//将QQ账号转换为Base64编码再发送
EncodeBase64(login, buf, strlen(buf));
sprintf_s(buf, 1500, "%s\r\n", login);
send(sockfd, buf, strlen(buf), 0);
cout << "Base64 UserName: " << buf << endl;
//接收服务器响应
ZeroMemory(buf, 1500);
recv(sockfd, buf, 1500, 0);
cout << "User Login Receive: \r\n" << buf << endl;
//发送密钥(相当于QQ密码,在上面的步骤得到)
sprintf_s(buf, 1500, "qkbekqqwudvcbjab");
//将密钥转换为Base64编码再发送
EncodeBase64(pass, buf, strlen(buf));
sprintf_s(buf, 1500, "%s\r\n", pass);
send(sockfd, buf, strlen(buf), 0);
cout << "Base64 Password:\r\n " << buf << endl;
//接收服务器响应
ZeroMemory(buf, 1500);
recv(sockfd, buf, 1500, 0);
cout << "Send Password Receive:\r\n " << buf << endl;
//发送MAIL FROM命令指定发件人
ZeroMemory(buf, 1500);
sprintf_s(buf, 1500, "MAIL FROM: <3093575@qq.com>\r\n");
send(sockfd, buf, strlen(buf), 0);
//接收服务器响应
ZeroMemory(buf, 1500);
recv(sockfd, buf, 1500, 0);
cout << "Set Mail From Receive:\r\n " << buf << endl;
//发送一个或多个RCPT TO命令指定收件人
sprintf_s(buf, 1500, "RCPT TO:<%s>\r\n", "109333576@qq.com"); //109333576@qq.com为收件人1
send(sockfd, buf, strlen(buf), 0);
//接收服务器响应
ZeroMemory(buf, 1500);
recv(sockfd, buf, 1500, 0);
cout << "Tell Sendto Receive: \r\n" << buf << endl;
sprintf_s(buf, 1500, "RCPT TO:<%s>\r\n", "509333576@qq.com"); //509333576@qq.com为收件人2
send(sockfd, buf, strlen(buf), 0);
//接收服务器响应
ZeroMemory(buf, 1500);
recv(sockfd, buf, 1500, 0);
cout << "Tell Sendto Receive: \r\n" << buf << endl;
//发送一个DATA命令表示准备开始发送邮件内容
sprintf_s(buf, 1500, "DATA\r\n");
send(sockfd, buf, strlen(buf), 0);
//接收服务器响应
ZeroMemory(buf, 1500);
recv(sockfd, buf, 1500, 0);
cout << "Send Mail Prepare Receive: \r\n" << buf << endl;
//发送邮件内容,格式:以\r\n.\r\n作为结束标记
sprintf_s(buf, 1500, "%s\r\n.\r\n", EmailContents); //邮件的内容
send(sockfd, buf, strlen(buf), 0);
//接收服务器响应
ZeroMemory(buf, 1500);
recv(sockfd, buf, 1500, 0);
cout << "Send Mail Receive: \r\n" << buf << endl;
//使用QUIT命令退出
sprintf_s(buf, 1500, "QUIT\r\n");
send(sockfd, buf, strlen(buf), 0);
ZeroMemory(buf, 1500);
recv(sockfd, buf, 1500, 0);
cout << "Quit Receive: " << buf << endl;
//清理工作
closesocket(sockfd);
WSACleanup();
getchar();
return 0;
}

补充说明:想要知道其中细节的问题,需要学习SMTP协议,例如SMTP命令以及SMTP服务器三位数字响应码

缺点:

1、360会报告有第三方软件在使用邮箱发件.

2、如果想要编写钓鱼软件的小朋友,这里给你们一个警告,密码不能写常量、全局变量中.

      应将账号与密码加密,函数内解密,尽量做复杂一点,而且不要使用常用邮箱,防止别人反汇编查到密码.

      (当然这对有网络基础的人拿到密码毫无压力,LSP和截封包都能取到密码)


与QQ的SMTP服务器交流过程:

编写SMTP协议邮箱发信(QQ邮箱为例)

测试发送结果:(我发送到了一个163邮箱、自己的QQ邮箱中)

编写SMTP协议邮箱发信(QQ邮箱为例)