C - 将多个函数地址作为可变参数函数中的参数传递

时间:2015-12-18 19:21:02

标签: c function-pointers memory-address variadic-functions

我正在尝试编写一个函数,该函数将获取前n个整数和可变数量的函数,并构建一个在第一列中具有数字“i”并在其他列中具有“function(i)”的表

但我似乎无法将我的函数地址传递给表生成器,因为我遇到了访问冲突错误。我做错了什么?

#include <stdio.h>
#include <math.h>
#include <stdarg.h>

typedef float(*f)(float);

// Some examples of f-type functions.
float square(float x) { return x*x; };
float root(float x) { return sqrt(x); };
float timesPi(float x) { return x * 3.14; };

// Display a table with first colon being the numbers from 1 to n, 
// then the other columns to be f(i)
void table(unsigned int n, unsigned int nr_functions, ...)
{
    va_list func;
    va_start(func, nr_functions);

    for (float i = 1; i <= n; i += 1)
    {
        printf("\n%6.0f |", i);
        for (unsigned int j = 0; j < nr_functions; j++)
        {
            f foo = va_arg(func, f);
            printf("%6.3f |", foo(i));
        }
        va_end(func);
    }
}

// Main function
int main()
{
    table(5, 3, &square, &root, &timesPi);
    system("pause");
    return 0;
}

对于上面的示例

table(5, 3, &square, &root, &timesPi);

我想回来

1   |   1.000 |  3.140 |
2   |   1.141 |  6.280 |
3   |   1.732 |  9.420 |
4   |   2.000 | 12.560 | 
5   |   2.236 | 15.700 |

1 个答案:

答案 0 :(得分:4)

你需要重用参数列表的变量部分,这意味着你需要在正确位置的va_start()va_end() - 在外部循环中:

void table(unsigned int n, unsigned int nr_functions, ...)
{
    for (unsigned int i = 1; i <= n; i++)
    {
        va_list func;
        printf("\n%6.0f |", (double)i);
        va_start(func, nr_functions);
        for (unsigned int j = 0; j < nr_functions; j++)
        {
            f foo = va_arg(func, f);
            printf("%6.3f |", foo(i));
        }
        va_end(func);
    }
}

否则,你在列表的末尾行进,除了你在循环中调用va_end(),导致善良只知道什么是伤害。

请注意,循环应该使用整数运算 - 随后对printf()进行更改 - 这里我转换了值,但是将格式更改为%6d也是理智的(事实上可能更好) )。

使用此功能,我得到了输出:

 1 | 1.000 | 1.000 | 3.140 |
 2 | 4.000 | 1.414 | 6.280 |
 3 | 9.000 | 1.732 | 9.420 |
 4 |16.000 | 2.000 |12.560 |
 5 |25.000 | 2.236 |15.700 |