我正试图通过指针调用函数。编译这段代码,我得到错误,说void类型的表达式不能转换为其他类型,但我在调用函数之前检查函数是否返回void。还有另一种方法可以实现这一目标吗?
class FuncBase {
public:
FuncBase(string n, string ret, string arg1, string arg2): name(n), retval(ret), a1(arg1), a2(arg2) {}
string getName() const { return name; }
string getRet() const { return retval; }
string getA1() const { return a1; }
string getA2() const { return a2; }
virtual void call() = 0;
private:
string name, retval, a1, a2;
};
template <typename ret, typename arg1, typename arg2>
class Func: public FuncBase {
public:
Func(string n, string r, string ar1, string ar2, ret(*fc)(arg1, arg2)):
FuncBase(n, r, ar1, ar2), func(fc) {}
void call() {
arg1 ar1;
arg2 ar2;
cout << "You chose the " << getName() << " function" << endl;
cout << "Enter a " << getA1() << ": ";
cin >> ar1;
cout << "Enter a " << getA2() << ": ";
cin >> ar2;
if (getRet() != "void") {
ret val = (*func)(ar1, ar2);
cout << getName() << " returned " << val << endl;
}
else (*func)(ar1, ar2);
cout << endl;
}
private:
// pointer to function
ret(*func)(arg1, arg2);
};
答案 0 :(得分:4)
对于失败的模板实例化,func
是一个函数指针,它接收两个参数并具有void
返回类型。那是ret
是void
。具有void返回类型的函数不返回任何内容。当然,您无法读取(*func)()
调用的返回值,因为您说没有返回值。
即使读取返回值的分支不会被执行,它仍然需要编译并传递静态类型检查。这里的基本问题是对*func()
的调用是在编译时处理的,并且是编译时静态类型检查。
这个Stack Overflow问题涵盖了与您完全相同的问题,并且接受的答案显示了如何处理问题
答案 1 :(得分:1)
我认为'arg1','arg2'和'ret'都是模板参数。鉴于此,即使代码将运行时路径转到(*func)(ar1, ar2)
,编译器仍然必须编译ret val = (*func)(ar1, ar2);
,这对返回void的函数没有意义。我认为这会给你错误。您能为我们提供更多代码吗?
编辑:好了,现在我看到你要做的事情,你需要这样的东西来执行void和non-void函数,每个函数都有不同的行为:
int foo(){ return 5; }
void bar(){}
struct executor
{
template <typename T>
void exec( T (*func)())
{
T ret = (*func)();
cout << "Non-void function called, returned " << ret << endl;
}
void exec( void (*func)())
{
(*func)();
cout << "void function called" << endl;
}
};
int main()
{
executor ex;
ex.exec(foo);
ex.exec(bar);
return 0;
}