Windows安装配置C/C++(VS2017)OpenSSL开发环境配置教程_C 语言_脚本之家

OpenSSL开发环境配置教程分享给大家,一种是通过源码进行编译的,还有一种是直接使用编译好的二进制安装包的,我是采用二进制安装包成功了,源码编译需要perl,nmake,dmake,不然会报不存在的错误,简单点就用别人编译好的了,附下载地址,特此记录,具体内容如下

【环境信息】

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.1b二进制安装包(已编译好的)

下载地址:https://slproweb.com/products/Win32OpenSSL.html

【提醒事项】

1。安装OpenSSL有两种方法,第一种为下载OpenSSL源代码并由自己编译后使用;第二种为下载已编译好的安装包,安装后直接可以使用。因第一种方法十分繁琐,需要安装Ruby进行编译,同时编译过程中可能产生各种各种的问题,因此在此采用第二种方法

2。每次新建工程之后都要配置一下OpenSSL开发环境

3。本人用的英文环境(英文操作系统+英文软件),因此操作界面语言可能会和中文环境有些不同,但并不影响功能

【具体步骤】

1。下载编译好的OpenSSL安装包(下载地址见上方),此处有Win32和Win64可选,这里的位数指的是你调用OpenSSL开发出来的软件的位数版本,而不是你计算机的位数。开发32位软件选择Win32,64位选择Win64,如果同时需要开发32位和64位的则下载两个。确定好位数后,下载Win32/64 OpenSSL v1.1.0f(版本号可能会因更新而有所不同),“Light版”为“轻量版”,即只包含了核心功能的版本,本文演示的为“非Light版”

2。下载完后打开安装,选择安装位置,64位和32位不要安装在同一个目录下

这里写图片描述

3。选择把dll复制到OpenSSL目录下(主要是为了以后好找,如果选择复制到Windows系统目录下,天知道复制到哪里去了。。。)

这里写图片描述

4。安装完成后,如果有捐款意向的可以捐款(别误会我不是在打广告。。。即使你捐款了我也拿不到一分钱),没有意向则取消掉钩,然后点“Finish”

这里写图片描述

5。新建一个解决方案,方便起见,这里新建一个控制台应用程序,这里我没有勾选“Create directory for solution”是因为这是一个单项目解决方案,所以我觉得没有必要多创建一层文件夹

这里写图片描述

6。勾选“Empty project”(这个是个人习惯,可能钩不钩并没有什么影响(“可能”这个词就很骚了。。。))

这里写图片描述

7。右击工程,选择“Properties”

这里写图片描述

8。点击左方的“VC++ Directories”,然后左上角选择“All Configurations”

这里写图片描述

9。右上角选择平台,这里演示x64的,如果是开发Win32软件就选Win32,如果要Win64和Win32兼顾,则分别配置一下(就是先选择Win32进行配置,然后再选择x64进行配置)

这里写图片描述

10。选中“Include Directories”,点击右边的下拉按钮,点击“Edit…”

这里写图片描述

11。点击右上方的“New Line”按钮,然后点击新行右边的“…”按钮

这里写图片描述

12。在弹出的目录选择界面中选择OpenSSL安装目录下的“include”文件夹,点击“Select Folder”,再点击“OK”完成添加

这里写图片描述

13。同样的方法,将安装目录下的“lib”文件夹添加到“Library Directories”中

这里写图片描述

14。将OpenSSL安装目录下bin文件夹中的“libcrypto-1_1-x64.dll”和“libssl-1_1-x64.dll”(名字后面的版本号可能因更新而不同)复制到工程目录下

这里写图片描述

这里写图片描述

15。将工程平台调整为自己需要的平台,这里演示x64平台

这里写图片描述

16。添加lib文件,这里有A、B两种方法,分别列出:

16-A:

在代码中添加

#pragma comment(lib,"libssl.lib")</p><p>#pragma comment(lib,"libcrypto.lib")

这里写图片描述

16-B:

在“Property Pages->Linker->Input->Additional Dependencies”添加libssl.lib和libcrypto.lib

这里写图片描述

这里写图片描述

【测试代码】

原文的测试代码有些复杂,主要是测试ssl,我这里贴一段测试各种hash函数的,比如sha1,sha256,sha512等的代码:

#include <iostream>
#include <string.h>
#include <openssl/sha.h>
#include <openssl/crypto.h>  // OPENSSL_cleanse  
 
