- Java 双重比较
- java - 比较器与 Apache BeanComparator
- Objective-C 完成 block 导致额外的方法调用?
- database - RESTful URI 是否应该公开数据库主键?
在寻找一种方法来为 c 风格的数组进行可移植、安全的元素计数时,我找到了这个解决方案:
template <typename T, unsigned N> char (&arrayCountofHelper(T(&)[N]))[N];
#define ARRAY_COUNTOF(arr) (sizeof(arrayCountofHelper(arr)))
arrayCountofHelper 似乎实际上是对函数的引用,宏 ARRAY_COUNTOF 使用函数的大小是其返回类型的大小这一事实。
它工作得很好。
然而,当我试图检查它是否可移植时,我没有找到任何证据。我在标准 (14882/1998) 中找不到任何对 sizeof(function reference) 的引用,事实上,我不完全确定我什至可以再创建一个函数引用(尽管该标准确实提到了几个次)。
那么,有人知道我应该看标准中的什么地方吗? (或者,如果我以某种方式误解了声明,那么正确的解释是什么?)
谢谢
奥伦
附言(对于那些认为我没有找到适合我的问题的解决方案的人)
我知道我可以随时使用
#define ARRAY_COUNTOF sizeof(arr)/sizeof(arr[0])
甚至
template <typename T, unsigned N> size_t arrayCountof(T(&)[N]) {return N;}
但第一个不会检查 arr 是否为数组(或指针),第二个在 static_assert 中不可用。
(我会使用 std::tr1::array 或 std::vector,但这是我正在维护的遗留代码)
最佳答案
我们将松散地把它分开,我使用的是 INCITS+ISO+IEC+14882-2003。我会引用一些小的东西,但一些更复杂的东西太大而无法引用。
sizeof
在 §5.3.3 中定义,它说(删节):
The sizeof operator yields the number of bytes in the object representation of its operand. The operand is either an expression, which is not evaluated, or a parenthesized type-id.
换句话说,它产生一个类型的大小(以字节为单位)或找到一个表达式的类型并产生它的大小。我们没有类型,我们有表达式 arrayCountofHelper(arr)
.
您可以通过查看 primary-expression
的定义来剖析此表达式和 postfix-expression
分别在 §5.1 和 §5.2 中定义。你会发现它是一个 postfix-expression
并符合函数调用的要求(§5.2.2)。
现在回到sizeof
.我们只关心这个函数调用表达式的类型(所以我们可以产生它的大小),§5.2.2/3 说:
The type of the function call expression is the return type of the statically chosen function [...]. This type shall be a complete object type, a reference type or the type
void
.
因此我们需要使用 arrayCountofHelper(arr)
找到将被调用的函数的返回类型(记住,这都是未计算的) . arrayCountofHelper
是一个函数模板,我们将对其进行实例化(第 14.7 节),因此我们需要在实例化函数之前执行此操作以获取其返回类型。
所有模板参数都需要有值(§14.8.2),通过使用§14.8.2.1 中定义的规则,我们将找到 T
和 N
通过将传递给函数的数组与函数参数(这是对数组的引用)进行匹配。 (例如,如果 arr
是 int[10]
,T
将是 int
而 N
将是 10。)一旦我们有了这些,就可以实例化该函数。
一旦实例化,函数的返回类型将为char(&)[N]
*,对 N
数组的引用char
的。 (如果您需要解析方面的帮助,请参阅§8.3.5。关于如何解析“复杂”类型的 SO 也有问题。)所以现在我们已经找到了表达式的类型,我们必须获取它的大小。
§5.3.3/2 定义了如何 sizeof
使用引用和数组(强调我的):
When applied to a reference or a reference type, the result is the size of the referenced type. When applied to a class, the result is the number of bytes in an object of that class including any padding required for placing objects of that type in an array. The size of a most derived class shall be greater than zero (1.8). The result of applying sizeof to a base class subobject is the size of the base class type.70) When applied to an array, the result is the total number of bytes in the array. This implies that the size of an array of n elements is n times the size of an element.
引用类型的大小是其引用类型的大小,所以我们需要char[N]
的大小.这个的大小是N * sizeof(char)
. char
是最基本的,因为它是最小的类型;即 sizeof(char)
总是一个。 (§5.3.3/1) 所以这个表达式产生的大小是 1 * N
,或者我们一直想要的:N
.
这就是它的工作原理。
这个比你最后一个例子 arrayCountof
更受欢迎的原因是因为 sizeof
的结果是一个常量表达式,所以可以用在需要常量表达式的地方。
应该注意的是,在 C++0x 中,我们可以通过以下方式获得干净的无宏语法:
template <typename T, unsigned N>
constexpr size_t arrayCountof(T(&)[N]) {return N;}
* 函数返回类型是对数组而不是数组的引用的原因很简单,因为您不能返回数组。如果可以,任何一个选择都可以。
关于c++ - 函数引用 sizeof 是可移植的吗?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/2944794/
#include using namespace std; class C{ private: int value; public: C(){ value = 0;
这个问题已经有答案了: What is the difference between char a[] = ?string?; and char *p = ?string?;? (8 个回答) 已关闭
关闭。此题需要details or clarity 。目前不接受答案。 想要改进这个问题吗?通过 editing this post 添加详细信息并澄清问题. 已关闭 7 年前。 此帖子已于 8 个月
除了调试之外,是否有任何针对 c、c++ 或 c# 的测试工具,其工作原理类似于将独立函数复制粘贴到某个文本框,然后在其他文本框中输入参数? 最佳答案 也许您会考虑单元测试。我推荐你谷歌测试和谷歌模拟
我想在第二台显示器中移动一个窗口 (HWND)。问题是我尝试了很多方法,例如将分辨率加倍或输入负值,但它永远无法将窗口放在我的第二台显示器上。 关于如何在 C/C++/c# 中执行此操作的任何线索 最
我正在寻找 C/C++/C## 中不同类型 DES 的现有实现。我的运行平台是Windows XP/Vista/7。 我正在尝试编写一个 C# 程序,它将使用 DES 算法进行加密和解密。我需要一些实
很难说出这里要问什么。这个问题模棱两可、含糊不清、不完整、过于宽泛或夸夸其谈,无法以目前的形式得到合理的回答。如需帮助澄清此问题以便重新打开,visit the help center . 关闭 1
有没有办法强制将另一个 窗口置于顶部? 不是应用程序的窗口,而是另一个已经在系统上运行的窗口。 (Windows, C/C++/C#) 最佳答案 SetWindowPos(that_window_ha
假设您可以在 C/C++ 或 Csharp 之间做出选择,并且您打算在 Windows 和 Linux 服务器上运行同一服务器的多个实例,那么构建套接字服务器应用程序的最明智选择是什么? 最佳答案 如
你们能告诉我它们之间的区别吗? 顺便问一下,有什么叫C++库或C库的吗? 最佳答案 C++ 标准库 和 C 标准库 是 C++ 和 C 标准定义的库,提供给 C++ 和 C 程序使用。那是那些词的共同
下面的测试代码,我将输出信息放在注释中。我使用的是 gcc 4.8.5 和 Centos 7.2。 #include #include class C { public:
很难说出这里问的是什么。这个问题是含糊的、模糊的、不完整的、过于宽泛的或修辞性的,无法以目前的形式得到合理的回答。如需帮助澄清此问题以便重新打开它,visit the help center 。 已关
我的客户将使用名为 annoucement 的结构/类与客户通信。我想我会用 C++ 编写服务器。会有很多不同的类继承annoucement。我的问题是通过网络将这些类发送给客户端 我想也许我应该使用
我在 C# 中有以下函数: public Matrix ConcatDescriptors(IList> descriptors) { int cols = descriptors[0].Co
我有一个项目要编写一个函数来对某些数据执行某些操作。我可以用 C/C++ 编写代码,但我不想与雇主共享该函数的代码。相反,我只想让他有权在他自己的代码中调用该函数。是否可以?我想到了这两种方法 - 在
我使用的是编写糟糕的第 3 方 (C/C++) Api。我从托管代码(C++/CLI)中使用它。有时会出现“访问冲突错误”。这使整个应用程序崩溃。我知道我无法处理这些错误[如果指针访问非法内存位置等,
关闭。这个问题不符合Stack Overflow guidelines .它目前不接受答案。 我们不允许提问寻求书籍、工具、软件库等的推荐。您可以编辑问题,以便用事实和引用来回答。 关闭 7 年前。
已关闭。此问题不符合Stack Overflow guidelines 。目前不接受答案。 要求我们推荐或查找工具、库或最喜欢的场外资源的问题对于 Stack Overflow 来说是偏离主题的,因为
我有一些 C 代码,将使用 P/Invoke 从 C# 调用。我正在尝试为这个 C 函数定义一个 C# 等效项。 SomeData* DoSomething(); struct SomeData {
这个问题已经有答案了: Why are these constructs using pre and post-increment undefined behavior? (14 个回答) 已关闭 6
我是一名优秀的程序员,十分优秀!