如何在Android中使用std::stoul和std::stoul
How to use std::stoul and std::stoull in Android?
C++11有两个新的字符串转换函数:unsigned long
和long long
:std::stoul()
和std::stoll()
。
最近的Android NDK r9引入了Clang 3.3编译器,据说它是C++11功能完整的。NDK内部有这些函数的原型,但我不能使用它们。
我需要做些什么才能使用它们?
p.S.我已经做了LOCAL_CPPFLAGS += -std=c++11
您不能使用这些函数的原因是根深蒂固的,不幸的是目前无法解决。
查看gnu stdlibc++文件夹中的libs/armeabi-v7a/include/bits/c++config.h
文件,您将看到以下内容:
...
/* 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功能的使用,因为仿生libc不支持复杂的数学(Android上的标准C库是仿生的)。
可能的修复(未测试):探索CrystalX的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)。
Daniel Tavares的解决方案,来自谷歌群组的这篇帖子。
- 使用std::multimap迭代器创建std::list
- C++中std::resize(n)和std::shrink_to_fit之间的区别
- 来自 std::list 的迭代器 .end() 按预期返回"0xcdcdcdcdcdcdcdcd"但 .begin()
- C++17复制构造函数,在std::unordereded_map上进行深度复制
- 如何导出包含具有"std::unique_ptr"值的"std::map"属性的
- 从持续时间构造std::chrono::system_clock::time_point
- std::具有相同基类的类的变体
- std::向量与传递值的动态数组
- 使用std::vector的OpenCL矩阵乘法
- std::map<struct,struct>::find 找不到匹配项,但是如果我循环通过 begin() 到 end(),我在那里看到匹配项
- std::condition_variable::wait()如何评估给定的谓词
- 如何获取std::result_of函数的返回类型
- std::原子加载和存储都需要吗
- 将对象移动到std::shared_ptr
- POCO::PostgreSQL:如何将std::vector支持添加到`Binder::bind`
- std::stoul not throwing std::out_of_range
- Std::stoul在C++失败后没有被抓住
- std::stoul 在带有 NDK r13b 的 Android 版 Qt 应用程序上出错
- C++98 alternative to std::stoul?
- 如何在Android中使用std::stoul和std::stoul