- android - RelativeLayout 背景可绘制重叠内容
- android - 如何链接 cpufeatures lib 以获取 native android 库?
- java - OnItemClickListener 不起作用,但 OnLongItemClickListener 在自定义 ListView 中起作用
- java - Android 文件转字符串
基本上,我想编写这样的代码:
std::vector<float> a = { 54, 25, 32.5 };
std::vector<int> b = { 55, 65, 6 };
std::cout << a << b << std::string("lol");
这是不可能的,因为 operator<<(ostream&, vector)
没有过载
所以,我编写了一个函数来完成这项工作:
template<template<typename...> typename T, typename ...Args>
std::enable_if_t<is_iterable_v<T<Args...>>>, std::ostream> &operator<<(std::ostream &out, T<Args...> const &t) {
for (auto const &e : t)
out << e << " ";
out << std::endl;
return out;
}
效果很好,但我对字符串有疑问。因为字符串是可迭代的并且字符串有 operator<<
功能。
所以我测试了另一个特征,比如 !is_streamable_out && _is_iterable
测试类似的东西:std::declval<std::ostream&>() << std::declval<T>()
以及它是否具有开始/结束功能。它在 MSVC 上运行良好,但在 Clang 上运行良好(我认为这是因为编译器也使用了我刚刚创建的函数,所以它发现一个重载可用于所有方法)。
所以,我目前正在使用 !is_same_v<string, T>
但这并不完美恕我直言。
有没有办法在不重新声明函数的情况下知道函数是否存在?
基本上,我想做那样的事情
if function foo does not exist for this type.
then function foo for this type is ...
这不是同一个问题 Is it possible to write a template to check for a function's existence?因为在另一个线程中,函数并不完全相同(toString 与 toOptionalString)。就我而言,功能是相同的。
这是我的完整代码:
template <class...>
struct make_void { using type = void; };
template <typename... T>
using void_t = typename make_void<T...>::type; // force SFINAE
namespace detail {
template<typename AlwaysVoid, template<typename...> typename Operator, typename ...Args>
struct _is_valid : std::false_type {};
template<template<typename...> typename Operator, typename ...Args>
struct _is_valid<void_t<Operator<Args...>>, Operator, Args...> : std::true_type { using type = Operator<Args...>; };
}
template<template<typename ...> typename Operator, typename ...Args>
using is_valid = detail::_is_valid<void, Operator, Args...>;
#define HAS_MEMBER(name, ...)\
template<typename T>\
using _has_##name = decltype(std::declval<T>().name(__VA_ARGS__));\
\
template<typename T>\
using has_##name = is_valid<_has_push_back, T>;\
\
template<typename T>\
constexpr bool has_##name##_v = has_##name<T>::value
HAS_MEMBER(push_back, std::declval<typename T::value_type>());
HAS_MEMBER(begin);
HAS_MEMBER(end);
template<typename T>
using is_iterable = std::conditional_t<has_begin_v<T> && has_end_v<T>, std::true_type, std::false_type>;
template<typename T>
constexpr bool is_iterable_v = is_iterable<T>::value;
template<class T, class Stream, class = void>
struct can_print : std::false_type {};
template<class T, class Stream>
struct can_print<T, Stream, void_t<decltype(std::declval<Stream&>() << std::declval<const T&>())>> : std::true_type {};
template<class T, class Stream = std::ostream>
constexpr bool can_print_v = can_print<T, Stream>::value;
template<typename T>
std::enable_if_t<is_iterable_v<T> && !can_print_v<T>, std::ostream> &operator<<(std::ostream &out, T const &t) {
for (auto const &e : t)
out << e << " ";
out << std::endl;
return out;
}
template<typename A, typename B>
std::ostream &operator<<(std::ostream &out, std::pair<A, B> const &p) {
out << p.first << " " << p.second << std::endl;
return out;
}
template<typename T>
std::enable_if_t<has_push_back_v<T>, T> &operator<<(T &c, typename T::value_type const &e) {
c.push_back(e);
return c;
}
和主要的:
#include <iostream>
#include <vector>
#include "Tools/stream.h"
#include <string>
#include <map>
int main() {
std::vector<float> a = { 54, 25, 32.5 };
std::vector<int> b = { 55, 65, 6 };
std::cout << a;
std::cout << has_push_back<float>::value << " " << has_push_back<std::vector<float>>::value << std::endl;
std::cout << is_iterable<std::vector<float>>::value << " " << is_iterable<float>::value << std::endl;
getchar();
return 0;
}
最佳答案
How to avoid this sentence is false in a template SFINAE?提供解决您问题的答案——重载 <<(ostream&, Ts...)
,将发现其优先级低于任何其他 <<
过载。
同时我要说你的计划很糟糕。 std
的重载运算符types 是一个糟糕的计划,原因有两个。
首先,除非有充分的理由,否则您应该不愿意为不属于您的类型重载运算符。
其次,如果你这样做,你应该在类型的命名空间中进行,你不能注入(inject)你的<<
。进入namespace std
而不会使您的程序格式错误。
在不同于所讨论类型的 namespace 中重载的运算符只能在您进行重载的 namespace 中找到。在与参数相关的 namespace 中重载的运算符随处可见。
这导致脆弱 <<
仅适用于一个命名空间。
因此,改为这样做:
struct print_nothing_t {};
inline std::ostream& operator<<(std::ostream& os, print_nothing_t) {
return os;
}
template<class C, class Sep=print_nothing_t>
struct stream_range_t {
C&& c;
Sep s;
template<class T=print_nothing_t>
stream_range_t( C&& cin, T&& sin = {} ):
c(std::forward<C>(cin)),
s(std::forward<T>(sin))
{}
friend std::ostream& operator<<(std::ostream& os, stream_range_t&& self) {
bool first = true;
for (auto&& x:self.c) {
if (!first) os << self.s;
os << decltype(x)(x);
first = false;
}
return os;
}
};
template<class C, class Sep = print_nothing_t>
stream_range_t<C, Sep> stream_range( C&& c, Sep&& s={} ) {
return {std::forward<C>(c), std::forward<Sep>(s)};
}
现在,如果您希望嵌套 vector 起作用,您必须实现一个流范围,将适配器应用于其内容。
Live example使用此测试代码:
std::vector<int> v{1,2,3,4};
std::cout << stream_range(v, ',') << "\n";
输出是1,2,3,4
.
如何使这个递归:
我们可以写一个adapt_for_streaming
分派(dispatch)给 identity
的函数(对于已经流式传输的东西),并发送到 stream_range
对于可迭代但尚未流式传输的事物。
然后用户可以添加新的 adapt_for_streaming
其他类型的重载。
stream_range_t
然后使用 adapt_for_streaming
进行流式传输关于其内容。
接下来,我们可以将元组/对/数组重载添加到 adapt_for_streaming
突然std::vector< std::vector< std::tuple<std::string, int> > >
可以流式传输。
终端用户可以调用stream_range
直接,或adapt_for_streaming
, 串一个可迭代的容器。您甚至可以调用 stream_range
直接在 std::string
上将其视为 char
的可流式集合而不是字符串。
关于c++ - SFINAE : Know if a function already exist or no,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/45086398/
我正在使用 COM 库 RCWs 。我一直发现最好的做法是在所有非托管资源超出范围之前手动释放它们并进行垃圾收集。我不希望这“减慢”我的应用程序的速度,并且我不在乎这实际上何时完成(如果有的话)在应用
我认为作为一个挑战,我应该编写一个基于 javascript 的游戏。我想要声音、图像和输入。模拟屏幕的背景(例如 640x480,其中包含我的所有图像)对于将页面的其余部分与“游戏”分开非常有用。我
在抓取网站的单个页面时,我使用 Beautiful Soup 取得了巨大成功,但我有一个新项目,我必须在其中检查大量网站,看看它们是否包含指向我网站的提及或链接。因此,我需要检查每个站点的整个站点。
我期待构建一个高度依赖地理数据的应用程序。 该应用程序将使用 HTML5 获取 GPS 数据的能力,并将进行诸如寻找最近的街道、寻找两点之间的最短路径等计算。我正在考虑使用 Google map 等平
我在使用 liquibase 时遇到问题。当我运行我的 J2EE 应用程序时,liquibase 告诉我: liquibase.exception.ValidationFailedException:
我的 iphone 上有一个应用程序,它允许我也可以通过 ftp 从我的 linux 桌面传输歌曲。应用显示主机地址为192.168.0.4并且要使用的端口是5021 .我可以从 filezilla
回答者已经知道的东西,但无论如何在这里展示我的思考过程: 从 HLL 到机器代码,这里有一组粗略的发生的事件(有链接器和其他东西,但现在让我们忽略它): HLL --> 编译器 --> 汇编器 ---
显然,下面的函数是不可能的,因为不可能永久地解开一个 IO 值(忽略 unsafePerformIO 或类似的): unwrapIO :: IO String -> String unwrapIO (
我在测试机器上安装了 Java 版本 45。高安全性设置表示无法在旧版本的 Java 上运行未签名或自签名的应用程序,事实上它无法运行自签名的小程序。 这个版本如何能够检测到它是旧版本?首次部署时它肯
怎么可能跳过 SemaphoreCI 上的一些测试?换句话说,是否有可能知道我们何时对 Semaphore 运行特定测试? 最佳答案 SemaphoreCI 将 CI 环境变量设置为 true。 有关
我知道我可以使用 xcodebuild 启动应用程序的单元测试,但我想知道什么告诉应用程序在启动期间运行测试,它是发送给应用程序的特殊参数还是它为了运行测试(使用 XCTest)进行不同的编译? 最佳
我对 Joomla 相当陌生(我更喜欢使用 Wordpress),我有一个关于模块位置的问题。 模块可以知道它所处的位置吗?例如我可以做这样的事情: if(modulePosition =='left
我有一个简单的 Spring Boot 应用程序,它从 Kafka 读取并写入 Kafka。我写了一个 SpringBootTest使用 EmbeddedKafka测试这一切。 主要问题是:有时测试失
我想知道 R 对它运行的硬件/系统有什么了解。 例如,我知道使用“sessionInfo()”会发现一些事情。但是是否也可以检查计算机的 CPU/内存?是否可以为所使用的计算机设置一些唯一标识符? 动
我试图掌握 PHP 和 MYSQL 编程,但在看到示例后,我无法理解 mysql_fetch_row 如何知道要返回哪一行。例如: 在上面的代码中,echo $row[0]; 返回表中第一列的数据,
已关闭。此问题需要 debugging details 。目前不接受答案。 编辑问题以包含 desired behavior, a specific problem or error, and the
我得到了一个 makefile,我将其修改为: ############################################ # Makefile using OCI (Oracle Ca
关闭。这个问题需要更多focused .它目前不接受答案。 想改进这个问题吗? 更新问题,使其只关注一个问题 editing this post . 关闭 4 年前。 Improve this qu
相关:What does casting do at compiler/machine level? 假设我有自定义类型。 A 型是 B 型的子类型。 最初,我的变量类型为 A 类型。然后我将其转换为
想到这里有个奇怪的问题。例如,假设您在服务器上部署了执行以下操作的代码: //GET request called when a URL is hit public void gETCalled(){
我是一名优秀的程序员,十分优秀!