CMake的使用
CMake是一个跨平台的Makefile生成工具,可以根据特定的规则生成相应的Makefile文件,并对C/C++源代码进行编译和管理。 有两篇博客介绍CMake的使用,比较通俗易懂,链接地址是:
- Cmake 详解
- CMake 入门实战
- CMake的官方下载地址为:https://cmake.org/download/
- 官方文档地址为:CMake 3.16 Documentation
- 官方的CMake指南地址为:CMake Tutorial
一、CMake中添加对C++11的支持
1、在对应的CMakeLists.txt文件中加入以下语句:
add_definitions(-std=c++11)
或者
if(CMAKE_COMPILER_IS_GNUCXX)
set(CMAKE_CXX_FLAGS "-std=c++11 -g ${CMAKE_CXX_FLAGS}")
endif(CMAKE_COMPILER_IS_GNUCXX)
2、延伸 如何写cmake使其包含c++11特性 (-std=c++11如何写进cmakeList.txt)
使用的g++版本和cmake版本分别是g++ 4.8.2和cmake 2.8 之前写cmkae编译带有c++11特性的代码有这么一句:
set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -std=c++11")
但是总会出现cc1plus: error: unrecognized command line option “-std=c++11” 报错。 所以set(QMAKE_CXXFLAGS “-std=c++11”) 类似的写法肯定不行。 后来发现是std=c++11 这种写法老版本不支持。 ok 直接测试新写法 CMakeLists.txt文件如下所示:
#CMakeLists.txt
project(test)
cmake_minimum_required(VERSION 2.8)
aux_source_directory(. SRC_LIST)
add_executable(${PROJECT_NAME} ${SRC_LIST}${PROJECT_NAME}.cpp)
include(CheckCXXCompilerFlag)
CHECK_CXX_COMPILER_FLAG("-std=c++11" COMPILER_SUPPORTS_CXX11)
CHECK_CXX_COMPILER_FLAG("-std=c++0x" COMPILER_SUPPORTS_CXX0X)
if(COMPILER_SUPPORTS_CXX11)
set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -std=c++11")
elseif(COMPILER_SUPPORTS_CXX0X)
set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -std=c++0x")
else()
message(STATUS "The compiler ${CMAKE_CXX_COMPILER} has no C++11 support. Please use a different C++ compiler.")
endif()
测试c++11代码如下:
//test.cc
#include <iostream>
#include<vector>
using namespace std;
int main()
{
const std::vector<int>v(1);
auto a = v[0];//a为int类型
cout <<"a : "<< a <<endl;
decltype(v[0]) b = 0;//b为const int&类型,即std::vector<int>::operator[](size_type)const的返回类型
auto c = 0;//c为int类型
auto d = c;//d为int类型
decltype(c) e;//e为int类型,c实体的类型
decltype((c)) f = e;//f为int&类型,因为(c)是左值
decltype(0) g;//g为int类型,因为0是右值
return 0;
}
examples_CMake项目
github上面有一个韩国人jacking75写的简单的cmake使用示例, examples_CMake项目地址是:https://github.com/jacking75/examples_CMake
CMake例子
范例介绍
示例代码在CMake_example目录中。
01 helloworld 一个简单文件中的-C ++代码
- main.cpp
#include <iostream>
int main()
{
auto name = "jacking";
std::cout << "hello world: " << name << std::endl;
return 0;
}
- CMakeLists.txt
cmake_minimum_required(VERSION 2.8)
add_definitions(-std=c++11)
add_executable(Main main.cpp)
02 helloworld-设置编译器选项。 -Wall,C ++ 14
- main.cpp
#include <iostream>
int main()
{
auto name = "jacking";
std::cout << "hello world: " << name << std::endl;
return 0;
}
- CMakeLists.txt
cmake_minimum_required(VERSION 2.8)
add_definitions("-Wall -std=c++14")
add_executable(Main main.cpp)
03 helloworld-如果您有除主代码文件以外的其他代码文件
- main.cpp
#include "test.h"
int main()
{
TEST test;
test.Print();
return 0;
}
- test.h
class TEST
{
public:
void Print();
};
- test.cpp
#include "test.h"
#include <iostream>
void TEST::Print()
{
std::cout << "Test::Print" << std::endl;
}
- CMakeLists.txt
cmake_minimum_required(VERSION 2.8)
add_executable(Main
main.cpp
test.cpp
)
04 helloworld-如果mai.cpp以外的文件位于其他目录中
源代码04_helloworld目录结构如下:
[root@ltcos01 04_helloworld]$ tree -L 2
.
├── CMakeLists.txt
├── main.cpp
├── test01
│ ├── test01.cpp
│ └── test01.h
└── test02
├── test02.cpp
└── test02.h
2 directories, 6 files
- main.cpp
#include "test01/test01.h"
#include "test02/test02.h"
int main()
{
TEST01 test01;
test01.Print();
TEST02 test02;
test02.Print();
return 0;
}
test01目录下 有test01.h和test01.cpp这两个文件
- test01/test01.h
class TEST01
{
public:
void Print();
};
- test01/test01.cpp
#include "test01.h"
#include <iostream>
void TEST01::Print()
{
std::cout << "Test01::Print" << std::endl;
}
test02目录下有test02.h和test02.cpp这两个文件
- test02/test02.h
class TEST02
{
public:
void Print();
};
- test02/test02.cpp
#include "test02.h"
#include <iostream>
void TEST02::Print()
{
std::cout << "Test02::Print" << std::endl;
}
- CMakeLists.txt
cmake_minimum_required(VERSION 2.8)
add_executable(Main
main.cpp
test01/test01.cpp
test02/test02.cpp
)
05 helloworld-reference 创建静态文件后
05_helloworld源代码目录树结构如下所示:
[root@ltcos01 05_helloworld]$ tree -L 2
.
├── CMakeLists.txt
├── main.cpp
├── test01
│ ├── CMakeLists.txt
│ ├── test01.cpp
│ └── test01.h
└── test02
├── CMakeLists.txt
├── test02.cpp
└── test02.h
2 directories, 8 files
- main.cpp
#include "test01/test01.h"
#include "test02/test02.h"
int main()
{
TEST01 test01;
test01.Print();
TEST02 test02;
test02.Print();
return 0;
}
- CMakeLists.txt
cmake_minimum_required(VERSION 2.8)
add_subdirectory(test01)
add_subdirectory(test02)
add_executable(Main main.cpp)
target_link_libraries(Main Test01 Test02)
test01目录下有test01.h和test01.cpp以及相应的CMakeLists.txt文件
- test01/test01.h
class TEST01
{
public:
void Print();
};
- test01/test01.cpp
#include "test01.h"
#include <iostream>
void TEST01::Print()
{
std::cout << "Test01::Print" << std::endl;
}
- test01/CMakeLists.txt
cmake_minimum_required(VERSION 2.8)
add_library(Test01 STATIC
test01.cpp
)
上面的test01目录下的CMakeLists.txt的add_library(Test01 STATIC test01.cpp)指令会生成相应的静态库文件libTest01.a
test02目录下和test01目录结构一样,也有test02.h和test02.cpp以及相应的CMakeLists.txt文件
- test01/test02.h
class TEST02
{
public:
void Print();
};
- test02/test02.cpp
#include "test02.h"
#include <iostream>
void TEST02::Print()
{
std::cout << "Test02::Print" << std::endl;
}
- test02/CMakeLists.txt
cmake_minimum_required(VERSION 2.8)
add_library(Test02 STATIC
test02.cpp
)
同样的,在上面的test02目录下执行cmake命令会生成相应的静态库文件libTest02.a。具体操作过程如下:新建一个build目录,然后进入到build目录下执行cmake …运行上一级目录即test02下的CMakeLists.txt文件,操作如下:
[root@ltcos01 test02]$ ls
build CMakeLists.txt test02.cpp test02.h
[root@ltcos01 test02]$ cd build/
[root@ltcos01 build]$ ls
[root@ltcos01 build]$ cmake ..
-- The C compiler identification is GNU 4.8.5
-- The CXX compiler identification is GNU 4.8.5
-- Check for working C compiler: /usr/bin/cc
-- Check for working C compiler: /usr/bin/cc -- works
-- Detecting C compiler ABI info
-- Detecting C compiler ABI info - done
-- Detecting C compile features
-- Detecting C compile features - done
-- Check for working CXX compiler: /usr/bin/c++
-- Check for working CXX compiler: /usr/bin/c++ -- works
-- Detecting CXX compiler ABI info
-- Detecting CXX compiler ABI info - done
-- Detecting CXX compile features
-- Detecting CXX compile features - done
-- Configuring done
-- Generating done
-- Build files have been written to: /data/public/home/cchufeng/GithubProjects/examples_CMake/CMake_example/05_helloworld/test02/build
[root@ltcos01 build]$ make
Scanning dependencies of target Test02
[ 50%] Building CXX object CMakeFiles/Test02.dir/test02.cpp.o
[100%] Linking CXX static library libTest02.a
[100%] Built target Test02
[root@ltcos01 build]$ ls
CMakeCache.txt CMakeFiles cmake_install.cmake libTest02.a Makefile
[root@ltcos01 build]$
06 helloworld-指定编译器
- main.cpp
#include <iostream>
int main()
{
auto name = "jacking";
std::cout << "hello world: " << name << std::endl;
return 0;
}
- CMakeLists.txt
PROJECT(hello)
set(CMAKE_CXX_COMPILER g++)
add_definitions("-Wall -std=c++14")
ADD_EXECUTABLE(main main.cpp)
07 helloworld-使用外部库(此处为Boost库)
- main.cpp
#include <boost/thread.hpp>
#include <iostream>
int main()
{
std::cout << "Boost.Thread !!!" << std::endl;
boost::thread Thread1( [] ()
{
for( int i = 0; i < 5; ++i )
{
std::cout << "Thread Num : " << i << std::endl;
}
} );
Thread1.join();
return 0;
}
- CMakeLists.txt
PROJECT(hello)
set(CMAKE_CXX_COMPILER g++)
set(CMAKE_CXX_FLAGS "-m64")
add_definitions("-Wall -std=c++14")
INCLUDE_DIRECTORIES(/$ENV{HOME}/Dev/C++/ThirdParty/boost_1_60_0)
LINK_DIRECTORIES(/$ENV{HOME}/Dev/C++/ThirdParty/boost_1_60_0/stage/gcc/lib)
ADD_EXECUTABLE(hello-boost hello-boost.cpp)
TARGET_LINK_LIBRARIES(hello-boost pthread boost_thread boost_system boost_chrono)
- JavaScript 教程
- JavaScript 编辑工具
- JavaScript 与HTML
- JavaScript 与Java
- JavaScript 数据结构
- JavaScript 基本数据类型
- JavaScript 特殊数据类型
- JavaScript 运算符
- JavaScript typeof 运算符
- JavaScript 表达式
- JavaScript 类型转换
- JavaScript 基本语法
- JavaScript 注释
- Javascript 基本处理流程
- Javascript 选择结构
- Javascript if 语句
- Javascript if 语句的嵌套
- Javascript switch 语句
- Javascript 循环结构
- Javascript 循环结构实例
- Javascript 跳转语句
- Javascript 控制语句总结
- Javascript 函数介绍
- Javascript 函数的定义
- Javascript 函数调用
- Javascript 几种特殊的函数
- JavaScript 内置函数简介
- Javascript eval() 函数
- Javascript isFinite() 函数
- Javascript isNaN() 函数
- parseInt() 与 parseFloat()
- escape() 与 unescape()
- Javascript 字符串介绍
- Javascript length属性
- javascript 字符串函数
- Javascript 日期对象简介
- Javascript 日期对象用途
- Date 对象属性和方法
- Javascript 数组是什么
- Javascript 创建数组
- Javascript 数组赋值与取值
- Javascript 数组属性和方法
- GORM V2 自动迁移和迁移接口的方法
- Vue Nginx反向代理配置 解决生产环境跨域
- react的事件处理为什么要bind this 改变this的指向?
- 没有用到React,为什么我需要import引入React?
- JSON对象和JavaScript对象直接量的区别--不同之处
- (Demo分享)利用原生JavaScript-随机数-实现做一个烟花案例
- 关于DOM的理解
- 关于闭包函数和递归函数的详细理解
- 新手怎么学JS?JavaScript基础入门
- 纯CSS制作加<div>制作动画版哆啦A梦
- Ubuntu 18.04 安装使用 Supervisor 进程守护并设置开机自动启动
- AWS CLI version 2 在 Windows 下的安装
- Vue事件绑定原理
- 组合
- 用navicat进行身份验证连接出现cannot connect to Mongodb authentication failed