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

语言: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邮箱作为例子,其余12六、163邮箱也同样能够,大同小异。ios

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


1、如何查询QQ的SMTP服务器:安全


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

从上面能够知道:socket

SMTP服务器:smtp.qq.com函数

监听端口号:46五、587.(使用25端口也行学习


2、打开QQ邮箱的SMTP/POP3服务:测试


打开QQ邮箱-->设置-->帐户-->找到如下选项:ui

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

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

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

若是你获取密钥后,修改了QQ密码,那么密钥会无效,那就要从新获取.

若是你早已打开了该SMTP服务,但没有密钥,关闭从新打开服务.

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


3、上传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 '/';  
}

4、开始编写:


通讯步骤:
一、链接QQ的SMTP服务器  //2开始,每个命令都要接收一个服务器响应码
二、编辑邮件内容和标题格式:
   From:<发件人>\r\n(只是用于显示
   To:<收件人>\r\n以;分割,只是用于显示
   Subject:主题\r\n\r\n"
   内容
三、发送EHLO命令标明身份
四、发送STARTTLS命令标明使用安全传输层协议(TLS)
五、发送AUTH LOGIN命令使用邮箱登陆
六、上传QQ帐号和邮箱密钥(Base64编码)
七、发送MAIL FROM命令指定发件人,就是当前QQ邮箱.
八、发送RCPT TO命令指定一个或多个收件人
九、发送DATA命令准备开始发送邮件内容
十、发送内容,内容以\r\n.\r\n结束
十一、发送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服务器三位数字响应码

缺点:

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

二、明文传输帐号和密码,有被人分析程序和抓包获得帐号密码的风险。


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


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