Как активировать C ++ 11 в CMake?

Когда я пытаюсь запустить сгенерированный CMake make-файл для компиляции моей программы, я получаю сообщение об ошибке

range based for loops are not supported in C++ 98 mode.

Я пытался добавитьadd_definitions(-std=c++0x) к моемуCMakeLists.txt, но это не помогло. Я тоже попробовал это:

if(CMAKE_COMPILER_IS_GNUCXX)
    add_definitions(-std=gnu++0x)
endif()

Когда я делаюg++ --version, Я получил:

g++ (Ubuntu/Linaro 4.6.1-9ubuntu3) 4.6.1

Я также пыталсяSET(CMAKE_CXX_FLAGS "-std=c++0x"), который тоже не работает.

Я не понимаю, как я могу активировать функции C ++ 11 с помощью CMake.

Ответы на вопрос(13)

Если вы хотите всегда активировать новейший стандарт C ++, вот мое расширениеОтвет Дэвида Грейсонав свете недавних (CMake 3.8, CMake 3.11) добавлений значений 17 и 20 для CMAKE_CXX_STANDARD):

IF (CMAKE_VERSION VERSION_LESS "3.8")
    SET(CMAKE_CXX_STANDARD 14)
ELSEIF (CMAKE_VERSION VERSION_LESS "3.11")
    SET(CMAKE_CXX_STANDARD 17)
ELSE() # CMake 3.11 or higher:
    SET(CMAKE_CXX_STANDARD 20)
ENDIF()
# Typically, you'll also want to turn off compiler-specific extensions:
SET(CMAKE_CXX_EXTENSIONS OFF)

(используйте этот код вместоset (CMAKE_CXX_STANDARD 11) в связанном ответе).

На современном CMake (& gt; = 3.1) лучший способ установить глобальные требования:

set(CMAKE_CXX_STANDARD 11)
set(CMAKE_CXX_STANDARD_REQUIRED ON)
set(CMAKE_CXX_EXTENSIONS OFF)

Это переводится как «Я хочу C ++ 11 для всех целей, это не является обязательным, я не хочу использовать какие-либо расширения GNU или MS». Начиная с c ++ 17, это ИМХО лучший способ.

Источник: https://crascit.com/2015/03/28/enabling-cxx11-in-cmake/

OS X и Homebrew LLVM связаны с:

Не забудьте вызвать cmake_minimum_required (ВЕРСИЯ 3.3) и project () после него!

Или CMake вставитproject() неявно перед строкой 1, вызывая проблемы с обнаружением версии Clang и, возможно, другие виды проблем. Вотсвязанная проблема.

Самый простой способ установить стандарт Cxx:

 set_property(TARGET tgt PROPERTY CXX_STANDARD 11)

Увидетьдокументация CMake Больше подробностей.

Я думаю, что только этих двух строк достаточно.

set(CMAKE_CXX_STANDARD 11)

set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -std=c++11")

Для CMake 3.8 и новее вы можете использовать

target_compile_features(target PUBLIC cxx_std_11)

CMake 3.1 представилCMAKE_CXX_STANDARD переменная, которую вы можете использовать. Если вы знаете, что у вас всегда будет доступен CMake 3.1, вы можете просто написать это в своем файле CMakeLists.txt верхнего уровня или поместить его прямо перед тем, как будет определена любая новая цель:

set (CMAKE_CXX_STANDARD 11)

Если вам нужно поддерживать более старые версии CMake, вот макрос, который я придумал, который вы можете использовать:

macro(use_cxx11)
  if (CMAKE_VERSION VERSION_LESS "3.1")
    if (CMAKE_CXX_COMPILER_ID STREQUAL "GNU")
      set (CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -std=gnu++11")
    endif ()
  else ()
    set (CMAKE_CXX_STANDARD 11)
  endif ()
endmacro(use_cxx11)

Макрос поддерживает только GCC прямо сейчас, но он должен быть прост, чтобы распространить его на другие компиляторы.

Тогда вы могли бы написатьuse_cxx11() в верхней части любого файла CMakeLists.txt, который определяет цель, которая использует C ++ 11.

CMake issue #15943 for clang users targeting macOS

Если вы используете CMake и Clang для MacOS, естьошибка что может вызватьCMAKE_CXX_STANDARD возможность просто не работать (не добавлять флаги компилятора). Убедитесь, что вы делаете одну из следующих вещей:

  • Use cmake_minimum_required to require CMake 3.0 or later, or
  • Set policy CMP0025 to NEW with the following code at the top of your CMakeLists.txt file before the project command:

    # Fix behavior of CMAKE_CXX_STANDARD when targeting macOS.
    if (POLICY CMP0025)
      cmake_policy(SET CMP0025 NEW)
    endif ()
    

Команда CMaketarget_compile_features() используется для указания требуемой функции C ++cxx_range_for, Затем CMake будет стимулировать использование стандарта C ++.

cmake_minimum_required(VERSION 3.1.0 FATAL_ERROR)
project(foobar CXX)
add_executable(foobar main.cc)
target_compile_features(foobar PRIVATE cxx_range_for)

Там нет необходимости использоватьadd_definitions(-std=c++11) или изменить переменную CMakeCMAKE_CXX_FLAGSпотому что CMake убедится, что компилятор C ++ вызывается с соответствующими флагами командной строки.

Может быть, ваша программа C ++ использует другие функции C ++, чемcxx_range_for, Глобальное свойство CMakeCMAKE_CXX_KNOWN_FEATURES перечисляет функции C ++, которые вы можете выбрать.

Вместо того, чтобы использоватьtarget_compile_features() Вы также можете явно указать стандарт C ++, установив свойства CMake CXX_STANDARD  а также CXX_STANDARD_REQUIRED для вашей цели CMake.

Смотрите такжемой более подробный ответ.

Что работает для меня, это установить следующую строку в вашем CMakeLists.txt:

set (CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -std=c++11")

Установка этой команды активирует функции C ++ 11 для компилятора и после выполненияcmake .. команда, вы должны быть в состоянии использоватьrange based for loops в вашем коде и скомпилируйте его без ошибок.

Самый простой способ:

add_compile_options(-std=c++11)

я использую

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, g++ 4.6.1 довольно старый Попробуй получить новееg++ версия.

Это еще один способ включить поддержку C ++ 11,

ADD_DEFINITIONS(
    -std=c++11 # Or -std=c++0x
    # Other flags
)

Я встречал случаи, когда только этот метод работает, а другие методы не работают. Возможно, это как-то связано с последней версией CMake.

Решение Вопроса

Как выясняется,SET(CMAKE_CXX_FLAGS "-std=c++0x") действительно активирует многие функции C ++ 11. Причина, по которой это не сработало, заключалась в том, что заявление выглядело так:

set(CMAKE_CXX_FLAGS "-std=c++0x ${CMAKE_CXX_FLAGS} -g -ftest-coverage -fprofile-arcs")

Следуя этому подходу, каким-то образом-std=c++0x Флаг был перезаписан, и это не сработало. Установка флагов одна за другой или использование метода списка работает.

list( APPEND CMAKE_CXX_FLAGS "-std=c++0x ${CMAKE_CXX_FLAGS} -g -ftest-coverage -fprofile-arcs")

Ваш ответ на вопрос