gpt4 book ai didi

c++ - C 中的 constexpr(或等价物)

转载 作者:可可西里 更新时间:2023-11-01 18:16:00 24 4
gpt4 key购买 nike

我正在尝试使用散列函数让基于字符串的开关表达式在 C 中工作。即使代码是 C,我已经能够使用“constexpr”和 Clang/LLVM 转向 C++,使用干净的语法让它工作。

但是,将其编译为 C++ 当然会产生奇怪的副作用,例如缺少 void* 隐式转换,这会变得非常尴尬。

所以问题是如何解决这个难题(而不是让 C11 委员会因为为什么没有将其添加到 C 规范中而大吃一惊)

  1. 有没有办法用 C 启用 constexpr 选项?
  2. 有没有办法使用 C++ 启用隐式 void* 转换?
  3. 在 C11/C99 中是否有另一种无需重新计算哈希值的简洁方法?

这是我当前的示例代码:

constexpr uint64 cHash(char const* text, uint64 last_value = basis)
{
return *str ? cHash(text+1, (*text ^ last_value) * prime) : last_value;
}

void SwitchFunction(char const* text)
{
switch(Hash(text))
{
case cHash("first"):
break;
case cHash("second"):
break;
case cHash("third"):
break;
default:
break;
}
}

最佳答案

我来晚了一点,但最近遇到了同样的问题。

对于这样一个简单的散列函数,您可以使用 C 预处理器来实现它。缺点是预处理器无法将字符串拆分为字符,因此您必须编写 HASH('f','i','r' 而不是 hash("first") ,'s','')HASH 宏是使用 __VA_ARGS__ 实现的并适用于最多八个字符的字符串。

我还把你的哈希函数从递归函数变成了迭代函数,这样更容易阅读并且不需要可选参数。生成的程序集几乎相同 ( https://godbolt.org/z/1g8LPI )。

#include <stdio.h>

typedef unsigned long uint64;

#define HASH_BASIS 17UL
#define HASH_PRIME 11UL

#define HASH_1(ARG1) ((ARG1 ^ HASH_BASIS) * HASH_PRIME)
#define HASH_2(ARG1, ARG2) ((ARG2 ^ HASH_1(ARG1)) * HASH_PRIME)
#define HASH_3(ARG1, ARG2, ARG3) ((ARG3 ^ HASH_2(ARG1, ARG2)) * HASH_PRIME)
#define HASH_4(ARG1, ARG2, ARG3, ARG4) \
((ARG4 ^ HASH_3(ARG1, ARG2, ARG3)) * HASH_PRIME)
#define HASH_5(ARG1, ARG2, ARG3, ARG4, ARG5) \
((ARG5 ^ HASH_4(ARG1, ARG2, ARG3, ARG4)) * HASH_PRIME)
#define HASH_6(ARG1, ARG2, ARG3, ARG4, ARG5, ARG6) \
((ARG6 ^ HASH_5(ARG1, ARG2, ARG3, ARG4, ARG5)) * HASH_PRIME)
#define HASH_7(ARG1, ARG2, ARG3, ARG4, ARG5, ARG6, ARG7) \
((ARG7 ^ HASH_6(ARG1, ARG2, ARG3, ARG4, ARG5, ARG6)) * HASH_PRIME)
#define HASH_8(ARG1, ARG2, ARG3, ARG4, ARG5, ARG6, ARG7, ARG8) \
((ARG8 ^ HASH_7(ARG1, ARG2, ARG3, ARG4, ARG5, ARG6, ARG7)) * HASH_PRIME)

#define HASH_COUNT(ARG1, ARG2, ARG3, ARG4, ARG5, ARG6, ARG7, ARG8, func, ...) \
func

#define HASH(...) \
HASH_COUNT(__VA_ARGS__, HASH_8(__VA_ARGS__), HASH_7(__VA_ARGS__), \
HASH_6(__VA_ARGS__), HASH_5(__VA_ARGS__), HASH_4(__VA_ARGS__), \
HASH_3(__VA_ARGS__), HASH_2(__VA_ARGS__), HASH_1(__VA_ARGS__))

uint64 hash(const char *text) {
uint64 h = HASH_BASIS;
char c;
while ((c = *text++) != '\0') {
h = (c ^ h) * HASH_PRIME;
}
return h;
}

int main(int argc, char *argv[]) {
const char *text = argc > 1 ? argv[1] : "";
switch (hash(text)) {
case HASH('f', 'i', 'r', 's', 't'):
puts(text);
break;
case HASH('s', 'e', 'c', 'o', 'n', 'd'):
puts(text);
break;
case HASH('t', 'h', 'i', 'r', 'd'):
puts(text);
break;
default:
puts("oops");
break;
}
}

关于c++ - C 中的 constexpr(或等价物),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/22956376/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com