C++11 有两个新的字符串转换函数为unsigned long
and long long
:std::stoul()
和std::stoll()
.
最近的 Android NDK r9 引入了 Clang 3.3 编译器,据说它是 C++11 功能完整的。在 NDK 内部有这些函数的原型,但是我不能使用它们。
我需要做什么才能使用它们?
PS我已经做了LOCAL_CPPFLAGS += -std=c++11
C++11 有两个新的字符串转换函数为unsigned long
and long long
:std::stoul()
和std::stoll()
.
最近的 Android NDK r9 引入了 Clang 3.3 编译器,据说它是 C++11 功能完整的。在 NDK 内部有这些函数的原型,但是我不能使用它们。
我需要做什么才能使用它们?
PS我已经做了LOCAL_CPPFLAGS += -std=c++11
您无法使用这些功能的原因是根深蒂固的,不幸的是目前无法解决。
查看libs/armeabi-v7a/include/bits/c++config.h
gnu stdlibc++ 文件夹中的文件,您会看到:
...
/* Define if C99 functions or macros from <wchar.h>, <math.h>, <complex.h>,
<stdio.h>, and <stdlib.h> can be used or exposed. */
/* #undef _GLIBCXX_USE_C99 */
...
bits/basic_string.h
上面的内容,以及来自法术坏消息的以下片段:
...
#if (defined(__GXX_EXPERIMENTAL_CXX0X__) && defined(_GLIBCXX_USE_C99) \
&& !defined(_GLIBCXX_HAVE_BROKEN_VSWPRINTF))
/* The definitions of Numeric Conversions [string.conversions] */
#endif
...
因此,这些函数在 NDK 中是不可用的。
根本原因:根本原因似乎是 armeabi-v7a 平台上的 GNU stdlibc++ 中禁用了 C99 功能使用,因为 Bionic libc 不支持复杂的数学(Android 上的标准 C 库是 Bionic) .
可能的修复(未经测试):探索CrystaX 的 Android NDK,它似乎具有对 Vanilla Android NDK 的扩展。
注: __GXX_EXPERIMENTAL_CXX0X__
通过添加-std=gnu++11
或APP_CXXFLAGS
定义LOCAL_CXXFLAGS
详细测试日志:使用 NDK 版本 r8e
jni/Application.mk构建:
APP_STL := gnustl_static
APP_CXXFLAGS += -std=gnu++11
NDK_TOOLCHAIN_VERSION := 4.7
jni/Android.mk
LOCAL_PATH := $(call my-dir)
include $(CLEAR_VARS)
LOCAL_MODULE := cxx11
LOCAL_SRC_FILES := cxx11.cpp
include $(BUILD_EXECUTABLE)
jni/cxx11.cpp:
#include <iostream>
#include <string>
int main(int argc, char* argv[]) {
#if defined(__GXX_EXPERIMENTAL_CXX0X__)
std::cout<<"__GXX_EXPERIMENTAL_CXX0X__ defined."<<std::endl;
#else
std::cout<<"__GXX_EXPERIMENTAL_CXX0X__ not defined."<<std::endl;
#endif
#if defined(_GLIBCXX_USE_C99)
std::cout<<"_GLIBCXX_USE_C99 defined."<<std::endl;
#else
std::cout<<"_GLIBCXX_USE_C99 not defined."<<std::endl;
#endif
#if defined(_GLIBCXX_HAVE_BROKEN_VSWPRINTF)
std::cout<<"_GLIBCXX_HAVE_BROKEN_VSWPRINTF defined."<<std::endl;
#else
std::cout<<"_GLIBCXX_HAVE_BROKEN_VSWPRINTF not defined."<<std::endl;
#endif
#if (defined(__GXX_EXPERIMENTAL_CXX0X__) && defined(_GLIBCXX_USE_C99) \
&& !defined(_GLIBCXX_HAVE_BROKEN_VSWPRINTF))
std::string s="1";
std::cout<<"ll:"<<std::stoll(s)<<std::endl<<"ul:"<<std::stoul(s)<<std::endl;
#else
std::cout<<"No support for stoll/stoul."<<std::endl;
#endif
return(0);
}
Nexus 4 (Android 4.3) 上的输出:
u0_a51@mako:/ $ /data/local/tmp/cxx11
__GXX_EXPERIMENTAL_CXX0X__ defined.
_GLIBCXX_USE_C99 not defined.
_GLIBCXX_HAVE_BROKEN_VSWPRINTF not defined.
No support for stoll/stoul.
添加:
APP_STL := c++_static
到 Application.mk 解决了我无法访问 std::stoi() 的问题(使用 gcc 4.8.4)。
来自此Google Groups 帖子的Daniel Tavares 的解决方案。