如果未检查函数返回值,如何强制编译错误?

joi*_*doi 3 c eclipse gcc compiler-errors return-value

如果我正在使用特定功能而不检查其返回值,我希望编译器发出错误。

我在Eclipse中使用GCC。

例如 :

int fun (){ return 3;}

void main ()
{
   printf("%d",fun ());
}
Run Code Online (Sandbox Code Playgroud)

我调用该函数fun并打印返回值,但不检查返回值。相反,我想强制执行以下操作:

int ret=fun();
if(ret != some_value) { /* Do something */ }
printf("%d",fun ());
Run Code Online (Sandbox Code Playgroud)

这可能吗?

klu*_*utt 5

您不能强制进行适当的检查

我认为没有任何方法可以按照您想要的方式进行。毕竟,在该语句中,printf("%d",fun ())您实际上是fun()通过将返回值发送到另一个函数来检查返回值。当然,printf并没有真正“检查”返回值,但是您可以这样使用它:

void exit_if_fun_returns_negative_value(int val) { if(val<0) exit(EXIT_FAILURE); }

int main(void)
{
    exit_if_fun_returns_negative_value(fun());
}
Run Code Online (Sandbox Code Playgroud)

但是没有办法使编译器理解此printf语句与您的语句之间的区别。我假设您想强迫程序员将返回值保存在变量中,但是我不知道该怎么做,即使您可以这样做,也不能确保正确的检查。看看这个例子:

char *array;
void *ptr;
ptr = realloc(array, 20);
strcpy(array, "Hello, World!");
Run Code Online (Sandbox Code Playgroud)

注意我们如何保存返回值,ptr但是由于我们没有做类似的事情,if(ptr == NULL) exit(EXIT_FAILURE);所以没有意义。

但是您可以防止完全忽略返回值

但是,有一种方法可以防止根本不使用返回值的语句(当然,这与实际检查不一样)。

据我所知,没有便携式的方法可以做到这一点。您将不得不依赖编译器扩展。Gcc有这样的扩展。

__attribute__ ((warn_unused_result)) int foo (void) 
{
    return 5;
}

int main(void)
{
    foo();
}
Run Code Online (Sandbox Code Playgroud)

进行编译将生成以下警告:

$ gcc main.c
main.c: In function ‘main’:
main.c:9:5: warning: ignoring return value of ‘foo’, declared with attribute warn_unused_result [-Wunused-result]
    9 |     foo();
      |     ^~~~~
Run Code Online (Sandbox Code Playgroud)

为了将其视为错误,请使用 -Werror

$ gcc main.c -Werror
main.c: In function ‘main’:
main.c:9:5: error: ignoring return value of ‘foo’, declared with attribute warn_unused_result [-Werror=unused-result]
    9 |     foo();
      |     ^~~~~
cc1: all warnings being treated as errors
Run Code Online (Sandbox Code Playgroud)

如果您希望所有其他警告都只是警告,请使用进行编译-Werror=unused-result。例:

$ cat main.c 
__attribute__ ((warn_unused_result))
int foo (void)
{
    return 5;
}

int main(void)
{
    int *x = 5;
    foo();
}

$ gcc main.c -Werror=unused-result
main.c: In function ‘main’:
main.c:9:14: warning: initialization of ‘int *’ from ‘int’ makes pointer from integer without a cast [-Wint-conversion]
    9 |     int *x = 5;
      |              ^
main.c:10:5: error: ignoring return value of ‘foo’, declared with attribute warn_unused_result [-Werror=unused-result]
   10 |     foo();
      |     ^~~~~
cc1: some warnings being treated as errors
Run Code Online (Sandbox Code Playgroud)

更改签名以包括输出参数

完成类似操作的一种方法是将返回值移至参数。这将强制将“返回值”保存到参数,并且您必须在不向其发送输出变量的情况下调用该函数。更改

int fun (){ return 3;} 
Run Code Online (Sandbox Code Playgroud)

void fun(int *ret) { *ret=3; }
Run Code Online (Sandbox Code Playgroud)

但是如上所述,我看不到如何强制对变量进行适当的检查。这仅强制执行任务。

包装功能

另一种选择是使用包装器功能。这通常会大大降低灵活性,因此在使用它之前请三思而后行,但这是一种在某些情况下可能有用的选项。假设您有一个标头/源对。将包装器的原型放入头文件中,并将包装器和函数的实现放入源文件中。这将对程序员隐藏原始功能。它看起来可能像这样:

。H

int fun_wrapper();
Run Code Online (Sandbox Code Playgroud)

。C

int fun() { return 3; }

int fun_wrapper() 
{ 
    int ret = fun(); 
    if(ret<0) exit(EXIT_FAILURE);
    return ret;
}
Run Code Online (Sandbox Code Playgroud)

如何将此技术用于库函数

我创建了一个具有答案的相关问题。它是关于如何将此fgets功能用于库中的以及其他功能的:如何为具有原始名称的库函数创建包装器?