CMake是一個跨平臺的Makefile生成工具茫藏,可以根據(jù)特定的規(guī)則生成相應(yīng)的Makefile文件,并對C/C++源代碼進行編譯和管理包个。
有一篇博客介紹CMake的使用刷允,比較通俗易懂冤留,鏈接地址是:Cmake 詳解
CMake的官方下載地址為:https://cmake.org/download/
官方文檔地址為:CMake 3.16 Documentation
官方的CMake指南地址為:CMake Tutorial
一碧囊、CMake中添加對C++11的支持
1、在對應(yīng)的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")
但是總會出現(xiàn)cc1plus: error: unrecognized command line option "-std=c++11" 報錯糯而。
所以set(QMAKE_CXXFLAGS "-std=c++11") 類似的寫法肯定不行。
后來發(fā)現(xiàn)是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-設(shè)置編譯器選項祭芦。 -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目錄結(jié)構(gòu)如下:
[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 創(chuàng)建靜態(tài)文件后
05_helloworld源代碼目錄樹結(jié)構(gòu)如下所示:
[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以及相應(yīng)的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)指令會生成相應(yīng)的靜態(tài)庫文件libTest01.a
test02目錄下和test01目錄結(jié)構(gòu)一樣,也有test02.h和test02.cpp以及相應(yīng)的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目錄下執(zhí)行cmake命令會生成相應(yīng)的靜態(tài)庫文件libTest02.a胃夏。具體操作過程如下:新建一個build目錄,然后進入到build目錄下執(zhí)行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)