- mongodb - 在 MongoDB mapreduce 中,如何展平值对象?
- javascript - 对象传播与 Object.assign
- html - 输入类型 ="submit"Vs 按钮标签它们可以互换吗?
- sql - 使用 MongoDB 而不是 MS SQL Server 的优缺点
好的,所以我们在 C++ 17 中,对于 C++ 中非常棒的 bitflags 接口(interface)仍然没有令人满意的答案。
我们有 enum
将它们的成员值渗入封闭范围,但隐式转换为它们的基础类型,因此可以用作位标志,但拒绝重新分配回枚举而不进行转换。
我们有 enum class
这解决了名称范围问题,以便它们的值必须显式命名 MyEnum::MyFlag
甚至 MyClass::MyEnum::MyFlag
,但它们不会隐式转换为它们的底层类型,因此不能在没有无休止的来回转换的情况下用作位标志。
最后,我们有来自 C
的旧位域。如:
struct FileFlags {
unsigned ReadOnly : 1;
unsigned Hidden : 1;
...
};
FileFlags::ReadOnly | FileFlags::Hidden
- 对于位域来说,根本没有一个好方法。
constexpr
或
#define
命名位值,然后根本不使用枚举。这有效,但将位值与基础位标志类型完全分离。也许这最终不是最糟糕的方法,特别是如果位标志值是
constexpr
在一个结构中给他们自己的名字范围?
struct FileFlags {
constexpr static uint16_t ReadOnly = 0x01u;
constexpr static uint16_t Hidden = 0x02u;
...
}
Here is a type which has the following valid bit-flags in it, it has its own name-scope, and these bits and type should be freely usable with standard bitwise operators such as | & ^ ~, and they should be comparable to integral values such as 0, and the result of any bitwise operators should remain the named type, and not devolve into an integral
DEFINE_ENUM_FLAG_OPERATORS(EnumType)
然后定义运算符 | & ^ ~ 以及相关的赋值操作,例如 |= 等。enable_if
元编程以允许给定的枚举转换为位标志类型,该类型支持缺少的运算符并再次静默返回。 bit_flags<EnumType> flags
然后 flags
具有按位语义。这不能做的是让枚举的基数实际上直接正确地处理按位运算符,所以你不能说 EnumType::ReadOnly | EnumType::Hidden
即使使用 bit_flags<EnumType>
因为底层枚举本身仍然不支持必要的运算符。我最终不得不做与上面的 #1 和 #2 基本相同的事情,并启用 operator | (EnumType, EnumType)
对于各种按位运算符,要求用户为其枚举声明元类型的特化,例如 template <> struct is_bitflag_enum<EnumType> : std::true_type {};
template <> struct is_bitflag_enum<EnumType> : std::true_type {};
如#2 和部分#3) 在类范围内。这些必须发生在类或结构之外,因为 C++ 根本没有我所知道的允许我在类中进行此类声明的机制。
最佳答案
您可以在将枚举作为值的封闭类中使用友元函数。这可以在宏中使用来定义必要的功能,所有功能都在类范围内。
例如,为了避免 is_bitflag_enum
trait to specialize,专门化一个包含枚举和操作符的结构体。这就像#2,仍然不能在类里面完成。
#include <type_traits>
template<class Tag>
struct bitflag {
enum class type;
#define DEFINE_BITFLAG_OPERATOR(OP) \
friend constexpr type operator OP(type lhs, type rhs) noexcept { \
typedef typename ::std::underlying_type<type>::type underlying; \
return static_cast<type>(static_cast<underlying>(lhs) OP static_cast<underlying>(rhs)); \
} \
friend constexpr type& operator OP ## = (type& lhs, type rhs) noexcept { \
return (lhs = lhs OP rhs); \
}
DEFINE_BITFLAG_OPERATOR(|)
DEFINE_BITFLAG_OPERATOR(&)
DEFINE_BITFLAG_OPERATOR(^)
#undef DEFINE_BITFLAG_OPERATOR
#define DEFINE_BITFLAG_OPERATOR(OP) \
friend constexpr bool operator OP(type lhs, typename ::std::underlying_type<type>::type rhs) noexcept { \
return static_cast<typename ::std::underlying_type<type>::type>(lhs) OP rhs; \
} \
friend constexpr bool operator OP(typename ::std::underlying_type<type>::type lhs, type rhs) noexcept { \
return lhs OP static_cast<typename ::std::underlying_type<type>::type>(rhs); \
}
DEFINE_BITFLAG_OPERATOR(==)
DEFINE_BITFLAG_OPERATOR(!=)
DEFINE_BITFLAG_OPERATOR(<)
DEFINE_BITFLAG_OPERATOR(>)
DEFINE_BITFLAG_OPERATOR(>=)
DEFINE_BITFLAG_OPERATOR(<=)
#undef DEFINE_BITFLAG_OPERATOR
friend constexpr type operator~(type e) noexcept {
return static_cast<type>(~static_cast<typename ::std::underlying_type<type>::type>(e));
}
friend constexpr bool operator!(type e) noexcept {
return static_cast<bool>(static_cast<typename ::std::underlying_type<type>::type>(e));
}
};
// The `struct file_flags_tag` (Which declares a new type) differentiates between different
// enum classes declared
template<> enum class bitflag<struct file_flags_tag>::type {
none = 0,
readable = 1 << 0,
writable = 1 << 1,
executable = 1 << 2,
hidden = 1 << 3
};
using file_flags = bitflag<file_flags_tag>::type;
bool is_executable(file_flags f) {
return (f & file_flags::executable) == 0;
}
#include <type_traits>
#define MAKE_BITFLAG_FRIEND_OPERATORS_BITWISE(OP, ENUM_TYPE) \
friend constexpr ENUM_TYPE operator OP(ENUM_TYPE lhs, ENUM_TYPE rhs) noexcept { \
typedef typename ::std::underlying_type<ENUM_TYPE>::type underlying; \
return static_cast<ENUM_TYPE>(static_cast<underlying>(lhs) OP static_cast<underlying>(rhs)); \
} \
friend constexpr ENUM_TYPE& operator OP ## = (ENUM_TYPE& lhs, ENUM_TYPE rhs) noexcept { \
return (lhs = lhs OP rhs); \
}
#define MAKE_BITFLAG_FRIEND_OPERATORS_BOOLEAN(OP, ENUM_TYPE) \
friend constexpr bool operator OP(ENUM_TYPE lhs, typename ::std::underlying_type<ENUM_TYPE>::type rhs) noexcept { \
return static_cast<typename ::std::underlying_type<ENUM_TYPE>::type>(lhs) OP rhs; \
} \
friend constexpr bool operator OP(typename ::std::underlying_type<ENUM_TYPE>::type lhs, ENUM_TYPE rhs) noexcept { \
return lhs OP static_cast<typename ::std::underlying_type<ENUM_TYPE>::type>(rhs); \
}
#define MAKE_BITFLAG_FRIEND_OPERATORS(ENUM_TYPE) \
public: \
MAKE_BITFLAG_FRIEND_OPERATORS_BITWISE(|, ENUM_TYPE) \
MAKE_BITFLAG_FRIEND_OPERATORS_BITWISE(&, ENUM_TYPE) \
MAKE_BITFLAG_FRIEND_OPERATORS_BITWISE(^, ENUM_TYPE) \
MAKE_BITFLAG_FRIEND_OPERATORS_BOOLEAN(==, ENUM_TYPE) \
MAKE_BITFLAG_FRIEND_OPERATORS_BOOLEAN(!=, ENUM_TYPE) \
MAKE_BITFLAG_FRIEND_OPERATORS_BOOLEAN(<, ENUM_TYPE) \
MAKE_BITFLAG_FRIEND_OPERATORS_BOOLEAN(>, ENUM_TYPE) \
MAKE_BITFLAG_FRIEND_OPERATORS_BOOLEAN(>=, ENUM_TYPE) \
MAKE_BITFLAG_FRIEND_OPERATORS_BOOLEAN(<=, ENUM_TYPE) \
friend constexpr ENUM_TYPE operator~(ENUM_TYPE e) noexcept { \
return static_cast<ENUM_TYPE>(~static_cast<typename ::std::underlying_type<ENUM_TYPE>::type>(e)); \
} \
friend constexpr bool operator!(ENUM_TYPE e) noexcept { \
return static_cast<bool>(static_cast<typename ::std::underlying_type<ENUM_TYPE>::type>(e)); \
}
// ^ The above in a header somewhere
class my_class {
public:
enum class my_flags {
none = 0, flag_a = 1 << 0, flag_b = 1 << 2
};
MAKE_BITFLAG_FRIEND_OPERATORS(my_flags)
bool has_flag_a(my_flags f) {
return (f & my_flags::flag_a) == 0;
}
};
关于c++ - 追求更好的位标志枚举,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/49653901/
我有两个维度 DimFlag 和 DimPNL 以及一个事实表 FactAmount 。我正在寻找:当 pnl 是 stat(Is Stat=1) 时:sum (Actual x FlagId)对于
我试图弄清楚登录模块标志在 JAAS 中是如何工作的(使用 JBoss 5.1 EAP),我遇到了一个令人费解的情况,我希望有人能为我澄清一下。 对于背景,我的 login-config.xml 如下
关闭。这个问题是off-topic .它目前不接受答案。 想改进这个问题吗? Update the question所以它是on-topic用于堆栈溢出。 关闭 9 年前。 Improve this
我正在通过 gradle 使用 doclet 运行 javadoc,当我运行 javadoc/doclet 任务时,我收到下一个错误: error - invalid flag: -doctitle
我尝试使用sqoop --where标志将特定的行从MySQL表导入到HDFS,但是结果不符合预期。 命令: sqoop import \ --connect "jdbc:mysql://XXXX
我有一个语言面板,其中有一个图像 (main-image),显示页面加载时的情况。我还有三个额外的图像,它们在页面加载时隐藏。 问题是当点击附加图像之一时如何切换主图像。我需要使用单击的 image
奇怪...在 StackOverflow 上有很多关于此 attr 的问题,但没有人回答我的以下问题: 我有一个span(仅作为示例),其中启用了ContentEditable。我只想保存更改的元素(
我正在使用 ChartJS 2.0 在 UI 上绘制图表。而且我能够呈现饼图。但我希望鼠标悬停时显示数据以及“%”符号。我如何追加 % 因此,如果在鼠标悬停时我得到 Rented: 93 我想看到 R
我使用的是 Servlet 3.0,我想用 HttpOnly 标志保护我的 cookie。我的 web.xml 是 true
我有一个简单的服务: public class TestService extends Service { final String LOG_TAG = "myLogs"; public void o
我正在尝试将 wget 与包含“#”符号的 url 一起使用。无论我做什么来逃避这个角色,它都不起作用。我用过\、' 和 "。但它们都不起作用。有人有什么建议吗? 谢谢! 最佳答案 如果您真的想让它有
我正在尝试创建一个数据库,但我不知道如何转义数据库名称中的 - 符号。 mysql> create database happy-face; 给我一个错误 mysql> create databa
我为我的计算机科学类(class)编写了一个程序,它读取一个文件并导入数据,然后只添加数字,但它似乎添加了一个额外的加号。 import java.io.*; //necessary for File
可能是个愚蠢的问题,但我怎样才能在与某些文本看到图像相同的行中获取图像(在本例中为标志)? 到目前为止我的 HTML 代码: FRA 最佳答案 试试这个: img { height:20px
我需要一些有关 clone() 系统调用的帮助。我试图将它与标志 CLONE_CHILD_CLEARTID 一起使用,但我看不到我指定为参数的字段值有任何变化。这是一个简单的代码: int the_c
查看 mySQL 转储时,我遇到了一些东西,想知道它们是什么。 我明白了: /*!50001 DROP TABLE IF EXISTS `xxx` */; flag 50001是什么意思,有什么意思的
是否可以传递任何 Java 编译器标志来告诉编译器不允许使用原始类型?也就是说,对于任何泛型类,让编译器强制使用参数化版本,否则抛出编译错误? 最佳答案 JDK7 (b38) 介绍 -Xlint:ra
[Flags] public enum MyEnum { None = 0, Setting1 = (1 GetAllEnums() where T : struct
我正在浏览 PackageManager API。我发现定义了以下常量: 1) GET_DISABLED_COMPONENTS 2) GET_DISABLED_UNTIL_USED_COMPONENT
我编写了一个 Go 程序来模拟按键操作。为此,我必须使用 cgo 和不同的 C 代码片段,具体取决于正在编译 Go 代码的操作系统。我编写的代码如下所示: package keyboard /* #i
我是一名优秀的程序员,十分优秀!