因为工作的原因,加上自己的一点点兴趣爱好,今天花了一整天的时间去探索和学习 libcurl。
libcurl 的文档是非常齐全的,并且在获取到了源代码之后,里面的代码、文档、工具等等都是非常完善的。可是对于新人来说,想要从这么复杂多样的文档中摸索出来适合自己入门的方向,实在是有些困难。这篇博客,就是想要给同样深陷于 libcurl 复杂的文档中不能自拔的程序员同僚们一条我自己走出来的道路。
在这篇博客之前,我在 Ubuntu 18.04 、RedHat 7.2 以及 Windows 10 环境中成功编译了 libcurl 库,并且使用编译出来的库成功运行了同一个实例代码。一方面为了记录中间我踩过的坑,另一方面也是为了方便广大网友,我尽力将这期间我的感悟与收获写在这篇博客里,希望能够对大家有所帮助。
在编译和使用 libcurl 库之前,有个似乎很容易被老手忽略的问题一定要说一下,那就是:
cURL、libcurl 还有 curl,他们究竟是什么?
也就是说,我们需要去了解下 libcurl 及其相关的概念。
可能对于新人来说,就连 cURL、libcurl 和 curl 的概念都是分不清楚的。这不怪我们,确实关于这一点,官方网站都没有说的很清楚,但是在源代码中的 FAQ 文档中却说的非常明白。
What is cURL?
cURL is the name of the project. The name is a play on ‘Client for URLs’, originally with URL spelled in uppercase to make it obvious it deals with URLs. The fact ti can also be pronounced ‘see URL’ alse helped, it works as an abbreviation for “Client URL Request Library” or why not the recursive version: “Curl URL Request Library”.
简而言之,cURL 是一个项目的名称。是 Client for URLs、see URL、Client URL Request Library 或者 Curl URL Request Library 的缩写,也就是一个客户端 URL 请求库的项目。
那么什么是 libcurl 呢?
The cURL project produces two products:
libcurl
A free and easy-to-use client-side URL transfer library.
…
curl
A command line tool for getting or sending files using URL syntax.
上面这段话很清晰的表现出了 cURL 与 libcurl 以及 curl 的关系,也就是说:
cURL 这个项目包含了 libcurl 和 curl 两个产品。
其中,libcurl 是一个客户端的 URL 支持库;而 curl 就是一个使用了 libcurl 库写出来的命令行工具,其可以使用 URL 标识来请求或者发送文件。
也就是说,如果我们想要编写代码来控制有关网络的行为的话,我们就需要使用到 libcurl 库而不是 curl 命令行工具;而如果我们想要直接调用 curl 命令行工具来完成一些操作,比如将其嵌入到脚本代码中去,那么这个时候,我们才会用到 curl 命令行工具。
其中 cURL、libcurl 以及 curl 三者之间的关系如下图所示(我亲手画的,轻喷 T_T):
另外有趣的是,我们可以把 cURL 发音发作 see URL;可以把 curl 发作以字母 k 开头的初始音,总体类似于 girl 或者 earl 的发音,甚至官方文档 FAQ 中还给出了 curl 的官方发音:
curl 的发音录音
现在搞清楚了一些必要的概念,以及一些有趣的题外话,现在让我们来看看,我们要在 Unix 环境和 Windows 环境下编译和使用 libcurl,我们需要哪些东西,以及我们能在哪些地方获取到这些东西。
我们想要了解 libcurl 这个库,最直接的资源获取来源当然是官方网站:
curl 官方网站
可能对于新手来说,一点开就会觉得有些迷茫,因为可能分不清楚 curl 和 libcurl 的区别。这也是为什么我一定要在介绍资料来源之前介绍 cURL、libcurl 和 curl 三者的区别的原因。相信在我上一节花了那么大篇幅来介绍这三者的区别之后,你应该不会那么迷茫了。
对于我们开发者来说,了解 libcurl 应该是最重要的。因为我们是想要使用 libcurl 库来编写代码的,而不是来学习 curl 的使用方法的。
从我上述图中画了红色框框的地方点进去,就是我们要仔细研读的重点了,也就是 libcurl 的官方文档:
libcurl 官方网站
这个网站中的信息量非常大,包括 Tutorial 学习教程、Examples 示例代码、Download 下载地址等等信息。通过自己的尝试点击浏览可以对 libcurl 官方网站有一些初步的了解。
其中,建议大家一定要看的,就是 Tutorial 标签下的教程文档,这份教程文档虽然跳跃度很大,并且不是很适合初学者,但是对于想要一开始就宏观上了解认识 libcurl 的人来说,这是唯一的一份具有官方背书的文档了(友情提示,网上有这份文档的的中文翻译,英语不好的同学可以搜索网上资料看看)。
libcurl programming tutorial
读完了上述的文档,你可能一时间非常迷茫,但是对于 libcurl 的基础的几个概念几个函数还是有了一定的了解。此时的你,需要的可能就是如何安装编译 libcurl,并且运行几个示例代码亲手调试一番来增强自己的自信心了。
那么问题来了,我们该如何下载 libcurl 的源代码呢?
当我们从 libcurl 网址中点开 Download Page 的时候,可能一下子又懵了,为什么呢?因为你看着这么复杂的页面,一时间不知道该下载哪个文件:
这里,通过我的尝试发现,点击我上图中花了红色框框的文件是最适合这里的需求的。
首先,在下载列表中,有很多 Packages 基于不同系统环境的包,这些包下载下来都只是一个支持的库文件而已,只是用来“用”的,不是用来编译和学习的。
另外,通过 Download Wizard 也就是下载模板下载下来的文件,也都是基于上述的那个方式下载的内容,也都是缺斤少两的产品,对于本篇博客想要去探索去编译去运行自己编写的代码的需求不甚相符。
所以,通过多次尝试的我最后发现,下载 Source Archives 也就是源代码的版本,是最好的。原因嘛,当你下载下来之后,解压到本地看看里面的内容你就知道了,因为它,实在是太完善了:
简单的说几个:
1. docs 文件夹
这里面有丰富的说明文档以及 libcurl 的运行示例代码。
有关 cURL、libcurl 以及 curl 的概念的定义,就是在这个文件夹下的 FAQ 文件里面。后面将要讲述的在 Unix 下编译使用 libcurl 库的内容,也是来源于这个文件夹下的 INSTALL.md 文件。除此之外,这个文件夹下还有很多说明文档,有待大家去探索去发现去思考去使用。
这个文件夹下的 examples 文件夹下,有着丰富的示例代码,其中的 https.c 就是本篇博客的测试运行代码。
2. winbuild 文件夹
这个文件夹介绍了如何使用 Visual Studio 编译 libcurl 的方法。基于 Windows 环境的编译与使用就是参考的这个文件夹下的 BUILD.WINDOWS.txt 文件中的内容。
并且这个文件夹下提供了编译的配置信息文件,大大方便了我们在 Windows 下使用 Visual Studio 编译 libcurl 的工作。
3. configure
这个脚本文件用于在 Unix 下配置 libcurl 的安装信息,用来之后安装 libcurl 环境使用。
等等等等,curl 的源代码文件中,包含了很多很多东西。有很多你可能接触不到,我们可以在学习中在使用中慢慢去发掘去使用。
让我们步入正题吧,libcurl 在 Unix 环境下怎么编译与使用呢?
这个问题在 curl-7.61.0\docs 下的 INSTALL.md 文件中讲述的非常清晰:
A normal Unix installation is made in three or four steps (after you’ve unpacked the source archive):
./configure
make
make test (optional)
make install
也就是说,在类 Unix 环境下,我们都可以在源代码文件目录下使用以下四句指令完成 libcurl 库的安装与编译:
$ ./configure
$ make
$ make test (optional)
$ make install
其中第三步,也就是测试那步是非常耗时间的,为了节约时间可以省略。
另外第四步,可能会涉及到权限问题,如果出现这个问题,需要切换到 root 权限安装。
简单总结下步骤吧:
1. 使用 WinSCP 以及类似的工具,将下载下来的源代码文件放置到指定目录下,比如我现在将下载下来的 curl-7.61.0.zip 文件放到了 RedHat 7.2 环境中的 /home/wangying/libcurl
文件夹下
2. 解压上一步放置的源代码文件
$ unzip curl-7.61.0.zip
3. 解压完成后进入 curl-7.61.0 文件夹下,运行上述所说的 4 条指令:
$ ./configure
$ make
$ make test (optional)
$ make install
建议大家省去第三步,因为实在太耗费时间了(T_T)。
4. 执行完了之后,现在直接运行 curl 还是会显示系统默认自带的版本。我们需要进入到默认安装的目录下 /usr/local/bin
中使用下列命令运行:
$ ./curl --version
$ ./curl-config --version
上述两行代码会输出 curl 以及 curl-config 两个工具的版本信息,与你下载的版本一致即为安装成功。根据我的试验经验,一旦你运行了上述两行代码,系统自动会记录下来这两个工具的路径,不过为了确保系统找得到这两个工具,你可以单独设置下环境变量。
5. libcurl 库的相关头文件与库文件所在地方,可以通过下列命令查看(这会在下一节中的使用中提到)
$ curl-config --cflags
$ curl-config --libs
其中 –cflags 输出的我们编写代码时需要包含的头文件路径, –libs 输出的是我们编写代码时需要包含的库文件路径,我们只需要将上述两行简单的添加到编译指令中即可运行我们的示例代码。
让我们点开源代码 docs 目录下的 examples 文件夹,其中的 README 文件详细介绍了 libcurl 示例代码的使用方式:
Most examples should build fine using a command line like this:
$curl-config --cc --cflags --libs
-o example example.c
Some compilers don’t like having the arguments in this order but instead want you do reorganize them like:
$curl-config --cc
-o example example.ccurl-config --cflags --libs
也就是说,只要我们在上一步中成功安装了 curl 以及 curl-config 工具,在这一步中,我们只需要简单的运行这行指令即可自动的指定代码的包含头文件以及库文件信息:
$ `curl-config --cc` -o example example.c `curl-config --cflags --libs`
让我们来尝试下 examples 中的 https.c 文件的编译(因为 https.c 文件可以在源代码中看到,这里就不再详细展示文件内容):
$ `curl-config --cc` -o https https.c `curl-config --cflags --libs`
可见,https.c 的运行是非常成功的,成功返回了获取的 html 信息。
同样的示例我在 Ubuntu 18.04 和 RedHat 7.2 上都运行通过,相信在类 Unix 上应该都是没有问题的。
因为我以前是搞 Windows 开发的,当然要探索一下,libcurl 在 Windows 平台下的编译与使用啦(生命在于不停的折腾!)。
libcurl 在 Windows 环境下的编译,在源代码文件夹下的 docs 文件夹下的 INSTALL 文件中有所提及,但是讲述的稍微有些晦涩难懂。欣喜的是源代码根目录下直接提供了一个 winbuilds 文件夹方便我们完成在 Windows 环境下的编译工作。
这里,让我们看看 winbuild 文件夹下的 BUILD.WINDOWS.txt 文件的内容(这个文件中的内容非常详尽,这里不再粘贴出来)。
我简单总结下步骤:
1. 运行 Developer Command Prompt for VS 工具
2. 进入到源代码文件夹下的 winbuild 目录下
3. 运行下列指令:
nmake /f Makefile.vc mode=<static or dll> <options>
其中的 mode= 后面填写 static 是编译静态库, dll 是动态库;另外 options 的填写说明如下:
where is one or many of:
VC=<6,7,8,9,10,11,12,14,15> - VC versions
WITH_DEVEL= - Paths for the development files (SSL, zlib, etc.)
Defaults to sibbling directory deps: ../deps
Libraries can be fetched at http://windows.php.net/downloads/php-sdk/deps/
Uncompress them into the deps folder.
WITH_SSL=or static> - Enable OpenSSL support, DLL or static
WITH_NGHTTP2=or static> - Enable HTTP/2 support, DLL or static
WITH_MBEDTLS=or static> - Enable mbedTLS support, DLL or static
WITH_CARES=or static> - Enable c-ares support, DLL or static
WITH_ZLIB=or static> - Enable zlib support, DLL or static
WITH_SSH2=or static> - Enable libSSH2 support, DLL or static
ENABLE_SSPI=or no> - Enable SSPI support, defaults to yes
ENABLE_IPV6=or no> - Enable IPv6, defaults to yes
ENABLE_IDN=or no> - Enable use of Windows IDN APIs, defaults to yes
Requires Windows Vista or later
ENABLE_WINSSL=or no> - Enable native Windows SSL support, defaults to yes
GEN_PDB=or no> - Generate Program Database (debug symbols for release build)
DEBUG=or no> - Debug builds
MACHINE=or x64> - Target architecture (default is x86)
CARES_PATH=to cares> - Custom path for c-ares
MBEDTLS_PATH=to mbedTLS> - Custom path for mbedTLS
NGHTTP2_PATH=to HTTP/2> - Custom path for nghttp2
SSH2_PATH=to libSSH2> - Custom path for libSSH2
SSL_PATH=to OpenSSL> - Custom path for OpenSSL
ZLIB_PATH=to zlib> - Custom path for zlib
大家可根据需要填写,这里我填写的命令是:
nmake /f Makefile.vc mode=static
4. 待编译完成,进入源代码文件夹下,其中多出来了一个 builds 文件夹,其中名字长度最短的那个点击进去,可以看到 bin、include 和 lib 子文件夹,其中 bin 中就是编译出来的 curl 命令行工具,include 就是我们在编写代码中需要包含的头文件,lib 就是我们在编写代码中需要包含的静态库文件(不过仍然需要 CRT 静态库的另外链接)
至此,使用 Visual Studio 2017 编译工具静态编译 libcurl 库的工作就算是完成了。这里值得注意的是,Windows 平台下的编译选项有很多,我们可以根据需要配置自己想要的 libcurl 库。比如静态动态,比如使用 VC 12 13 14 15 版本等等等等。
在这篇博客里,主要展示了使用静态库的方式。接下来,我们来看看如何使用刚刚编译出来的这些文件进行示例代码的运行使用。
libcurl 在 Windows 环境下的使用,参考了另外一篇博客:
Visual Studio(VS2017)编译并配置C/C++-libcurl开发环境
这篇博客讲述的非常清楚,不仅仅是要把我们刚才生成的 include 头文件和 lib 静态库文件包含进去,还需要包含一些另外的静态库文件,以及设置一个预处理字段才能正常运行我们的示例代码。
这里,我简单总结下步骤:
1. 使用 Visual Studio 2017 新建一个 Visual C++ 的空项目,并将示例代码 https.c 拷贝到项目中去
2. 为了使得我们的 https.c 代码运行后控制台不会一闪而过,我们需要修改 https.c 的代码,不多,只需要添加两行即可:
// 添加在 include 的地方
#include
// 添加在 return 语句之前
system("pause");
这样,就可以使得控制台窗口运行完代码逻辑之后暂停,不至于关闭掉窗口了。
3. 将上一节中生成的 include 文件夹以及 lib 文件夹拷贝到项目中去
4. 配置 VC++ 目录:需要让代码能够找到需要包含的头文件和静态库文件
包含目录添加:
// $(SolutionDir) 是当前的解决方案目录
$(SolutionDir)include
库目录添加:
// $(SolutionDir) 是当前的解决方案目录
$(SolutionDir)lib
5. 配置需要链接的静态库文件
链接器->输入->附加依赖项
libcurl_a.lib
Ws2_32.lib
Wldap32.lib
winmm.lib
Crypt32.lib
Normaliz.lib
6. 修改预处理器定义:这块的修改来源于源文件文件夹下的 docs 文件夹下的 FAQ 文件中的第 5.7 点:
…
When building an application that uses the static libcurl library, you must
add -DCURL_STATICLIB to your CFLAGS. Otherwise the linker will look for
dynamic import symbols. If you’re using Visual Studio, you need to instead
add CURL_STATICLIB in the “Preprocessor Definitions” section.
…
也就是说,如果我们要静态链接 libcurl 库,我们需要添加一个预处理定义标记为 CURL_STATICLIB,否则链接器将会寻找该库的动态版本。具体的修改方式如下:
C/C++ -> 预处理器 -> 预处理器定义
// 添加
CURL_STATICLIB;
7. 统一代码生成方式,如果是 Debug 就是多线程调试 /MDd,如果是 Release 就是多线程 /MD
C/C++ -> 代码生成 -> 运行库
至此,完结撒花,^_^
学习 libcurl 库的最好方法,就是先看官方网站,然后顺着官方网站的介绍下载源代码,最后通过编译源代码运行示例代码去掌握 libcurl 库的使用方法。
可以说 libcurl 库的文档真的是非常的齐全,只要你愿意去啃英文,愿意去尝试去搜索去钻研,你就会得到自己想要探索的东西。
本篇博客记录了我从一开始什么也不懂的小白,到最后成功在 Ubuntu 18.04、RedHat 7.2 以及 Windows 10 上编译 libcurl 库和使用其运行示例代码的过程。过程中遇到了很多难点,也都在官方网站的指引下、源代码中的文档的指引下以及其他人的博客的指引下成功的走了出来。
这篇博客真的写了好长好长时间,也是我第一次写博客写到凌晨一点多,尽管很疲惫,也难掩我的兴奋之情,只有写,才能让我感到存在的意义。写这么多,只希望能够给大家带来一点绵薄的帮助就好 ^_^
libcurl 库的学习还要继续 ~~~
To be Stronger:)