我在C ++中做一个解析函数,它接受一个字符串和一个double作为参数,并返回"值"字符串。
以下是代码:
double evaluate (char * toParse, int length, double x)
{
// Case 'x'
if ((toParse[0] == 'x') &&
(length == 1))
{
return x;
}
// Case value
char * endptr;
double num = strtod(toParse, &endptr);
if(endptr - toParse == length)
{
return num;
}
// Parsing
int nBrackets = 0;
for (int i = 0; i < length; i++)
{
if (toParse[i] == '(')
{
nBrackets++;
}
else if (toParse[i] == ')')
{
nBrackets--;
}
// Remove brackets.
double _x = (toParse[0] == '(' && toParse[i-1] == ')' ) ?
evaluate(&toParse[1], i-2, x) : evaluate(toParse, i, x);
double _y = (toParse[i+1] == '(' && toParse[length-1] == ')' ) ?
evaluate(&toParse[i+2], length - (i+1) - 2, x) : evaluate (&toParse[i+1] , length - (i+1), x);
// Supports +, -, * and /
if (nBrackets == 0 &&
toParse[i] == '+')
{
return _x + _y;
}
else if (nBrackets == 0 &&
toParse[i] == '-')
{
return _x - _y;
}
else if (nBrackets == 0 &&
toParse[i] == '*')
{
return _x * _y;
}
else if (nBrackets == 0 &&
toParse[i] == '/')
{
return _x / _y;
}
}
return 0.;
}
int main()
{
cout << evaluate("((4*x)+7)-x", 11, 5.) << endl;
// Outputs 22, which sounds correct.
return 0;
}
它远没有完美无缺(对运算符没有优先权,如果字符串包含太多括号等,它不起作用),但我想删除双x参数,并继续工作功能直接。 (因为我想绘制函数,如果我不处理函数,我将不得不为x的每个值解析相同的字符串...)
有可能吗?我的意思是,做一些像:
double (double) operator+ (double f(double), double g(double))
{
double h (double x)
{
return f(x)+g(x);
}
return h;
}
但它当然没有用。有任何想法吗 ? (课等)
感谢。
答案 0 :(得分:2)
您可以使用函数指针。如果你想要一个函数指针作为参数,我也不太清楚,但是你可以这样做:
typedef double (*handler) (double);
double add(handler x, handler y);
double sub(handler x, handler y);
double func1(double n);
double func2(double n);
int main()
{
double (*funcPtr[256]) (double);
funcPtr['+'] = func1;
funcPtr['-'] = func2;
double answer = funcPtr['+'](2));
}
double func1(double n)
{
return n;
}
double func2(double n)
{
return n;
}
double add(handler x, handler y)
{
return x(2) + y(2);
}
double sub(handler x, handler y)
{
return x(4) - y(2);
}
如果它不是你想要的,但是下面的代码让我知道,我会编辑我的答案:
funcPtr[toParse[i]](2, 2); // toParse[i] is '+' it will then call add(2,2)
256个funcPtr
数组与sizeof(char)
有关。您应该确保index
是数组中的实际值,否则您将访问越界。
答案 1 :(得分:1)
想到一个lambda。不幸的是,我相信从捕获函数参数的函数返回lambda将是未定义的行为。 您也不能重载像函数指针这样的类型的构建。 你可以这样做:
#include <iostream>
#include <functional>
struct CombinedFunction{
enum class Operator{ add, subtract, multiply, divide } op;
CombinedFunction(std::function<double(double)> f1, std::function<double(double)> f2, Operator op) : f1(std::move(f1)), f2(std::move(f2)), op(op){
}
std::function<double(double)> f1, f2;
double operator()(double d){
switch (op){
case Operator::add:
return f1(d) + f2(d);
break;
case Operator::subtract:
//...
break;
//...
}
}
};
std::function<double(double)> operator+ (std::function<double(double)> f, std::function<double(double)> g){
return CombinedFunction(f, g, CombinedFunction::Operator::add);
}
double f(double x){
return 2 * x;
}
double g(double x){
return 3 * x;
}
int main(){
auto h = std::function<double(double)>(f) + g;
std::cout << h(2);
auto h2 = f + h + g;
std::cout << h2(2);
}
它不能很好地扩展,并且效率不高。也许有一个模板解决方案可以在编译时完成所有这些工作。