使用带数组参数的函数指针的分段错误

问题描述

今天我开始学习 C 语言,但我陷入了函数指针的困境。这是我的代码

main.c:


#include <stdio.h>

int sumOfElements(int *arr,int arr_elements);

int main()
{

    int (*ptr)(int,int) = NULL;
    ptr = sumOfElements;
    int a[] = {128,64,32,16,8,4,2,1};
    printf("Total of price is: %d",ptr(a,8));

}

int sumOfElements(int *arr,int arr_elements)
{
    int k =0;
    int total;
    for(;k < arr_elements;k++)
    {
        total += arr[k];
    }
    return total;
}

我想要做的是在 sumOfElements 函数中访问数组的元素。当我正常调用它时,一切都会顺利。当我尝试使用 function pointer 时,编译器之前向我抛出了一些警告,然后是 Segmentation Fault 错误

main.c:17:9: warning: assignment from incompatible pointer type [-Wincompatible-pointer-types]                               
main.c:19:41: warning: passing argument 1 of ‘ptr’ makes integer from pointer without a cast [-Wint-conversion]              
main.c:19:41: note: expected ‘int’ but argument is of type ‘int *’    
Segmentation fault (core dumped) 

因为我还在学习它,所以我不确定是否要接触代码,因为,就像我之前说的,它在没有 function pointer 的情况下也能工作。现在,错误 main.c:17:9: warning: assignment from incompatible pointer type [-Wincompatible-pointer-types],我并没有真正理解它,因为它们都是 int。所以,如果你也能解释一下,那会节省我很多时间。那么,为什么当我使用 Segmentation Fault (core dumped) 时它只抛出 funptr?我只知道分段错误是当我们尝试访问只有 read-onlyout-of-bound

的内存时

解决方法

对于初学者来说,指针声明为

int (*ptr)(int,int) = NULL;

也就是说,它是一个指向具有两个 int 类型参数的函数的指针。

但是函数sumOfElements有不同类型的参数

int sumOfElements(int *arr,int arr_elements);

这是第一个参数的类型为 int * 而不是 int

此外,由于数组在函数内未更改,因此最好像这样声明函数

long long int sumOfElements( const int *arr,size_t arr_elements);

函数返回类型是long long int而不是int,因为它降低了数组元素总和溢出的风险。

相应地,指针应声明为

long long int (*ptr)( const int *,size_t ) = NULL;

该函数应该像

一样被调用
printf("Total of price is: %lld",ptr(a,sizeof( a ) / sizeof( *a ) ) );

在函数中您忘记初始化变量 total

int total;

可以通过以下方式定义函数

long long int sumOfElements( const int *arr,size_t arr_elements )
{
    long long int total = 0;

    while( arr_elements-- )
    {
        total += arr[arr_elements];
    }

    return total;
}
,

您的函数指针声明类型不正确。

您的指针的签名为 int (*ptr)(int,int) = NULL;,而您的函数为 int ()(int *,int)

尝试将您的指针声明为 int (*ptr)(int *,int) = NULL;

,

函数指针需要与函数本身具有相同的参数和返回类型。在您的 ptr 声明中,您将第一个参数声明为 int,但它应该是 int*

    int (*ptr)(int*,int) = NULL;
,

您必须使用 int (*ptr)(int*,int) = NULL; 而不是 int (*ptr)(int,int) = NULL;,因为您的数组是一个 int 指针。 由于您的常规函数​​ sumOfElements 已经收到一个 int 指针,它可以正常工作。