Windows 10 Enterprise Version 10.0.15063
(下载地址:http://www.itellyou.cn/)
Visual Studio Enterprise 2017 Version 15.2(26430.12)
(下载地址:https://www.visualstudio.com/downloads)
OpenSSL v1.1.0f
(下载地址:https://slproweb.com/products/Win32OpenSSL.html)
1。使用OpenSSL有两种方法,第一种为下载源码自行编译后使用;第二种为下载已编译好的安装包,安装后直接可以使用。因第一种方法十分繁琐,需要安装Ruby进行编译,同时编译过程中可能产生各种各种蛋疼的问题,因此本文采用第二种方法
2。很多情况下可能并不需要直接调用OpenSSL进行开发,有时libcurl可能是更好的选择(libcurl调用了OpenSSL并进行了高度封装,libcurl官网拥有大量的代码例子和详细的开发文档),本博客亦提供配置libcurl开发环境方法
1。下载编译好的OpenSSL安装包(下载地址见上方),此处有Win32
和Win64
可选,这里的位数指的是你调用OpenSSL开发出来的软件的位数版本,而不是你自己计算机的位数。开发32位软件选择Win32
,64位选择Win64
,如果同时需要开发32位和64位的则下载两个。
确定好位数后,下载Win32/64 OpenSSL v1.1.0f
(版本号可能会因更新而有所不同),Light
版为“轻量版”,即只包含了核心功能的版本,本文演示的为“非Light版”
2。下载完后进行安装,本文选择把dll复制到OpenSSL目录下(为了以后好找)
6。点击左方的VC++ Directories
,然后左上角选择All Configurations
7。右上角选择平台,这里演示x64
的,如果是开发Win32软件就选Win32
8。找到Include Directories
,将OpenSSL安装目录下的include
文件夹添加进去
9。同样的,将OpenSSL安装目录下的lib
文件夹添加到Library Directories
中
10。将OpenSSL安装目录下bin
文件夹中的libcrypto-1_1-x64.dll
和libssl-1_1-x64.dll
(名字后面的版本号可能因更新而不同)复制到工程目录下
12。添加lib文件,这里有A、B两种方法,分别列出:
12-A:
在代码中添加:
#pragma comment(lib,"libssl.lib")
#pragma comment(lib,"libcrypto.lib")
12-B:
在Property Pages-
>Linker
->Input
->Additional Dependencies
中添加libssl.lib
和libcrypto.lib
1。这段代码是用C++和Windows API写的,添加lib使用了方法A
2。这段代码是用https://www.baidu.com来测试的,如果想要修改为其他测试地址,除了修改第25行的wstrHost外,还应修改第77行的strWrite
3。代码真特么冗长,真的不考虑一下使用libcurl吗?
#include
#include
#include
#include
#include
#pragma comment(lib,"ws2_32.lib")
#pragma comment(lib,"libssl.lib")
#pragma comment(lib,"libcrypto.lib")
CONST INT RECV_SIZE = 8192;
INT _tmain(INT argc, LPTSTR argv[])
{
//启动wsa
WSADATA wsadData;
WSAStartup(MAKEWORD(2, 2), &wsadData);
//获取Host的IP地址等信息
ADDRINFOT aiHints;
ZeroMemory(&aiHints, sizeof(ADDRINFOT));
aiHints.ai_family = AF_INET;
aiHints.ai_flags = AI_PASSIVE;
aiHints.ai_protocol = 0;
aiHints.ai_socktype = SOCK_STREAM;
std::wstring wstrHost = TEXT("www.baidu.com");
PADDRINFOT paiResult;
GetAddrInfo(wstrHost.c_str(), NULL, &aiHints, &paiResult);
//创建套接字
SOCKET sSocket = socket(AF_INET, SOCK_STREAM, 0);
if (sSocket == SOCKET_ERROR)
{
std::wcout << "Error socket" << std::endl;
return -1;
}
//连接Host
SOCKADDR_IN sinHost;
sinHost.sin_addr = ((LPSOCKADDR_IN)paiResult->ai_addr)->sin_addr;
sinHost.sin_family = AF_INET;
sinHost.sin_port = htons(443);
if (connect(sSocket, (LPSOCKADDR)&sinHost, sizeof(SOCKADDR_IN)) == SOCKET_ERROR)
{
std::wcout << "Error connect" << std::endl;
return -1;
}
//初始化OpenSSL库
SSL_library_init();
SSLeay_add_ssl_algorithms();
SSL_load_error_strings();
//创建SSL会话环境等
SSL_CTX *pctxSSL = SSL_CTX_new(TLSv1_2_client_method());
if (pctxSSL == NULL)
{
std::wcout << "Error SSL_CTX_new" << std::endl;
return -1;
}
SSL *psslSSL = SSL_new(pctxSSL);
if (psslSSL == NULL)
{
std::wcout << "Error SSL_new" << std::endl;
return -1;
}
SSL_set_fd(psslSSL, sSocket);
INT iErrorConnect = SSL_connect(psslSSL);
if (iErrorConnect < 0)
{
std::wcout << "Error SSL_connect, iErrorConnect=" << iErrorConnect << std::endl;
return -1;
}
std::wcout << "SSL connection using " << SSL_get_cipher(psslSSL) << std::endl;
//发包
std::string strWrite =
"GET https://www.baidu.com/ HTTP/1.1\r\n"
"Host: www.baidu.com\r\n"
"Connection: close\r\n\r\n";
INT iErrorWrite = SSL_write(psslSSL, strWrite.c_str(), strWrite.length()) < 0;
if (iErrorWrite < 0)
{
std::wcout << "Error SSL_write" << std::endl;
return -1;
}
//收包并输出
//这里接受的是char形式的,所以中文会乱码
//如果要正常显示中文,需要再转换为wchar_t或std::wstring
LPSTR lpszRead = new CHAR[RECV_SIZE];
INT iLength=1;
while (iLength >= 1)
{
iLength = SSL_read(psslSSL, lpszRead, RECV_SIZE - 1);
if (iLength < 0)
{
std::wcout << "Error SSL_read" << std::endl;
delete[] lpszRead;
return -1;
}
lpszRead[iLength] = TEXT('\0');
std::wcout << lpszRead;
}
delete[] lpszRead;
return 0;
}