如果我正在使用特定功能而不检查其返回值,我希望编译器发出错误。
我在Eclipse中使用GCC。
例如 :
int fun (){ return 3;} void main () { printf("%d",fun ()); }
我调用该函数fun
并打印返回值,但不检查返回值。相反,我想强制执行以下操作:
int ret=fun(); if(ret != some_value) { /* Do something */ } printf("%d",fun ());
这可能吗?
我认为没有任何方法可以按照您想要的方式进行。毕竟,在该语句中,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()); }
但是没有办法使编译器理解此printf
语句与您的语句之间的区别。我假设您想强迫程序员将返回值保存在变量中,但是我不知道该怎么做,即使您可以这样做,也不能确保正确的检查。看看这个例子:
char *array; void *ptr; ptr = realloc(array, 20); strcpy(array, "Hello, World!");
注意我们如何保存返回值,ptr
但是由于我们没有做类似的事情,if(ptr == NULL) exit(EXIT_FAILURE);
所以没有意义。
但是,有一种方法可以防止根本不使用返回值的语句(当然,这与实际检查不一样)。
据我所知,没有便携式的方法可以做到这一点。您将不得不依赖编译器扩展。Gcc有这样的扩展。
__attribute__ ((warn_unused_result)) int foo (void) { return 5; } int main(void) { foo(); }
进行编译将生成以下警告:
$ 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(); | ^~~~~
为了将其视为错误,请使用 -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
如果您希望所有其他警告都只是警告,请使用进行编译-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
完成类似操作的一种方法是将返回值移至参数。这将强制将“返回值”保存到参数,并且您必须在不向其发送输出变量的情况下调用该函数。更改
int fun (){ return 3;}
至
void fun(int *ret) { *ret=3; }
但是如上所述,我看不到如何强制对变量进行适当的检查。这仅强制执行任务。
另一种选择是使用包装器功能。这通常会大大降低灵活性,因此在使用它之前请三思而后行,但这是一种在某些情况下可能有用的选项。假设您有一个标头/源对。将包装器的原型放入头文件中,并将包装器和函数的实现放入源文件中。这将对程序员隐藏原始功能。它看起来可能像这样:
。H
int fun_wrapper();
。C
int fun() { return 3; } int fun_wrapper() { int ret = fun(); if(ret<0) exit(EXIT_FAILURE); return ret; }
我创建了一个具有答案的相关问题。它是关于如何将此fgets
功能用于库中的以及其他功能的:如何为具有原始名称的库函数创建包装器?