- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
我有一个函数,它接受一个函数指针作为参数,然后用它自己的参数调用该函数:
typedef int (*my_func_ptr)( int );
int foo( my_func_ptr f ) {
static int i = 0;
return i = f( i );
}
有时,我需要将函数传递给 foo
这不仅仅取决于整数输入来吐出结果。
int add_strlen( int i, const char* s ) {
return i + strlen( s );
}
我可以修改上面的代码以利用 std::function
然后使用 std::bind
, 但我更喜欢在编译时创建这些函数,所以我使用模板。
template<const char* S>
int add_strlen( int i ) {
return i + strlen( S );
}
/**
* Usage:
* char bar[] = "bar";
* foo( add_strlen<bar> );
*/
我的问题出现在使用指针作为模板参数时。每当我使用指向任何类型的常量数据的指针作为模板参数时,如果传递的参数被声明为该类型的非常量数组,它只会设法编译。
char char_array[] = "works";
const char const_char_array[] = "error";
char *char_ptr = "error";
const char *const_char_ptr = "error";
Clang (ver. 3.0-6) 中的相关错误(char_ptr
和const_char_ptr
的错误相同):
func_ptr.cpp:29:9: error: no matching function for call to 'foo'
foo( add_strlen<const_char_array> );
^~~
func_ptr.cpp:6:5: note: candidate function not viable: no overload of 'add_strlen' matching 'my_func_ptr' (aka 'int (*)(int)') for 1st argument
int foo( my_func_ptr f )
谁能给我解释一下这是为什么?我的看法,模板参数S
预计类型为 const char*
,这在任何其他情况下都意味着我可以传入任何常量或非常量指针或 char
类型的数组。并期望它起作用。我希望能够将我的数组声明为 const
,因为我什至不想暗示它们是要在运行时修改的。有什么方法可以让我的数组保持常量并将它们用作模板参数?
编辑:感谢一些帮助(以及具有更好错误的较新版本的 Clang),我能够确定提供具有内部链接的模板参数是问题的一部分。通过将上述变量声明为 extern,我可以使用 add_strlen<const_char_array>
没有错误。我还创建了一个简化的测试用例。它包含在下面:
#include <cstring>
typedef int (*my_func_ptr)( int );
int foo( my_func_ptr f ) {
static int i = 0;
return i = f( i );
}
template<const char* S>
int add_strlen( int i ) {
return i + strlen( S );
}
extern char char_array[];
extern const char const_char_array[];
extern char *char_ptr;
extern const char *const_char_ptr;
char char_array[] = "foo";
const char const_char_array[] = "bar";
// assigning to string literal is deprecated
char *char_ptr = char_array;
const char *const_char_ptr = "baz";
int main(int argc, const char *argv[])
{
foo( add_strlen<char_array> ); // works
foo( add_strlen<const_char_array> ); // works
//foo( add_strlen<char_ptr> ); // doesn't work
//foo( add_strlen<const_char_ptr> ); // doesn't work
return 0;
}
最佳答案
这个错误好像和你是什么以及不允许什么作为非类型模板参数有关,引用IBM Linux Compilers documentation for Non-type template parameters他们有话要说:
The syntax of a non-type template parameter is the same as a declaration of one of the following types:
- integral or enumeration
- pointer to object or pointer to function
- reference to object or reference to function
- pointer to member
char_array[]
和 const_char_array[]
在传入时工作的原因是因为它们在编译时是常量,并且在程序运行时永远不会在程序下改变.整数类型可以传入,但是指向整数类型的指针不能传入。
模板需要一种const char *
a.k.a const char[x]
,但它也需要一些永远不会改变的东西,所以指针指向可能永远不会改变。当在编译器时传入您的 const_char_array
时,它被传递给 char[6]
(“错误”)。位置永远不会改变,内容永远不会改变。然而,当传入 const_char_ptr
时,它得到一个 const char *
,虽然指针本身可能永远不会改变,但它指向的位置完全有可能改变。它本身不是静态的。
char *_arr = new char[20];
const char* _ptr_arr = _arr;
我们可以在这里同意我的 _ptr_arr
与您的 const_char_ptr
是完全相同的类型,但内容的存储位置可能会在运行时发生变化。在模板中这是不允许的,因为它可能需要模板的全新实例化,并且从创建模板时起就不确定。 char [6]
是静态的,不会改变。
foo( add_strlen<_ptr_arr> );
导致以下编译器错误:
test.cpp:36:5: error: no matching function for call to 'foo'
foo( add_strlen<_ptr_arr>);
^~~
test.cpp:6:5: note: candidate function not viable: no overload of 'add_strlen' matching 'my_func_ptr' (aka 'int (*)(int)') for 1st argument
int foo( my_func_ptr f ) {
^
这不是很有帮助,我们想弄清楚为什么没有有效的重载,使用函数独立编译代码而不作为函数指针传递我们得到以下内容:
add_strlen<_ptr_arr>(0);
将导致:
test.cpp:36:5: error: no matching function for call to 'add_strlen'
add_strlen<_ptr_arr>(0);
^~~~~~~~~~~~~~~~~~~~
test.cpp:16:5: note: candidate template ignored: invalid explicitly-specified argument for template parameter 'S'
int add_strlen( int i ) {
^
所以显式指定的参数是无效的,具体来说,我们不能传入一个指向整数的指针。
关于c++ - 无法将指向 const 数据的指针作为函数模板参数传递,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/15013333/
#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
我是一名优秀的程序员,十分优秀!