在进行C++时,抉择什么样的编译器显得十分的重要。与其它的语言不同,C++的程序最终由头文件(.h)以及库文件(.dll或.so)组件。
不同的利用(库)帮助工作时,同样须要对文件及库文件的反对,而不同的平台上的库文件是不统一的,所以在跨平台的开发中,咱们更喜爱应用cmake来进行编译链接。
所以cmake并不是包管理工具,他并不可能解决包的依赖问题,而仅仅解决的跨平台的问题。
编译器
将源代码变成可执行的利用须要一个叫做编译器的货色,如果在利用中应用了曾经成型的第三方库,或是本人的利用较大,须要多个模块配合的话。还须要将这些模仿连接起来,最终造成一个能够应用的应用程序。
就这像在一个规范的Spring boot我的项目在构建时,可能构建出一个蕴含本人和其它第三方库,可被间接运行的xxx.jar
和一个只蕴含本人的可被用做第三方库的xxx.jar.original
文件。
而将源代码变成可执行的字节码的过程叫做编译,将源代码本人生成的库与其它库连在一起,最初造成可执行应用程序的过程则叫做链接。
实现上述性能的工具则称为编译器。
在java开发中,咱们须要针对不同的操作系统下载不同的jdk。这个jdk中有一部分的作用便是java的编译器。在C++开发中,macos中曾经默认装置了一款名为clang的编译器,所以咱们并不需要独自的装置它。
须要叫各个系统中c++的编译器不同,比方macos中默认装置的是clang。除此以外,编译器还有注明的GNU(gcc、g++)。而不同的编译器均领有本人的特点,咱们在开发利用时,也会将这些编译器的特点退出到咱们的源代码中。而后编译器在依据源代码来编译为字节文件。在有些个性只有特定的编译器才反对,这也是为什么有的利用只能应用特定编译器编译能力工作的起因。
依赖治理
java中有比拟驰名的maven
、gradle
包管理器,它可能主动的为咱们下载依赖,接着调用指定的jdk进行编译,可能实现java利用的打包工作。
不太侥幸的是c++利用中并没有这么弱小的包管理工具。但一个多模块有依赖的我的项目,实现的思维依然是雷同的。
如同npm
下治理的包在package.json
中申明了须要的包及版本号,而后应用npm install
的命令来装置。C++的利用则能够在CMakeLists.txt
来申明版本号。而后应用其它的形式来装置。
加载依赖
后面讲了cmake是一款跨平台的编译器,而它对应的配置文件则是CMakeLists.txt
,在该文件中阐明了对其它第三方包的依赖。
find_package()
CMakeLists.txt
申明依赖的形式大体分为两种,第一种是应用find_package
。
比方:find_package(JsonCpp REQUIRED)
。此时CMake会先查找其外部包,如果没有在外部包中找到,则会在当前目录下查找文件FindJsonCpp.cmake
文件。如果找到了FindJsonCpp.cmake
文件,则会按该文件的设置加载包;如果没有找到,因为第二个参数设置的值为REQUIRED
,则会抛出一个异样。
所以如果某些第三方包提供了FindXxxx.cmake
文件的话,能够轻松的应用find_package(name, REQUIRED)
函数来加载第三方包。
个别状况下,如胜利的找到第三方包将会定义如下变量:
# 找到包时,该值为trueXxxx_FOUND# 该包头文件所在位置Xxxx_INCLUDE_DIRS 或 Xxxx_INCLUDES# 该包库文件所在位置Xxxx_LIBRARY 或 Xxxx_LIBRARIES 或 Xxxx_LIBS# 能够应用message(${Xxxx_LIBRARIES})来打印变量的值
pkg_check_modules()
第二种形式是借助于PkgConfig
:
find_package(PkgConfig REQUIRED)
当零碎未装置PkgConfig
则会提到一个未找到PkgConfig
的谬误,此时则须要在零碎中装置PkgConfig
。
有了PkgConfig
后便能够应用其提供的pkg_check_modules
包检测办法来获取第三方包的信息了:
pkg_check_modules(JSONCPP jsoncpp)
其中JSONCPP
为别名能够轻易起,jsoncpp
为模块名。
此时当存在jsoncpp
时,则零碎变量JSONCPP__FOUND
的值为true
,否则为false
。如果以后我的项目必须依赖于某个第三方包,则能够退出REQUIRED
,比方:pkg_check_modules(XXX xxx REQUIRED)
此时当不存在xxx
时将会产生一个异样。
加载
无论是通过find_package()
还是通过pkg_check_modules
来获取第三方依赖包,其本质上都是在:查看是否存在某个包,如果存在则返回该包的信息,如果不存在,则将零碎变量Xxxx_FOUND设置为false。除此以外,并没有其它作用。
也就是说上述两个办法并没有加载对其第三方的依赖。这所以要应用命令来查找某个包的信息(包的库文件、头文件地位),也是思考了跨平台的起因。在跨平台的状况下,咱们不能应用诸如windows中的c:\xxx\xxx
或xunix中的~/xxx/xxx
任一模式。所以在加载某个依赖前,须要通过上述办法来动静获取包的信息,如果以后应用的是windows零碎,则能够是c:\xxx\xx
的模式,如果应用的是xunix
零碎,则可能是/usr/local/xxx
的模式。
加载第三方库须要将头文件使在地位应用include_directories
蕴含进来,并将库应用target_link_libraries
链接进来。
include_directories(project-name ${Xxxxx_INCLUDE_DIRS})target_link_libraries(project-name ${Xxxxx_LIBRARIES})
包管理器
Microsoft提供了vcpkg来装置c++利用的依赖。应用shell装置形式如下:
# clone仓库git clone https://github.com/Microsoft/vcpkg.git --depth=1# 进入仓库cd vcpkg# 运行装置脚本./bootstrap-vcpkg.sh# 设置为全局装置,当前再装置其它包的时候,都是全局的./vcpkg integrate install
执行齐全局装置后,将会失去一个提醒:
panjie@panjies-iMac vcpkg % ./vcpkg integrate installApplied user-wide integration for this vcpkg root.CMake projects should use: "-DCMAKE_TOOLCHAIN_FILE=/Users/panjie/github/Microsoft/vcpkg/scripts/buildsystems/vcpkg.cmake"
它在说如果咱们再应用CMake工具的话,则须要在CMake命令后追加上述参数。如果咱们应用了Clion编辑器,则须要将其增加到CMake options中:
此时Clion中的Cmake则会在执行时主动退出上述参数。自此以后Clion便能够和cmake及vcpkg包管理器欢快的工作在一起了。
装置特定的包
比方咱们装置一个用于json序列化的jsoncpp
,则能够在vcpkg
中执行:./vcpkg install jsoncpp
:
panjie@panjies-iMac vcpkg % ./vcpkg install jsoncpp Computing installation plan......The package jsoncpp:x64-osx provides CMake targets: find_package(jsoncpp CONFIG REQUIRED) target_link_libraries(main PRIVATE jsoncpp_object jsoncpp_static)
最初依照提醒将上述代码增加到本人我的项目中的CMakeLists.txt
中,而后便能够欢快的在我的项目中应用第三方jsoncpp包了:
add_executable(yz-main yz-main.cpp) find_package(jsoncpp CONFIG REQUIRED) target_link_libraries(yz-main PRIVATE jsoncpp_object jsoncpp_static)
尔后便能够在以后工程中自在的应用jsoncpp
来实现json的序列化与反序列化工作了。
总结
最初应用java与c++我的项目做比照总结:
- 两者都属于编译型语言,都须要先编译。
- java源文件应用jdk来编译,不同的操作系统须要装置不同的JDK安装包;C++须要应用C++编译器进行编译,不同的操作系统须要下载不同的编译器。
- java的编译器有多个版本,比方oracle自家的jdk,或是第三方的openjdk;C++的编译器也有多个版本,比方clang、GNU。在理论的开发过程中,须要按本人的需要进行下载、装置配置。
- java中有对立的包治理、装置工具
maven
或gradle
;C++有对立的包管理工具vcpkg
,有对立的装置工具cmake
。 vcpkg
想与cmake
欢快的工作,须要配置一些参数。该参数不同的计算机会有不同,须要按装置vcpkg
状况进行配置。