using namespace std;
 
const char *orgStr = "BoEnGhnXNhe0YC+7loQD6Q==258EAFA5-E914-47DA-95CA-C5AB0DC85B11"; //待哈希的串
 
// 打印前, 有必要转换
void printHash(unsigned char *md, int len)
{
    int i = 0;
    for (i = 0; i < len; i++)
    {
        printf("%02x", md[i]);
    }
 
    printf("\n");
}
 
void myHash1()
{
    SHA_CTX c;
    unsigned char md[SHA_DIGEST_LENGTH];
    SHA1((unsigned char *)orgStr, strlen(orgStr), md);
    printHash(md, SHA_DIGEST_LENGTH);
 
    SHA1_Init(&c);
    SHA1_Update(&c, orgStr, strlen(orgStr));
    SHA1_Final(md, &c);
    OPENSSL_cleanse(&c, sizeof(c));
    printHash(md, SHA_DIGEST_LENGTH);
}
 
void myHash224()
{
    SHA256_CTX c;
    unsigned char md[SHA224_DIGEST_LENGTH];
    SHA224((unsigned char *)orgStr, strlen(orgStr), md);
    printHash(md, SHA224_DIGEST_LENGTH);
void myHash256()
{
    SHA256_CTX c;
    unsigned char md[SHA256_DIGEST_LENGTH];
    SHA256((unsigned char *)orgStr, strlen(orgStr), md);
    printHash(md, SHA256_DIGEST_LENGTH);
 
    SHA256_Init(&c);
    SHA256_Update(&c, orgStr, strlen(orgStr));
    SHA256_Final(md, &c);
    OPENSSL_cleanse(&c, sizeof(c));
    printHash(md, SHA256_DIGEST_LENGTH);
}
 
void myHash384()
{
    SHA512_CTX c;
    unsigned char md[SHA384_DIGEST_LENGTH];
    SHA384((unsigned char *)orgStr, strlen(orgStr), md);
    printHash(md, SHA384_DIGEST_LENGTH);
 
    SHA384_Init(&c);
    SHA384_Update(&c, orgStr, strlen(orgStr));
    SHA384_Final(md, &c);
    OPENSSL_cleanse(&c, sizeof(c));
    printHash(md, SHA384_DIGEST_LENGTH);
}
 
void myHash512()
{
    SHA512_CTX c;
    unsigned char md[SHA512_DIGEST_LENGTH];
    SHA512((unsigned char *)orgStr, strlen(orgStr), md);
    printHash(md, SHA512_DIGEST_LENGTH);
 
    SHA512_Init(&c);
    SHA512_Update(&c, orgStr, strlen(orgStr));
    SHA512_Final(md, &c);
    OPENSSL_cleanse(&c, sizeof(c));
    printHash(md, SHA512_DIGEST_LENGTH);
}
int main()
{
    myHash1();
    myHash224();
    myHash256();
    myHash384();
    myHash512();
 
    return 0;
}

原文代码:

1。这段代码是用C++和Windows API写的

2。这里添加lib文件用的方法A,如果想用方法B则注释或删掉对应的“#pragma comment…”即可

3。这段代码是用https://www.baidu.com来测试的,如果想要修改为其他测试地址,除了修改第25行的wstrHost外,还应修改第77行的strWrite

#include&lt;tchar.h&gt;</p><p>#include&lt;WinSock2.h&gt;</p><p>#include&lt;WS2tcpip.h&gt;</p><p>#include&lt;iostream&gt;</p><p>#include&lt;openssl\ssl.h&gt;</p><p>#pragma comment(lib,"ws2_32.lib")</p><p>#pragma comment(lib,"libssl.lib")</p><p>#pragma comment(lib,"libcrypto.lib")</p><p>CONST INT RECV_SIZE = 8192;</p><p>INT _tmain(INT argc, LPTSTR argv[])</p><p>{</p><p> //启动wsa</p><p> WSADATA wsadData;</p><p> WSAStartup(MAKEWORD(2, 2), &amp;wsadData);</p><p> //获取Host的IP地址等信息</p><p> ADDRINFOT aiHints;</p><p> ZeroMemory(&amp;aiHints, sizeof(ADDRINFOT));</p><p> aiHints.ai_family = AF_INET;</p><p> aiHints.ai_flags = AI_PASSIVE;</p><p> aiHints.ai_protocol = 0;</p><p> aiHints.ai_socktype = SOCK_STREAM;</p><p> std::wstring wstrHost = TEXT("www.baidu.com");</p><p> PADDRINFOT paiResult;</p><p> GetAddrInfo(wstrHost.c_str(), NULL, &amp;aiHints, &amp;paiResult);</p><p> //创建套接字</p><p> SOCKET sSocket = socket(AF_INET, SOCK_STREAM, 0);</p><p> if (sSocket == SOCKET_ERROR)</p><p> {</p><p>  std::wcout &lt;&lt; "Error socket" &lt;&lt; std::endl;</p><p>  return -1;</p><p> }</p><p> //连接Host</p><p> SOCKADDR_IN sinHost;</p><p> sinHost.sin_addr = ((LPSOCKADDR_IN)paiResult-&gt;ai_addr)-&gt;sin_addr;</p><p> sinHost.sin_family = AF_INET;</p><p> sinHost.sin_port = htons(443);</p><p> if (connect(sSocket, (LPSOCKADDR)&amp;sinHost, sizeof(SOCKADDR_IN)) == SOCKET_ERROR)</p><p> {</p><p>  std::wcout &lt;&lt; "Error connect" &lt;&lt; std::endl;</p><p>  return -1;</p><p> }</p><p> //初始化OpenSSL库</p><p> //(虽然不知道为什么,但是不加这三行似乎并不会导致什么问题,在不加这3行的情况下测试了几个网站并没有发现任何问题喵)</p><p> SSL_library_init();</p><p> SSLeay_add_ssl_algorithms();</p><p> SSL_load_error_strings();</p><p> //创建SSL会话环境等</p><p> SSL_CTX *pctxSSL = SSL_CTX_new(TLSv1_2_client_method());</p><p> if (pctxSSL == NULL)</p><p> {</p><p>  std::wcout &lt;&lt; "Error SSL_CTX_new" &lt;&lt; std::endl;</p><p>  return -1;</p><p> }</p><p> SSL *psslSSL = SSL_new(pctxSSL);</p><p> if (psslSSL == NULL)</p><p> {</p><p>  std::wcout &lt;&lt; "Error SSL_new" &lt;&lt; std::endl;</p><p>  return -1;</p><p> }</p><p> SSL_set_fd(psslSSL, sSocket);</p><p> INT iErrorConnect = SSL_connect(psslSSL);</p><p> if (iErrorConnect &lt; 0)</p><p> {</p><p>  std::wcout &lt;&lt; "Error SSL_connect, iErrorConnect=" &lt;&lt; iErrorConnect &lt;&lt; std::endl;</p><p>  return -1;</p><p> }</p><p> std::wcout &lt;&lt; "SSL connection using " &lt;&lt; SSL_get_cipher(psslSSL) &lt;&lt; std::endl;</p><p> //发包</p><p> std::string strWrite =</p><p>  "GET https://www.baidu.com/ HTTP/1.1\r\n"</p><p>  "Host: www.baidu.com\r\n"</p><p>  "Connection: close\r\n\r\n";</p><p> INT iErrorWrite = SSL_write(psslSSL, strWrite.c_str(), strWrite.length()) &lt; 0;</p><p> if (iErrorWrite &lt; 0)</p><p> {</p><p>  std::wcout &lt;&lt; "Error SSL_write" &lt;&lt; std::endl;</p><p>  return -1;</p><p> }</p><p> //收包并输出</p><p> //这里接受的是char形式的,所以中文会乱码</p><p> //如果要正常显示中文,需要再转换为wchar_t或std::wstring</p><p> LPSTR lpszRead = new CHAR[RECV_SIZE];</p><p> INT iLength=1;</p><p> while (iLength &gt;= 1)</p><p> {</p><p>  iLength = SSL_read(psslSSL, lpszRead, RECV_SIZE - 1);</p><p>  if (iLength &lt; 0)</p><p>  {</p><p>   std::wcout &lt;&lt; "Error SSL_read" &lt;&lt; std::endl;</p><p>   delete[] lpszRead;</p><p>   return -1;</p><p>  }</p><p>  lpszRead[iLength] = TEXT('\0');</p><p>  std::wcout &lt;&lt; lpszRead;</p><p> }</p><p> delete[] lpszRead;</p><p> return 0;</p><p>}</p><p>

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

来源URL:https://www.jb51.net/article/119025.htm