C函数隐式声明要注意检查

C函数隐式声明要注意检查

 

gcc编译代码的时候(其他编译器估计类似),会碰到这个提示:

 

implicit declaration of function‘xxx’;

 

这个提示的意思是找不到该函数的原型声明。那会产生什么问题呢?我们知道,C编译器在早期是没有原型声明的要求的(C99标准之前)。如果没有原型声明,那编译器是如何知道怎么处理这些函数的参数和返回值呢?有一些约定:按照int和double两种类型来传递参数。这种做法也能支持实在是无法提供原型声明的函数使用(printf之类的变参函数)。

 

如果提供了原型声明,则编译器会按照参数类型生成准确的传参压栈代码。所以实际函数的代码是按照原型声明的方式取参数并进行相应的处理的。刚才说的约定如果和实际函数代码中的处理方法不兼容,就会出现各类不确定的问题,严重时候还会导致coredump。

 

有时候忘记添加系统的一些头文件,例如<string.h>之类的引用,就会提示这个警告,而忽略它就可能导致隐含的运行时错误。过去曾数次碰到这个问题,所以记录下来提醒自己一定要注意检查这类问题。

 

默认-Wall的时候,这个隐式的声明是警告的方式提示,容易被忽略,从而导致如果测试不完备,程序代码就会带有运行时的错误了呀。

 

所以一定要加上类似下面的参数,确保这类问题被高优先级的处理。

-Werror-implicit-function-declaration

这个参数保证会用error的级别提示隐式函数声明。

 

另外,还建议添加-Wall  -Wextra和-Werror,将各类可能的隐患都提示出来,只有你确保这些警告是风险可预知和接受的情况下,才能忽略他们。

 

我常用的编译选项是:

gcc -O0 -g3 -Wall -Wextra -Werror-Wconversion -Werror-implicit-function-declaration -Wno-unused-parameter-Wno-unused-variable … …


  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
以下是一个判断素数的函数 is_prime() 的实现,以及符合要求的主函数实现: ```c #include <stdio.h> #include <stdbool.h> bool is_prime(int num) { if (num < 2) { return false; } for (int i = 2; i < num; i++) { if (num % i == 0) { return false; } } return true; } int main() { int test_nums[5] = {17, 34, 2, 1, 0}; for (int i = 0; i < 5; i++) { if (is_prime(test_nums[i])) { printf("%d is a prime number.\n", test_nums[i]); } else { printf("%d is not a prime number.\n", test_nums[i]); } } printf("\n"); for (int i = 100; i <= 200; i++) { if (is_prime(i)) { printf("%d is a prime number.\n", i); } } return 0; } ``` 程序中,is_prime() 函数接收一个整数参数 num,并返回一个布尔值,表示 num 是否为素数。在主函数中,我们先定义了一个整型数组 test_nums,包含了 5 个测试用例,然后通过 for 循环依次调用 is_prime() 函数,并输出测试结果。接下来,我们使用 for 循环输出从 100 到 200 之间的所有素数,如果一个数是素数,就输出它的值。 在编译和运行程序之前,我们需要对主函数中使用的 is_prime() 函数声明。因此,我们可以在程序最开始处添加如下代码: ```c bool is_prime(int num); ``` 这段代码表示 is_prime() 函数声明,告诉编译器我们将在主函数中使用这个函数。接下来,我们可以进行编译和运行程序,检查其输出结果是否符合预期。 接下来,我们将主函数函数声明删掉,这样程序就没有对 is_prime() 函数声明了。在编译时,我们会收到一个警告,表示 is_prime() 函数没有被声明过,但是程序仍然可以编译通过。这是因为 C 语言允许函数隐式声明,即如果一个函数没有被声明过,但是在程序中使用了它,编译器会默认为这个函数做一个声明。虽然这个程序可以编译通过,但是这种做法是不规范的,不建议使用。 接下来,我们将主函数的位置改为在其他函数之后,在主函数中不含函数声明。这样程序在编译时会报错,提示找不到 is_prime() 函数的定义。这是因为在主函数之前没有声明 is_prime() 函数,所以编译器无法找到这个函数的定义。因此,我们需要在主函数之前添加函数声明,告诉编译器我们将在主函数中使用这个函数。 最后,我们保留判别素数的函数,修改主函数,要求实现输出 100 到 200 之间的素数。我们可以使用一个 for 循环,从 100 到 200 遍历所有整数,判断每个整数是否为素数,如果是素数,就输出它的值。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值